viz.go 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. /*
  2. Copyright 2023 Docker Compose CLI authors
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package compose
  14. import (
  15. "context"
  16. "strconv"
  17. "strings"
  18. "github.com/compose-spec/compose-go/v2/types"
  19. "github.com/docker/compose/v2/pkg/api"
  20. )
  21. // maps a service with the services it depends on
  22. type vizGraph map[*types.ServiceConfig][]*types.ServiceConfig
  23. func (s *composeService) Viz(_ context.Context, project *types.Project, opts api.VizOptions) (string, error) {
  24. graph := make(vizGraph)
  25. for _, service := range project.Services {
  26. graph[&service] = make([]*types.ServiceConfig, 0, len(service.DependsOn))
  27. for dependencyName := range service.DependsOn {
  28. // no error should be returned since dependencyName should exist
  29. dependency, _ := project.GetService(dependencyName)
  30. graph[&service] = append(graph[&service], &dependency)
  31. }
  32. }
  33. // build graphviz graph
  34. var graphBuilder strings.Builder
  35. // graph name
  36. graphBuilder.WriteString("digraph ")
  37. writeQuoted(&graphBuilder, project.Name)
  38. graphBuilder.WriteString(" {\n")
  39. // graph layout
  40. // dot is the perfect layout for this use case since graph is directed and hierarchical
  41. graphBuilder.WriteString(opts.Indentation + "layout=dot;\n")
  42. addNodes(&graphBuilder, graph, project.Name, &opts)
  43. graphBuilder.WriteByte('\n')
  44. addEdges(&graphBuilder, graph, &opts)
  45. graphBuilder.WriteString("}\n")
  46. return graphBuilder.String(), nil
  47. }
  48. // addNodes adds the corresponding graphviz representation of all the nodes in the given graph to the graphBuilder
  49. // returns the same graphBuilder
  50. func addNodes(graphBuilder *strings.Builder, graph vizGraph, projectName string, opts *api.VizOptions) *strings.Builder {
  51. for serviceNode := range graph {
  52. // write:
  53. // "service name" [style="filled" label<<font point-size="15">service name</font>
  54. graphBuilder.WriteString(opts.Indentation)
  55. writeQuoted(graphBuilder, serviceNode.Name)
  56. graphBuilder.WriteString(" [style=\"filled\" label=<<font point-size=\"15\">")
  57. graphBuilder.WriteString(serviceNode.Name)
  58. graphBuilder.WriteString("</font>")
  59. if opts.IncludeNetworks && len(serviceNode.Networks) > 0 {
  60. graphBuilder.WriteString("<font point-size=\"10\">")
  61. graphBuilder.WriteString("<br/><br/><b>Networks:</b>")
  62. for _, networkName := range serviceNode.NetworksByPriority() {
  63. graphBuilder.WriteString("<br/>")
  64. graphBuilder.WriteString(networkName)
  65. }
  66. graphBuilder.WriteString("</font>")
  67. }
  68. if opts.IncludePorts && len(serviceNode.Ports) > 0 {
  69. graphBuilder.WriteString("<font point-size=\"10\">")
  70. graphBuilder.WriteString("<br/><br/><b>Ports:</b>")
  71. for _, portConfig := range serviceNode.Ports {
  72. graphBuilder.WriteString("<br/>")
  73. if portConfig.HostIP != "" {
  74. graphBuilder.WriteString(portConfig.HostIP)
  75. graphBuilder.WriteByte(':')
  76. }
  77. graphBuilder.WriteString(portConfig.Published)
  78. graphBuilder.WriteByte(':')
  79. graphBuilder.WriteString(strconv.Itoa(int(portConfig.Target)))
  80. graphBuilder.WriteString(" (")
  81. graphBuilder.WriteString(portConfig.Protocol)
  82. graphBuilder.WriteString(", ")
  83. graphBuilder.WriteString(portConfig.Mode)
  84. graphBuilder.WriteString(")")
  85. }
  86. graphBuilder.WriteString("</font>")
  87. }
  88. if opts.IncludeImageName {
  89. graphBuilder.WriteString("<font point-size=\"10\">")
  90. graphBuilder.WriteString("<br/><br/><b>Image:</b><br/>")
  91. graphBuilder.WriteString(api.GetImageNameOrDefault(*serviceNode, projectName))
  92. graphBuilder.WriteString("</font>")
  93. }
  94. graphBuilder.WriteString(">];\n")
  95. }
  96. return graphBuilder
  97. }
  98. // addEdges adds the corresponding graphviz representation of all edges in the given graph to the graphBuilder
  99. // returns the same graphBuilder
  100. func addEdges(graphBuilder *strings.Builder, graph vizGraph, opts *api.VizOptions) *strings.Builder {
  101. for parent, children := range graph {
  102. for _, child := range children {
  103. graphBuilder.WriteString(opts.Indentation)
  104. writeQuoted(graphBuilder, parent.Name)
  105. graphBuilder.WriteString(" -> ")
  106. writeQuoted(graphBuilder, child.Name)
  107. graphBuilder.WriteString(";\n")
  108. }
  109. }
  110. return graphBuilder
  111. }
  112. // writeQuoted writes "str" to builder
  113. func writeQuoted(builder *strings.Builder, str string) {
  114. builder.WriteByte('"')
  115. builder.WriteString(str)
  116. builder.WriteByte('"')
  117. }