viz.go 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  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. service := service
  27. graph[&service] = make([]*types.ServiceConfig, 0, len(service.DependsOn))
  28. for dependencyName := range service.DependsOn {
  29. // no error should be returned since dependencyName should exist
  30. dependency, _ := project.GetService(dependencyName)
  31. graph[&service] = append(graph[&service], &dependency)
  32. }
  33. }
  34. // build graphviz graph
  35. var graphBuilder strings.Builder
  36. // graph name
  37. graphBuilder.WriteString("digraph ")
  38. writeQuoted(&graphBuilder, project.Name)
  39. graphBuilder.WriteString(" {\n")
  40. // graph layout
  41. // dot is the perfect layout for this use case since graph is directed and hierarchical
  42. graphBuilder.WriteString(opts.Indentation + "layout=dot;\n")
  43. addNodes(&graphBuilder, graph, project.Name, &opts)
  44. graphBuilder.WriteByte('\n')
  45. addEdges(&graphBuilder, graph, &opts)
  46. graphBuilder.WriteString("}\n")
  47. return graphBuilder.String(), nil
  48. }
  49. // addNodes adds the corresponding graphviz representation of all the nodes in the given graph to the graphBuilder
  50. // returns the same graphBuilder
  51. func addNodes(graphBuilder *strings.Builder, graph vizGraph, projectName string, opts *api.VizOptions) *strings.Builder {
  52. for serviceNode := range graph {
  53. // write:
  54. // "service name" [style="filled" label<<font point-size="15">service name</font>
  55. graphBuilder.WriteString(opts.Indentation)
  56. writeQuoted(graphBuilder, serviceNode.Name)
  57. graphBuilder.WriteString(" [style=\"filled\" label=<<font point-size=\"15\">")
  58. graphBuilder.WriteString(serviceNode.Name)
  59. graphBuilder.WriteString("</font>")
  60. if opts.IncludeNetworks && len(serviceNode.Networks) > 0 {
  61. graphBuilder.WriteString("<font point-size=\"10\">")
  62. graphBuilder.WriteString("<br/><br/><b>Networks:</b>")
  63. for _, networkName := range serviceNode.NetworksByPriority() {
  64. graphBuilder.WriteString("<br/>")
  65. graphBuilder.WriteString(networkName)
  66. }
  67. graphBuilder.WriteString("</font>")
  68. }
  69. if opts.IncludePorts && len(serviceNode.Ports) > 0 {
  70. graphBuilder.WriteString("<font point-size=\"10\">")
  71. graphBuilder.WriteString("<br/><br/><b>Ports:</b>")
  72. for _, portConfig := range serviceNode.Ports {
  73. graphBuilder.WriteString("<br/>")
  74. if len(portConfig.HostIP) > 0 {
  75. graphBuilder.WriteString(portConfig.HostIP)
  76. graphBuilder.WriteByte(':')
  77. }
  78. graphBuilder.WriteString(portConfig.Published)
  79. graphBuilder.WriteByte(':')
  80. graphBuilder.WriteString(strconv.Itoa(int(portConfig.Target)))
  81. graphBuilder.WriteString(" (")
  82. graphBuilder.WriteString(portConfig.Protocol)
  83. graphBuilder.WriteString(", ")
  84. graphBuilder.WriteString(portConfig.Mode)
  85. graphBuilder.WriteString(")")
  86. }
  87. graphBuilder.WriteString("</font>")
  88. }
  89. if opts.IncludeImageName {
  90. graphBuilder.WriteString("<font point-size=\"10\">")
  91. graphBuilder.WriteString("<br/><br/><b>Image:</b><br/>")
  92. graphBuilder.WriteString(api.GetImageNameOrDefault(*serviceNode, projectName))
  93. graphBuilder.WriteString("</font>")
  94. }
  95. graphBuilder.WriteString(">];\n")
  96. }
  97. return graphBuilder
  98. }
  99. // addEdges adds the corresponding graphviz representation of all edges in the given graph to the graphBuilder
  100. // returns the same graphBuilder
  101. func addEdges(graphBuilder *strings.Builder, graph vizGraph, opts *api.VizOptions) *strings.Builder {
  102. for parent, children := range graph {
  103. for _, child := range children {
  104. graphBuilder.WriteString(opts.Indentation)
  105. writeQuoted(graphBuilder, parent.Name)
  106. graphBuilder.WriteString(" -> ")
  107. writeQuoted(graphBuilder, child.Name)
  108. graphBuilder.WriteString(";\n")
  109. }
  110. }
  111. return graphBuilder
  112. }
  113. // writeQuoted writes "str" to builder
  114. func writeQuoted(builder *strings.Builder, str string) {
  115. builder.WriteByte('"')
  116. builder.WriteString(str)
  117. builder.WriteByte('"')
  118. }