// Copyright 2018 jsonnet-bundler 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 main import ( "context" "encoding/json" "fmt" "io/ioutil" "net/url" "os" "path" "path/filepath" "regexp" "github.com/jsonnet-bundler/jsonnet-bundler/pkg" "github.com/jsonnet-bundler/jsonnet-bundler/spec" "github.com/pkg/errors" "gopkg.in/alecthomas/kingpin.v2" ) const ( installActionName = "install" updateActionName = "update" initActionName = "init" basePath = ".jsonnetpkg" srcDirName = "src" ) var ( availableSubcommands = []string{ initActionName, installActionName, } gitSSHRegex = regexp.MustCompile("git\\+ssh://git@([^:]+):([^/]+)/([^/]+).git") gitSSHWithVersionRegex = regexp.MustCompile("git\\+ssh://git@([^:]+):([^/]+)/([^/]+).git@(.*)") gitSSHWithPathRegex = regexp.MustCompile("git\\+ssh://git@([^:]+):([^/]+)/([^/]+).git/(.*)") gitSSHWithPathAndVersionRegex = regexp.MustCompile("git\\+ssh://git@([^:]+):([^/]+)/([^/]+).git/(.*)@(.*)") githubSlugRegex = regexp.MustCompile("github.com/([-_a-zA-Z0-9]+)/([-_a-zA-Z0-9]+)") githubSlugWithVersionRegex = regexp.MustCompile("github.com/([-_a-zA-Z0-9]+)/([-_a-zA-Z0-9]+)@(.*)") githubSlugWithPathRegex = regexp.MustCompile("github.com/([-_a-zA-Z0-9]+)/([-_a-zA-Z0-9]+)/(.*)") githubSlugWithPathAndVersionRegex = regexp.MustCompile("github.com/([-_a-zA-Z0-9]+)/([-_a-zA-Z0-9]+)/(.*)@(.*)") ) func main() { os.Exit(Main()) } func Main() int { cfg := struct { JsonnetHome string }{} a := kingpin.New(filepath.Base(os.Args[0]), "A jsonnet package manager") a.HelpFlag.Short('h') a.Flag("jsonnetpkg-home", "The directory used to cache packages in."). Default("vendor").StringVar(&cfg.JsonnetHome) initCmd := a.Command(initActionName, "Initialize a new empty jsonnetfile") installCmd := a.Command(installActionName, "Install all dependencies or install specific ones") installCmdURLs := installCmd.Arg("packages", "URLs to package to install").URLList() updateCmd := a.Command(updateActionName, "Update all dependencies.") command, err := a.Parse(os.Args[1:]) if err != nil { fmt.Fprintln(os.Stderr, errors.Wrapf(err, "Error parsing commandline arguments")) a.Usage(os.Args[1:]) return 2 } workdir, err := os.Getwd() if err != nil { return 1 } switch command { case initCmd.FullCommand(): return initCommand(workdir) case installCmd.FullCommand(): return installCommand(workdir, cfg.JsonnetHome, *installCmdURLs...) case updateCmd.FullCommand(): return updateCommand(cfg.JsonnetHome) default: installCommand(workdir, cfg.JsonnetHome) } return 0 } func parseDepedency(urlString string) *spec.Dependency { if spec := parseGitSSHDependency(urlString); spec != nil { return spec } if spec := parseGithubDependency(urlString); spec != nil { return spec } return nil } func parseGitSSHDependency(urlString string) *spec.Dependency { if !gitSSHRegex.MatchString(urlString) { return nil } subdir := "" host := "" org := "" repo := "" version := "master" if gitSSHWithPathAndVersionRegex.MatchString(urlString) { matches := gitSSHWithPathAndVersionRegex.FindStringSubmatch(urlString) host = matches[1] org = matches[2] repo = matches[3] subdir = matches[4] version = matches[5] } else if gitSSHWithPathRegex.MatchString(urlString) { matches := gitSSHWithPathRegex.FindStringSubmatch(urlString) host = matches[1] org = matches[2] repo = matches[3] subdir = matches[4] } else if gitSSHWithVersionRegex.MatchString(urlString) { matches := gitSSHWithVersionRegex.FindStringSubmatch(urlString) host = matches[1] org = matches[2] repo = matches[3] version = matches[4] } else { matches := gitSSHRegex.FindStringSubmatch(urlString) host = matches[1] org = matches[2] repo = matches[3] } return &spec.Dependency{ Name: repo, Source: spec.Source{ GitSource: &spec.GitSource{ Remote: fmt.Sprintf("git@%s:%s/%s", host, org, repo), Subdir: subdir, }, }, Version: version, } } func parseGithubDependency(urlString string) *spec.Dependency { if !githubSlugRegex.MatchString(urlString) { return nil } name := "" user := "" repo := "" subdir := "" version := "master" if githubSlugWithPathRegex.MatchString(urlString) { if githubSlugWithPathAndVersionRegex.MatchString(urlString) { matches := githubSlugWithPathAndVersionRegex.FindStringSubmatch(urlString) user = matches[1] repo = matches[2] subdir = matches[3] version = matches[4] name = path.Base(subdir) } else { matches := githubSlugWithPathRegex.FindStringSubmatch(urlString) user = matches[1] repo = matches[2] subdir = matches[3] name = path.Base(subdir) } } else { if githubSlugWithVersionRegex.MatchString(urlString) { matches := githubSlugWithVersionRegex.FindStringSubmatch(urlString) user = matches[1] repo = matches[2] name = repo version = matches[3] } else { matches := githubSlugRegex.FindStringSubmatch(urlString) user = matches[1] repo = matches[2] name = repo } } return &spec.Dependency{ Name: name, Source: spec.Source{ GitSource: &spec.GitSource{ Remote: fmt.Sprintf("https://github.com/%s/%s", user, repo), Subdir: subdir, }, }, Version: version, } } func updateCommand(jsonnetHome string, urls ...*url.URL) int { jsonnetfile := pkg.JsonnetFile m, err := pkg.LoadJsonnetfile(jsonnetfile) if err != nil { kingpin.Fatalf("failed to load jsonnetfile: %v", err) return 1 } err = os.MkdirAll(jsonnetHome, os.ModePerm) if err != nil { kingpin.Fatalf("failed to create jsonnet home path: %v", err) return 3 } // When updating, the lockfile is explicitly ignored. isLock := false lock, err := pkg.Install(context.TODO(), isLock, jsonnetfile, m, jsonnetHome) if err != nil { kingpin.Fatalf("failed to install: %v", err) return 3 } b, err := json.MarshalIndent(lock, "", " ") if err != nil { kingpin.Fatalf("failed to encode jsonnet file: %v", err) return 3 } b = append(b, []byte("\n")...) err = ioutil.WriteFile(pkg.JsonnetLockFile, b, 0644) if err != nil { kingpin.Fatalf("failed to write lock file: %v", err) return 3 } return 0 }