This repository has been archived by the owner on Jan 4, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 4
/
go.go
93 lines (79 loc) · 2.06 KB
/
go.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
package gen
import (
"errors"
"fmt"
"io/fs"
"os"
"os/exec"
"path"
"path/filepath"
"strings"
"github.com/loft-orbital/cuebe/pkg/modfile"
)
type void struct{}
var null void
func GenGoPkg(root string) error {
pkgs, err := CollectGoPkg(root)
if err != nil {
return fmt.Errorf("could not collect go pkg: %w", err)
}
excmd := func(cmd string, args ...string) error {
c := exec.Command(cmd, args...)
c.Dir = root
c.Stdout = os.Stdout
c.Stderr = os.Stderr
return c.Run()
}
if _, err := os.Stat(path.Join(root, "go.mod")); errors.Is(err, os.ErrNotExist) {
// go.mod file dos not exist, let's create it
mf, err := modfile.Parse(path.Join(root, modfile.CueModFile))
if err != nil {
return fmt.Errorf("cannot get cue mod file: %w", err)
}
if err := excmd("go", "mod", "init", mf.Module); err != nil {
return fmt.Errorf("go mod init failed: %w", err)
}
}
// go get
args := append([]string{"get", "-d"}, pkgs...)
if err := excmd("go", args...); err != nil {
return fmt.Errorf("failed to download packages: %w", err)
}
// cue get go
args = make([]string, 0, len(pkgs)+2)
args = append(args, "get", "go")
for _, p := range pkgs {
args = append(args, strings.SplitN(p, "@", 2)[0])
}
if err := excmd("cue", args...); err != nil {
return fmt.Errorf("failed to generate cue definitions: %w", err)
}
return nil
}
func CollectGoPkg(root string) ([]string, error) {
dedupe := make(map[string]void)
gendir := path.Join("cue.mod", "gen")
err := filepath.WalkDir(root, func(dpath string, d fs.DirEntry, err error) error {
if err != nil {
return err
}
if d.IsDir() && strings.HasSuffix(dpath, gendir) {
// skip gen dir
return fs.SkipDir
} else if d.Type().IsRegular() && strings.HasSuffix(dpath, modfile.CueModFile) {
mf, err := modfile.Parse(dpath)
if err != nil {
return fmt.Errorf("failed to parse %s: %w", dpath, err)
}
for _, pkg := range mf.GoDefinitions {
dedupe[pkg] = null
}
}
return nil
})
pkgs := make([]string, 0, len(dedupe))
for k := range dedupe {
pkgs = append(pkgs, k)
}
return pkgs, err
}