mirror of https://gitlab.com/gitlab-org/cli.git
117 lines
2.7 KiB
Go
117 lines
2.7 KiB
Go
package lint
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"os"
|
|
|
|
"gitlab.com/gitlab-org/cli/api"
|
|
"gitlab.com/gitlab-org/cli/commands/cmdutils"
|
|
"gitlab.com/gitlab-org/cli/pkg/git"
|
|
|
|
"github.com/MakeNowJust/heredoc"
|
|
"github.com/spf13/cobra"
|
|
)
|
|
|
|
var (
|
|
ref string
|
|
dryRun bool
|
|
includeJobs bool
|
|
)
|
|
|
|
func NewCmdLint(f *cmdutils.Factory) *cobra.Command {
|
|
pipelineCILintCmd := &cobra.Command{
|
|
Use: "lint",
|
|
Short: "Checks if your `.gitlab-ci.yml` file is valid.",
|
|
Args: cobra.MaximumNArgs(1),
|
|
Example: heredoc.Doc(`
|
|
$ glab ci lint
|
|
# Uses .gitlab-ci.yml in the current directory
|
|
|
|
$ glab ci lint .gitlab-ci.yml
|
|
|
|
$ glab ci lint path/to/.gitlab-ci.yml
|
|
`),
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
path := ".gitlab-ci.yml"
|
|
if len(args) == 1 {
|
|
path = args[0]
|
|
}
|
|
return lintRun(f, path)
|
|
},
|
|
}
|
|
|
|
pipelineCILintCmd.Flags().BoolVarP(&dryRun, "dry-run", "", false, "Run pipeline creation simulation.")
|
|
pipelineCILintCmd.Flags().BoolVarP(&includeJobs, "include-jobs", "", false, "The response should include the list of jobs that would exist in a static check or pipeline simulation.")
|
|
pipelineCILintCmd.Flags().StringVar(&ref, "ref", "", "When dry-run is true, sets the branch or tag context for validating the CI/CD YAML configuration.")
|
|
|
|
return pipelineCILintCmd
|
|
}
|
|
|
|
func lintRun(f *cmdutils.Factory, path string) error {
|
|
var err error
|
|
out := f.IO.StdOut
|
|
c := f.IO.Color()
|
|
|
|
apiClient, err := f.HttpClient()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
repo, err := f.BaseRepo()
|
|
if err != nil {
|
|
return fmt.Errorf("You need to be in a GitLab project repository for this action.\nError: %s", err)
|
|
}
|
|
|
|
project, err := repo.Project(apiClient)
|
|
if err != nil {
|
|
return fmt.Errorf("You need to be in a GitLab project repository for this action.\nError: %s", err)
|
|
}
|
|
|
|
projectID := project.ID
|
|
|
|
var content []byte
|
|
var stdout bytes.Buffer
|
|
|
|
if git.IsValidURL(path) {
|
|
resp, err := http.Get(path)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, err = io.Copy(&stdout, resp.Body)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
content = stdout.Bytes()
|
|
} else {
|
|
content, err = os.ReadFile(path)
|
|
if err != nil {
|
|
if os.IsNotExist(err) {
|
|
return fmt.Errorf("%s: no such file or directory", path)
|
|
}
|
|
return err
|
|
}
|
|
}
|
|
|
|
fmt.Fprintln(f.IO.StdOut, "Validating...")
|
|
|
|
lint, err := api.ProjectNamespaceLint(apiClient, projectID, string(content), ref, dryRun, includeJobs)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if !lint.Valid {
|
|
fmt.Fprintln(out, c.Red(path+" is invalid"))
|
|
for i, err := range lint.Errors {
|
|
i++
|
|
fmt.Fprintln(out, i, err)
|
|
}
|
|
// Returning the error to cobra here causes the process to exit with code 0, so we exit manually.
|
|
os.Exit(1)
|
|
}
|
|
fmt.Fprintln(out, c.GreenCheck(), "CI/CD YAML is valid!")
|
|
return nil
|
|
}
|