| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596 | /*   Copyright 2023 Docker Compose CLI authors   Licensed under the Apache License, Version 2.0 (the "License");   you may not use this file except in compliance with the License.   You may obtain a copy of the License at       http://www.apache.org/licenses/LICENSE-2.0   Unless required by applicable law or agreed to in writing, software   distributed under the License is distributed on an "AS IS" BASIS,   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.   See the License for the specific language governing permissions and   limitations under the License.*/package composeimport (	"context"	"fmt"	"os"	"strings"	"github.com/docker/compose/v2/pkg/api"	"github.com/spf13/cobra")type vizOptions struct {	*ProjectOptions	includeNetworks  bool	includePorts     bool	includeImageName bool	indentationStr   string}func vizCommand(p *ProjectOptions, backend api.Service) *cobra.Command {	opts := vizOptions{		ProjectOptions: p,	}	var indentationSize int	var useSpaces bool	cmd := &cobra.Command{		Use:   "viz [OPTIONS]",		Short: "EXPERIMENTAL - Generate a graphviz graph from your compose file",		PreRunE: Adapt(func(ctx context.Context, args []string) error {			var err error			opts.indentationStr, err = preferredIndentationStr(indentationSize, useSpaces)			return err		}),		RunE: Adapt(func(ctx context.Context, args []string) error {			return runViz(ctx, backend, &opts)		}),	}	cmd.Flags().BoolVar(&opts.includePorts, "ports", false, "Include service's exposed ports in output graph")	cmd.Flags().BoolVar(&opts.includeNetworks, "networks", false, "Include service's attached networks in output graph")	cmd.Flags().BoolVar(&opts.includeImageName, "image", false, "Include service's image name in output graph")	cmd.Flags().IntVar(&indentationSize, "indentation-size", 1, "Number of tabs or spaces to use for indentation")	cmd.Flags().BoolVar(&useSpaces, "spaces", false, "If given, space character ' ' will be used to indent,\notherwise tab character '\\t' will be used")	return cmd}func runViz(ctx context.Context, backend api.Service, opts *vizOptions) error {	_, _ = fmt.Fprintln(os.Stderr, "viz command is EXPERIMENTAL")	project, err := opts.ToProject(nil)	if err != nil {		return err	}	// build graph	graphStr, _ := backend.Viz(ctx, project, api.VizOptions{		IncludeNetworks:  opts.includeNetworks,		IncludePorts:     opts.includePorts,		IncludeImageName: opts.includeImageName,		Indentation:      opts.indentationStr,	})	fmt.Println(graphStr)	return nil}// preferredIndentationStr returns a single string given the indentation preferencefunc preferredIndentationStr(size int, useSpace bool) (string, error) {	if size < 0 {		return "", fmt.Errorf("invalid indentation size: %d", size)	}	indentationStr := "\t"	if useSpace {		indentationStr = " "	}	return strings.Repeat(indentationStr, size), nil}
 |