Skip to content

Commit 1b8198d

Browse files
committedSep 21, 2022
build: support go:embed
1 parent e13d1c9 commit 1b8198d

File tree

8 files changed

+246
-22
lines changed

8 files changed

+246
-22
lines changed
 

‎.std_test_pkg_exclusions

-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,3 @@
1-
embed/internal/embedtest
21
encoding/xml
32
go/build
43
go/internal/srcimporter

‎build/build.go

+43-16
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ import (
1313
"go/scanner"
1414
"go/token"
1515
"go/types"
16+
"io/fs"
1617
"os"
1718
"os/exec"
1819
"path"
@@ -398,11 +399,12 @@ func (p *PackageData) InternalBuildContext() *build.Context {
398399
func (p *PackageData) TestPackage() *PackageData {
399400
return &PackageData{
400401
Package: &build.Package{
401-
Name: p.Name,
402-
ImportPath: p.ImportPath,
403-
Dir: p.Dir,
404-
GoFiles: append(p.GoFiles, p.TestGoFiles...),
405-
Imports: append(p.Imports, p.TestImports...),
402+
Name: p.Name,
403+
ImportPath: p.ImportPath,
404+
Dir: p.Dir,
405+
GoFiles: append(p.GoFiles, p.TestGoFiles...),
406+
Imports: append(p.Imports, p.TestImports...),
407+
EmbedPatternPos: joinEmbedPatternPos(p.EmbedPatternPos, p.TestEmbedPatternPos),
406408
},
407409
IsTest: true,
408410
JSFiles: p.JSFiles,
@@ -414,11 +416,12 @@ func (p *PackageData) TestPackage() *PackageData {
414416
func (p *PackageData) XTestPackage() *PackageData {
415417
return &PackageData{
416418
Package: &build.Package{
417-
Name: p.Name + "_test",
418-
ImportPath: p.ImportPath + "_test",
419-
Dir: p.Dir,
420-
GoFiles: p.XTestGoFiles,
421-
Imports: p.XTestImports,
419+
Name: p.Name + "_test",
420+
ImportPath: p.ImportPath + "_test",
421+
Dir: p.Dir,
422+
GoFiles: p.XTestGoFiles,
423+
Imports: p.XTestImports,
424+
EmbedPatternPos: p.XTestEmbedPatternPos,
422425
},
423426
IsTest: true,
424427
bctx: p.bctx,
@@ -548,12 +551,30 @@ func (s *Session) BuildFiles(filenames []string, pkgObj string, cwd string) erro
548551
return fmt.Errorf("named files must all be in one directory; have: %v", strings.Join(dirList, ", "))
549552
}
550553

554+
root := dirList[0]
555+
ctx := build.Default
556+
ctx.UseAllFiles = true
557+
ctx.ReadDir = func(dir string) ([]fs.FileInfo, error) {
558+
n := len(filenames)
559+
infos := make([]fs.FileInfo, n)
560+
for i := 0; i < n; i++ {
561+
info, err := os.Stat(filenames[i])
562+
if err != nil {
563+
return nil, err
564+
}
565+
infos[i] = info
566+
}
567+
return infos, nil
568+
}
569+
p, err := ctx.Import(".", root, 0)
570+
if err != nil {
571+
return err
572+
}
573+
p.Name = "main"
574+
p.ImportPath = "main"
575+
551576
pkg := &PackageData{
552-
Package: &build.Package{
553-
Name: "main",
554-
ImportPath: "main",
555-
Dir: dirList[0],
556-
},
577+
Package: p,
557578
// This ephemeral package doesn't have a unique import path to be used as a
558579
// build cache key, so we never cache it.
559580
SrcModTime: time.Now().Add(time.Hour),
@@ -562,7 +583,6 @@ func (s *Session) BuildFiles(filenames []string, pkgObj string, cwd string) erro
562583

563584
for _, file := range filenames {
564585
if !strings.HasSuffix(file, ".inc.js") {
565-
pkg.GoFiles = append(pkg.GoFiles, filepath.Base(file))
566586
continue
567587
}
568588

@@ -676,6 +696,13 @@ func (s *Session) BuildPackage(pkg *PackageData) (*compiler.Archive, error) {
676696
if err != nil {
677697
return nil, err
678698
}
699+
embed, err := embedFiles(pkg, fileSet, files)
700+
if err != nil {
701+
return nil, err
702+
}
703+
if embed != nil {
704+
files = append(files, embed)
705+
}
679706

680707
importContext := &compiler.ImportContext{
681708
Packages: s.Types,

‎build/embed.go

+175
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,175 @@
1+
package build
2+
3+
import (
4+
"bytes"
5+
"fmt"
6+
"go/ast"
7+
"go/parser"
8+
"go/token"
9+
"strconv"
10+
11+
"github.com/visualfc/goembed"
12+
)
13+
14+
func buildIdent(name string) string {
15+
return fmt.Sprintf("__gopherjs_embed_%x__", name)
16+
}
17+
18+
var embed_head = `package %v
19+
20+
import (
21+
"embed"
22+
_ "unsafe"
23+
)
24+
25+
//go:linkname __gopherjs_embed_buildFS__ embed.buildFS
26+
func __gopherjs_embed_buildFS__(list []struct {
27+
name string
28+
data string
29+
hash [16]byte
30+
}) (f embed.FS)
31+
`
32+
33+
// embedFiles generates an additional source file, which initializes all variables in the package with a go:embed directive.
34+
func embedFiles(pkg *PackageData, fset *token.FileSet, files []*ast.File) (*ast.File, error) {
35+
if len(pkg.EmbedPatternPos) == 0 {
36+
return nil, nil
37+
}
38+
39+
ems, err := goembed.CheckEmbed(pkg.EmbedPatternPos, fset, files)
40+
if err != nil {
41+
return nil, err
42+
}
43+
44+
r := goembed.NewResolve()
45+
for _, em := range ems {
46+
fs, err := r.Load(pkg.Dir, fset, em)
47+
if err != nil {
48+
return nil, err
49+
}
50+
switch em.Kind {
51+
case goembed.EmbedMaybeAlias:
52+
// value = Type(data)
53+
// valid alias string or []byte type used by types.check
54+
em.Spec.Values = []ast.Expr{
55+
&ast.CallExpr{
56+
Fun: em.Spec.Type,
57+
Args: []ast.Expr{
58+
&ast.Ident{Name: buildIdent(fs[0].Name),
59+
NamePos: em.Spec.Names[0].NamePos},
60+
},
61+
}}
62+
case goembed.EmbedBytes:
63+
// value = []byte(data)
64+
em.Spec.Values = []ast.Expr{
65+
&ast.CallExpr{
66+
Fun: em.Spec.Type,
67+
Args: []ast.Expr{ast.NewIdent(buildIdent(fs[0].Name))},
68+
}}
69+
case goembed.EmbedString:
70+
// value = data
71+
em.Spec.Values = []ast.Expr{ast.NewIdent(buildIdent(fs[0].Name))}
72+
case goembed.EmbedFiles:
73+
// value = __gopherjs_embed_buildFS__([]struct{name string; data string; hash [16]byte}{...})
74+
fs = goembed.BuildFS(fs)
75+
elts := make([]ast.Expr, len(fs))
76+
for i, f := range fs {
77+
if len(f.Data) == 0 {
78+
elts[i] = &ast.CompositeLit{
79+
Elts: []ast.Expr{
80+
&ast.BasicLit{Kind: token.STRING, Value: strconv.Quote(f.Name)},
81+
&ast.BasicLit{Kind: token.STRING, Value: `""`},
82+
&ast.CompositeLit{
83+
Type: &ast.ArrayType{
84+
Len: &ast.BasicLit{Kind: token.INT, Value: "16"},
85+
Elt: ast.NewIdent("byte"),
86+
},
87+
},
88+
},
89+
}
90+
} else {
91+
var hash [16]ast.Expr
92+
for j, v := range f.Hash {
93+
hash[j] = &ast.BasicLit{Kind: token.INT, Value: strconv.Itoa(int(v))}
94+
}
95+
elts[i] = &ast.CompositeLit{
96+
Elts: []ast.Expr{
97+
&ast.BasicLit{Kind: token.STRING, Value: strconv.Quote(f.Name)},
98+
ast.NewIdent(buildIdent(f.Name)),
99+
&ast.CompositeLit{
100+
Type: &ast.ArrayType{
101+
Len: &ast.BasicLit{Kind: token.INT, Value: "16"},
102+
Elt: ast.NewIdent("byte"),
103+
},
104+
Elts: hash[:],
105+
},
106+
},
107+
}
108+
}
109+
}
110+
call := &ast.CallExpr{
111+
Fun: ast.NewIdent("__gopherjs_embed_buildFS__"),
112+
Args: []ast.Expr{
113+
&ast.CompositeLit{
114+
Type: &ast.ArrayType{
115+
Elt: &ast.StructType{
116+
Fields: &ast.FieldList{
117+
List: []*ast.Field{
118+
&ast.Field{
119+
Names: []*ast.Ident{ast.NewIdent("name")},
120+
Type: ast.NewIdent("string"),
121+
},
122+
&ast.Field{
123+
Names: []*ast.Ident{ast.NewIdent("data")},
124+
Type: ast.NewIdent("string"),
125+
},
126+
&ast.Field{
127+
Names: []*ast.Ident{ast.NewIdent("hash")},
128+
Type: &ast.ArrayType{
129+
Len: &ast.BasicLit{Kind: token.INT, Value: "16"},
130+
Elt: ast.NewIdent("byte"),
131+
},
132+
},
133+
},
134+
},
135+
},
136+
},
137+
Elts: elts,
138+
},
139+
},
140+
}
141+
em.Spec.Values = []ast.Expr{call}
142+
}
143+
}
144+
145+
var buf bytes.Buffer
146+
fmt.Fprintf(&buf, embed_head, pkg.Name)
147+
buf.WriteString("\nconst (\n")
148+
for _, f := range r.Files() {
149+
if len(f.Data) == 0 {
150+
fmt.Fprintf(&buf, "\t%v = \"\"\n", buildIdent(f.Name))
151+
} else {
152+
fmt.Fprintf(&buf, "\t%v = \"%v\"\n", buildIdent(f.Name), goembed.BytesToHex(f.Data))
153+
}
154+
}
155+
buf.WriteString(")\n\n")
156+
f, err := parser.ParseFile(fset, "js_embed.go", buf.String(), parser.ParseComments)
157+
if err != nil {
158+
return nil, err
159+
}
160+
return f, nil
161+
}
162+
163+
func joinEmbedPatternPos(m1, m2 map[string][]token.Position) map[string][]token.Position {
164+
if len(m1) == 0 && len(m2) == 0 {
165+
return nil
166+
}
167+
m := make(map[string][]token.Position)
168+
for k, v := range m1 {
169+
m[k] = v
170+
}
171+
for k, v := range m2 {
172+
m[k] = append(m[k], v...)
173+
}
174+
return m
175+
}

‎compiler/natives/src/embed/embed.go

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
//go:build js
2+
// +build js
3+
4+
package embed
5+
6+
func buildFS(list []struct {
7+
name string
8+
data string
9+
hash [16]byte
10+
}) (f FS) {
11+
n := len(list)
12+
files := make([]file, n)
13+
for i := 0; i < n; i++ {
14+
files[i].name = list[i].name
15+
files[i].data = list[i].data
16+
files[i].hash = list[i].hash
17+
}
18+
f.files = &files
19+
return
20+
}

‎go.mod

+1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ require (
1313
github.com/sirupsen/logrus v1.8.1
1414
github.com/spf13/cobra v1.2.1
1515
github.com/spf13/pflag v1.0.5
16+
github.com/visualfc/goembed v0.3.3
1617
golang.org/x/crypto v0.0.0-20220411220226-7b82a4e95df4
1718
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c
1819
golang.org/x/sys v0.0.0-20220412211240-33da011f77ad

‎go.sum

+2
Original file line numberDiff line numberDiff line change
@@ -235,6 +235,8 @@ github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/
235235
github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY=
236236
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
237237
github.com/subosito/gotenv v1.2.0/go.mod h1:N0PQaV/YGNqwC0u51sEeR/aUtSLEXKX9iv69rRypqCw=
238+
github.com/visualfc/goembed v0.3.3 h1:pOL02L715tHKsLQVMcZz06tTzRDAHkJKJLRnCA22G9Q=
239+
github.com/visualfc/goembed v0.3.3/go.mod h1:jCVCz/yTJGyslo6Hta+pYxWWBuq9ADCcIVZBTQ0/iVI=
238240
github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
239241
github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
240242
github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=

‎tests/testdata/legacy_syscall/main.go

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
1-
//go:build legacy_syscall,gopherjs
1+
//go:build legacy_syscall && gopherjs
2+
// +build legacy_syscall,gopherjs
23

34
// This program tests GopherJS's ability to perform raw syscalls using the
45
// deprecated node_syscall extension. See TestLegacySyscall.

‎tool.go

+3-4
Original file line numberDiff line numberDiff line change
@@ -135,16 +135,15 @@ func main() {
135135
if err != nil {
136136
return fmt.Errorf("failed to expand patterns %v: %w", args, err)
137137
}
138-
139138
for _, pkgPath := range pkgs {
140139
if s.Watcher != nil {
141-
pkg, err := xctx.Import(pkgPath, "", build.FindOnly)
140+
pkg, err := xctx.Import(pkgPath, currentDirectory, build.FindOnly)
142141
if err != nil {
143142
return err
144143
}
145144
s.Watcher.Add(pkg.Dir)
146145
}
147-
pkg, err := xctx.Import(pkgPath, ".", 0)
146+
pkg, err := xctx.Import(pkgPath, currentDirectory, 0)
148147
if err != nil {
149148
return err
150149
}
@@ -208,7 +207,7 @@ func main() {
208207
}
209208
}
210209
for _, pkgPath := range pkgs {
211-
pkg, err := xctx.Import(pkgPath, ".", 0)
210+
pkg, err := xctx.Import(pkgPath, currentDirectory, 0)
212211
if s.Watcher != nil && pkg != nil { // add watch even on error
213212
s.Watcher.Add(pkg.Dir)
214213
}

0 commit comments

Comments
 (0)
Please sign in to comment.