Skip to content

Commit 6c7d1e1

Browse files
authored
feat: Handle symlinks, support pnpm and monorepo, (#89)
* feat: Handle symlinks, support pnpm and monorepo, ref: #84 * chore: suggested changes
1 parent c92c3a2 commit 6c7d1e1

12 files changed

+272
-6
lines changed

.gitignore

+2-1
Original file line numberDiff line numberDiff line change
@@ -7,4 +7,5 @@ app-builder-bin/readme.md
77
app-builder-bin/**/app-builder*
88
app-builder-bin/win/**/app-builder.exe
99

10-
/.idea/shelf/
10+
/.idea/shelf/
11+
node_modules/

go.mod

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
module github.com/develar/app-builder
22

3-
go 1.17
3+
go 1.18
44

55
require (
66
github.com/aclements/go-rabin v0.0.0-20170911142644-d0b643ea1a4c
@@ -46,11 +46,14 @@ require (
4646
require (
4747
github.com/hpcloud/tail v1.0.0 // indirect
4848
github.com/jmespath/go-jmespath v0.4.0 // indirect
49+
github.com/samber/lo v1.38.1
4950
golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd // indirect
5051
golang.org/x/text v0.3.7 // indirect
5152
gopkg.in/fsnotify.v1 v1.4.7 // indirect
5253
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 // indirect
5354
gopkg.in/yaml.v2 v2.2.8 // indirect
5455
)
5556

57+
require golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17 // indirect
58+
5659
//replace github.com/develar/go-pkcs12 => ../go-pkcs12

go.sum

+6
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,8 @@ github.com/dustin/go-humanize v1.0.0 h1:VSnTsYCnlFHaM2/igO1h6X3HA71jcobQuxemgkq4
3131
github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk=
3232
github.com/fsnotify/fsnotify v1.4.7 h1:IXs+QLmnXW2CcXuY+8Mzv/fWEsPGWxqefPtCP5CnV9I=
3333
github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
34+
github.com/go-bindata/go-bindata v1.0.0 h1:DZ34txDXWn1DyWa+vQf7V9ANc2ILTtrEjtlsdJRF26M=
35+
github.com/go-bindata/go-bindata v1.0.0/go.mod h1:xK8Dsgwmeed+BBsSy2XTopBn/8uK2HWuGSnA11C3Joo=
3436
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
3537
github.com/golang/protobuf v1.3.2 h1:6nsPYzhq5kReh6QImI3k5qWzO4PEbvbIW2cwSfR/6xs=
3638
github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
@@ -82,6 +84,8 @@ github.com/pkg/xattr v0.4.6 h1:0vqthLIMxQKA9VscyMcxjvAUGvyfzlk009vwLE8OZJg=
8284
github.com/pkg/xattr v0.4.6/go.mod h1:sBD3RAqlr8Q+RC3FutZcikpT8nyDrIEEBw2J744gVWs=
8385
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
8486
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
87+
github.com/samber/lo v1.38.1 h1:j2XEAqXKb09Am4ebOg31SpvzUTTs6EN3VfgeLUhPdXM=
88+
github.com/samber/lo v1.38.1/go.mod h1:+m/ZKRl6ClXCE2Lgf3MsQlWfh4bn1bz6CXEOxnEXnEA=
8589
github.com/segmentio/ksuid v1.0.4 h1:sBo2BdShXjmcugAMwjugoGUdUV0pcxY5mW4xKRn3v4c=
8690
github.com/segmentio/ksuid v1.0.4/go.mod h1:/XUiZBD3kVx5SmUOl55voK5yeAbBNNIed+2O73XgrPE=
8791
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
@@ -104,6 +108,8 @@ go.uber.org/zap v1.21.0 h1:WefMeulhovoZ2sYXz7st6K0sLj7bBhpiFaud4r4zST8=
104108
go.uber.org/zap v1.21.0/go.mod h1:wjWOCqI0f2ZZrJF/UufIOkiC8ii6tm1iqIsLo76RfJw=
105109
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
106110
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
111+
golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17 h1:3MTrJm4PyNL9NBqvYDSj3DHl46qQakyfqfWo4jgfaEM=
112+
golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17/go.mod h1:lgLbSvA5ygNOMpwM/9anMpWVlVJ7Z+cHWq/eFuinpGE=
107113
golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0=
108114
golang.org/x/image v0.0.0-20220302094943-723b81ca9867 h1:TcHcE0vrmgzNH1v3ppjcMGbhG5+9fMuvOmUYwNEF4q4=
109115
golang.org/x/image v0.0.0-20220302094943-723b81ca9867/go.mod h1:023OzeP/+EPmXeapQh35lcL3II3LrY8Ic+EFFKVhULM=

pkg/fs/findParent.go

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
package fs
2+
3+
import (
4+
"os"
5+
"path"
6+
"path/filepath"
7+
)
8+
9+
func pathExists(path string) bool {
10+
_, err := os.Stat(path)
11+
return err == nil
12+
}
13+
14+
func FindParentWithFile(cwd string, file string) string {
15+
if pathExists(path.Join(cwd, file)) {
16+
return cwd
17+
}
18+
parent := filepath.Dir(cwd)
19+
if parent == cwd {
20+
return ""
21+
}
22+
return FindParentWithFile(parent, file)
23+
}

pkg/node-modules/cli_test.go

+49
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
package node_modules
2+
3+
import (
4+
"encoding/json"
5+
"fmt"
6+
"os/exec"
7+
"path"
8+
"testing"
9+
10+
"github.com/develar/app-builder/pkg/fs"
11+
. "github.com/onsi/gomega"
12+
"github.com/samber/lo"
13+
)
14+
15+
type NodeTreeDepItem struct {
16+
Name string `json:"name"`
17+
Version string `json:"version"`
18+
}
19+
20+
type NodeTreeItem struct {
21+
Dir string `json:"dir"`
22+
Deps []NodeTreeDepItem `json:"deps"`
23+
}
24+
25+
func nodeDepTree(t *testing.T, dir string) {
26+
g := NewGomegaWithT(t)
27+
rootPath := fs.FindParentWithFile(Dirname(), "go.mod")
28+
cmd := exec.Command("go", "run", path.Join(rootPath, "main.go"), "node-dep-tree", "--dir", dir)
29+
output, err := cmd.Output()
30+
if err != nil {
31+
fmt.Println("err", err)
32+
}
33+
g.Expect(err).NotTo(HaveOccurred())
34+
var j []NodeTreeItem
35+
json.Unmarshal(output, &j)
36+
r := lo.FlatMap(j, func(it NodeTreeItem, i int) []string {
37+
return lo.Map(it.Deps, func(it NodeTreeDepItem, i int) string {
38+
return it.Name
39+
})
40+
})
41+
g.Expect(r).To(ConsistOf([]string{
42+
"react", "js-tokens", "loose-envify",
43+
}))
44+
}
45+
46+
func TestNodeDepTreeCmd(t *testing.T) {
47+
nodeDepTree(t, path.Join(Dirname(), "npm-demo"))
48+
nodeDepTree(t, path.Join(Dirname(), "pnpm-demo"))
49+
}

pkg/node-modules/helper_test.go

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
package node_modules
2+
3+
import (
4+
"path"
5+
"runtime"
6+
)
7+
8+
func Dirname() string {
9+
_, filename, _, _ := runtime.Caller(1)
10+
return path.Dir(filename)
11+
}

pkg/node-modules/nodeModuleCollector.go

+15-4
Original file line numberDiff line numberDiff line change
@@ -6,9 +6,10 @@ import (
66
"path/filepath"
77
"strings"
88

9+
"github.com/develar/app-builder/pkg/fs"
910
"github.com/develar/app-builder/pkg/log"
1011
"github.com/develar/errors"
11-
"github.com/json-iterator/go"
12+
jsoniter "github.com/json-iterator/go"
1213
"go.uber.org/zap"
1314
)
1415

@@ -21,9 +22,9 @@ type Dependency struct {
2122
Version string `json:"version"`
2223
Dependencies map[string]string `json:"dependencies"`
2324
OptionalDependencies map[string]string `json:"optionalDependencies"`
24-
Binary* DependencyBinary `json:"binary`
25+
Binary *DependencyBinary `json:"binary`
2526

26-
dir string
27+
dir string
2728
isOptional int
2829
}
2930

@@ -188,7 +189,11 @@ func (t *Collector) resolveDependency(parentNodeModuleDir string, name string) (
188189
}
189190
}
190191

191-
dependencyDir := filepath.Join(parentNodeModuleDir, name)
192+
realParentNodeModuleDir := fs.FindParentWithFile(parentNodeModuleDir, name)
193+
if realParentNodeModuleDir == "" {
194+
return nil, nil
195+
}
196+
dependencyDir := filepath.Join(realParentNodeModuleDir, name)
192197
dependency, err := readPackageJson(dependencyDir)
193198
if err != nil {
194199
if os.IsNotExist(err) {
@@ -220,6 +225,12 @@ func findNearestNodeModuleDir(dir string) (string, error) {
220225
return "", nil
221226
}
222227

228+
realDir, err := filepath.EvalSymlinks(dir)
229+
if err != nil {
230+
return "", errors.WithStack(err)
231+
}
232+
dir = realDir
233+
223234
guardCount := 0
224235
for {
225236
nodeModuleDir := filepath.Join(dir, "node_modules")
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
package node_modules
2+
3+
import (
4+
"path"
5+
"testing"
6+
7+
. "github.com/onsi/gomega"
8+
"github.com/samber/lo"
9+
)
10+
11+
func TestReadDependencyTreeByNpm(t *testing.T) {
12+
g := NewGomegaWithT(t)
13+
14+
collector := &Collector{
15+
unresolvedDependencies: make(map[string]bool),
16+
excludedDependencies: make(map[string]bool),
17+
NodeModuleDirToDependencyMap: make(map[string]*map[string]*Dependency),
18+
}
19+
20+
dir := path.Join(Dirname(), "npm-demo")
21+
22+
dependency, err := readPackageJson(dir)
23+
dependency.dir = dir
24+
g.Expect(err).NotTo(HaveOccurred())
25+
26+
err = collector.readDependencyTree(dependency)
27+
g.Expect(err).NotTo(HaveOccurred())
28+
r := lo.FlatMap(lo.Values(collector.NodeModuleDirToDependencyMap), func(it *map[string]*Dependency, i int) []string {
29+
return lo.Keys(*it)
30+
})
31+
g.Expect(r).To(ConsistOf([]string{
32+
"js-tokens", "react", "loose-envify",
33+
}))
34+
}
35+
36+
func TestReadDependencyTreeByPnpm(t *testing.T) {
37+
g := NewGomegaWithT(t)
38+
39+
collector := &Collector{
40+
unresolvedDependencies: make(map[string]bool),
41+
excludedDependencies: make(map[string]bool),
42+
NodeModuleDirToDependencyMap: make(map[string]*map[string]*Dependency),
43+
}
44+
45+
dir := path.Join(Dirname(), "pnpm-demo")
46+
47+
dependency, err := readPackageJson(dir)
48+
dependency.dir = dir
49+
g.Expect(err).NotTo(HaveOccurred())
50+
51+
err = collector.readDependencyTree(dependency)
52+
g.Expect(err).NotTo(HaveOccurred())
53+
r := lo.FlatMap(lo.Values(collector.NodeModuleDirToDependencyMap), func(it *map[string]*Dependency, i int) []string {
54+
return lo.Keys(*it)
55+
})
56+
g.Expect(r).To(ConsistOf([]string{
57+
"js-tokens", "react", "loose-envify",
58+
}))
59+
}

pkg/node-modules/npm-demo/package-lock.json

+43
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
{
2+
"name": "npm-demo",
3+
"version": "1.0.0",
4+
"description": "",
5+
"main": "index.js",
6+
"scripts": {
7+
"test": "echo \"Error: no test specified\" && exit 1"
8+
},
9+
"keywords": [],
10+
"author": "",
11+
"license": "ISC",
12+
"dependencies": {
13+
"react": "^18.2.0"
14+
}
15+
}
+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
{
2+
"name": "pnpm-demo",
3+
"version": "1.0.0",
4+
"description": "",
5+
"main": "index.js",
6+
"scripts": {
7+
"test": "echo \"Error: no test specified\" && exit 1"
8+
},
9+
"keywords": [],
10+
"author": "",
11+
"license": "ISC",
12+
"dependencies": {
13+
"react": "^18.2.0"
14+
}
15+
}

pkg/node-modules/pnpm-demo/pnpm-lock.yaml

+30
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)