gomod2nix/cmd/root.go
adisbladis 1ffea526a0 Add package generation for non development packages
This makes it possible to generate packages that you do not have a
local checkout for, e.g. running:
`gomod2nix generate --outdir example/ golang.org/x/tools/cmd/stringer`

This will be useful for packaging dependencies that you are not
developing, but just simply packaging.
2022-06-14 05:15:43 +08:00

177 lines
3.9 KiB
Go

package cmd
import (
"fmt"
"io"
"os"
"path/filepath"
log "github.com/sirupsen/logrus"
"github.com/spf13/cobra"
generate "github.com/tweag/gomod2nix/generate"
schema "github.com/tweag/gomod2nix/schema"
)
const directoryDefault = "./"
var (
flagDirectory string
flagOutDir string
maxJobs int
)
func generateFunc(cmd *cobra.Command, args []string) {
directory := flagDirectory
outDir := flagOutDir
// If we are dealing with a project packaged by passing packages on the command line
// we need to create a temporary project.
var tmpProj *generate.TempProject
if len(args) > 0 {
var err error
if directory != directoryDefault {
panic(fmt.Errorf("directory flag not supported together with import arguments"))
}
if outDir == "" {
pwd, err := os.Getwd()
if err != nil {
panic(err)
}
outDir = pwd
}
tmpProj, err = generate.NewTempProject(args)
if err != nil {
panic(err)
}
defer func() {
err := tmpProj.Remove()
if err != nil {
panic(err)
}
}()
directory = tmpProj.Dir
} else if outDir == "" {
// Default out to current working directory if we are developing some software in the current repo.
outDir = directory
}
// Write gomod2nix.toml
{
goMod2NixPath := filepath.Join(outDir, "gomod2nix.toml")
outFile := goMod2NixPath
pkgs, err := generate.GeneratePkgs(directory, goMod2NixPath, maxJobs)
if err != nil {
panic(fmt.Errorf("error generating pkgs: %v", err))
}
var goPackagePath string
var install []string
if tmpProj != nil {
install = tmpProj.Install
goPackagePath = tmpProj.GoPackagePath
}
output, err := schema.Marshal(pkgs, goPackagePath, install)
if err != nil {
panic(fmt.Errorf("error marshaling output: %v", err))
}
err = os.WriteFile(outFile, output, 0644)
if err != nil {
panic(fmt.Errorf("error writing file: %v", err))
}
log.Info(fmt.Sprintf("Wrote: %s", outFile))
}
// If we are dealing with a project packaged by passing packages on the command line, copy go.mod
if tmpProj != nil {
outMod := filepath.Join(outDir, "go.mod")
fin, err := os.Open(filepath.Join(tmpProj.Dir, "go.mod"))
if err != nil {
panic(fmt.Errorf("error opening go.mod: %v", err))
}
fout, err := os.Create(outMod)
if err != nil {
panic(fmt.Errorf("error creating go.mod: %v", err))
}
_, err = io.Copy(fout, fin)
if err != nil {
panic(fmt.Errorf("error writing go.mod: %v", err))
}
log.Info(fmt.Sprintf("Wrote: %s", outMod))
}
}
var rootCmd = &cobra.Command{
Use: "gomod2nix",
Short: "Convert applications using Go modules -> Nix",
Run: generateFunc,
}
var generateCmd = &cobra.Command{
Use: "generate",
Short: "Run gomod2nix.toml generator",
Run: generateFunc,
// func(cmd *cobra.Command, args []string) error {
// outDir := flagOutDir
// dir := flagDirectory
// if len(args) > 0 {
// tmpProj, err := generate.NewTempProject(args)
// if err != nil {
// panic(err)
// }
// defer tmpProj.Remove()
// dir = tmpProj.Dir
// outDir = "./tmp"
// }
// // dir := "/home/adisbladis/go/pkg/mod/github.com/kyleconroy/sqlc@v1.14.0"
// // fmt.Println(args)
// err := generateInternal(dir, outDir, flagMaxJobs)
// if err != nil {
// panic(err)
// }
// return nil
// },
}
var importCmd = &cobra.Command{
Use: "import",
Short: "Import Go sources into the Nix store",
Run: func(cmd *cobra.Command, args []string) {
err := generate.ImportPkgs(flagDirectory, maxJobs)
if err != nil {
panic(err)
}
},
}
func init() {
rootCmd.PersistentFlags().StringVar(&flagDirectory, "dir", "./", "Go project directory")
rootCmd.PersistentFlags().StringVar(&flagOutDir, "outdir", "", "Output directory (defaults to project directory)")
rootCmd.PersistentFlags().IntVar(&maxJobs, "jobs", 10, "Max number of concurrent jobs")
rootCmd.AddCommand(generateCmd)
rootCmd.AddCommand(importCmd)
}
func Execute() {
err := rootCmd.Execute()
if err != nil {
os.Exit(1)
}
}