Skip to content

Commit

Permalink
fix: watch for changes in replace paths (#1323)
Browse files Browse the repository at this point in the history
Fixes #1271
  • Loading branch information
wesbillman authored Apr 29, 2024
1 parent 4edee31 commit ff1e609
Show file tree
Hide file tree
Showing 8 changed files with 233 additions and 30 deletions.
8 changes: 4 additions & 4 deletions buildengine/discover_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ func TestDiscoverModules(t *testing.T) {
DeployDir: "_ftl",
Schema: "schema.pb",
Errors: "errors.pb",
Watch: []string{"**/*.go", "go.mod", "go.sum"},
Watch: []string{"**/*.go", "go.mod", "go.sum", "../../../../go-runtime/ftl/**/*.go"},
},
},
Module{
Expand All @@ -38,7 +38,7 @@ func TestDiscoverModules(t *testing.T) {
DeployDir: "_ftl",
Schema: "schema.pb",
Errors: "errors.pb",
Watch: []string{"**/*.go", "go.mod", "go.sum"},
Watch: []string{"**/*.go", "go.mod", "go.sum", "../../../../go-runtime/ftl/**/*.go"},
},
},
Module{
Expand Down Expand Up @@ -117,7 +117,7 @@ func TestDiscoverModules(t *testing.T) {
DeployDir: "_ftl",
Schema: "schema.pb",
Errors: "errors.pb",
Watch: []string{"**/*.go", "go.mod", "go.sum"},
Watch: []string{"**/*.go", "go.mod", "go.sum", "../../../../go-runtime/ftl/**/*.go"},
},
},
Module{
Expand All @@ -130,7 +130,7 @@ func TestDiscoverModules(t *testing.T) {
DeployDir: "_ftl",
Schema: "schema.pb",
Errors: "errors.pb",
Watch: []string{"**/*.go", "go.mod", "go.sum"},
Watch: []string{"**/*.go", "go.mod", "go.sum", "../../../../go-runtime/ftl/**/*.go"},
},
},
ExternalLibrary{
Expand Down
64 changes: 45 additions & 19 deletions buildengine/filehash.go
Original file line number Diff line number Diff line change
Expand Up @@ -67,32 +67,35 @@ func ComputeFileHashes(project Project) (FileHashes, error) {
config := project.Config()

fileHashes := make(FileHashes)
err := WalkDir(config.Dir, func(srcPath string, entry fs.DirEntry) error {
if entry.IsDir() {
rootDirs := computeRootDirs(config.Dir, config.Watch)

for _, rootDir := range rootDirs {
err := WalkDir(rootDir, func(srcPath string, entry fs.DirEntry) error {
if entry.IsDir() {
return nil
}
hash, matched, err := ComputeFileHash(rootDir, srcPath, config.Watch)
if err != nil {
return err
}
if !matched {
return nil
}
fileHashes[srcPath] = hash
return nil
}
hash, matched, err := ComputeFileHash(project, srcPath)
})

if err != nil {
return err
return nil, err
}
if !matched {
return nil
}
fileHashes[srcPath] = hash
return nil
})
if err != nil {
return nil, err
}

return fileHashes, err
return fileHashes, nil
}

func ComputeFileHash(project Project, srcPath string) (hash []byte, matched bool, err error) {
config := project.Config()

for _, pattern := range config.Watch {
relativePath, err := filepath.Rel(config.Dir, srcPath)
func ComputeFileHash(baseDir, srcPath string, watch []string) (hash []byte, matched bool, err error) {
for _, pattern := range watch {
relativePath, err := filepath.Rel(baseDir, srcPath)
if err != nil {
return nil, false, err
}
Expand Down Expand Up @@ -122,3 +125,26 @@ func ComputeFileHash(project Project, srcPath string) (hash []byte, matched bool
}
return nil, false, nil
}

// computeRootDirs computes the unique root directories for the given baseDir and patterns.
func computeRootDirs(baseDir string, patterns []string) []string {
uniqueRoots := make(map[string]struct{})
uniqueRoots[baseDir] = struct{}{}

for _, pattern := range patterns {
fullPath := filepath.Join(baseDir, pattern)
dirPath, _ := doublestar.SplitPattern(fullPath)
cleanedPath := filepath.Clean(dirPath)

if _, err := os.Stat(cleanedPath); err == nil {
uniqueRoots[cleanedPath] = struct{}{}
}
}

roots := make([]string, 0, len(uniqueRoots))
for root := range uniqueRoots {
roots = append(roots, root)
}

return roots
}
53 changes: 53 additions & 0 deletions buildengine/filehash_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
package buildengine

import (
"os"
"path/filepath"
"testing"

"github.com/alecthomas/assert/v2"
)

func TestComputeRootDirs(t *testing.T) {
dir := t.TempDir()
libDir := filepath.Join(dir, "libs", "test")

err := os.MkdirAll(libDir, 0700)
assert.NoError(t, err)

tests := []struct {
name string
baseDir string
patterns []string
roots []string
}{
{
name: "Simple Patterns",
baseDir: filepath.Join(dir, "modules", "test"),
patterns: []string{
"**/*.go",
"go.mod",
"go.sum",
},
roots: []string{filepath.Join(dir, "modules", "test")},
},
{
name: "Upward Traversal",
baseDir: filepath.Join(dir, "modules", "test"),
patterns: []string{
"../../libs/test/**/*.go",
},
roots: []string{
filepath.Join(dir, "modules", "test"),
filepath.Join(dir, "libs", "test"),
},
},
}

for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
roots := computeRootDirs(tt.baseDir, tt.patterns)
assert.Compare(t, roots, tt.roots)
})
}
}
2 changes: 1 addition & 1 deletion buildengine/testdata/projects/highgoversion/go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -43,4 +43,4 @@ require (
google.golang.org/protobuf v1.33.0 // indirect
)

replace github.com/TBD54566975/ftl => /Users/dli/Development/ftl
replace github.com/TBD54566975/ftl => ../../../..
2 changes: 1 addition & 1 deletion buildengine/watch.go
Original file line number Diff line number Diff line change
Expand Up @@ -210,7 +210,7 @@ func (t *modifyFilesTransaction) ModifiedFiles(paths ...string) error {
}

for _, path := range paths {
hash, matched, err := ComputeFileHash(projectHashes.Project, path)
hash, matched, err := ComputeFileHash(projectHashes.Project.Config().Dir, path, projectHashes.Project.Config().Watch)
if err != nil {
return err
}
Expand Down
103 changes: 98 additions & 5 deletions common/moduleconfig/moduleconfig.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,14 @@ package moduleconfig

import (
"fmt"
"go/parser"
"go/token"
"os"
"path/filepath"
"strings"

"github.com/BurntSushi/toml"
"golang.org/x/mod/modfile"
)

// ModuleGoConfig is language-specific configuration for Go modules.
Expand Down Expand Up @@ -95,6 +99,11 @@ func setConfigDefaults(moduleDir string, config *ModuleConfig) error {
}
if len(config.Watch) == 0 {
config.Watch = []string{"**/*.go", "go.mod", "go.sum"}
watches, err := replacementWatches(moduleDir, config.DeployDir)
if err != nil {
return err
}
config.Watch = append(config.Watch, watches...)
}
}

Expand All @@ -107,15 +116,99 @@ func setConfigDefaults(moduleDir string, config *ModuleConfig) error {
return fmt.Errorf("deploy files must be relative to the module directory")
}
}
for _, watch := range config.Watch {
if !isBeneath(moduleDir, watch) {
return fmt.Errorf("watch files must be relative to the module directory")
}
}

return nil
}

func isBeneath(moduleDir, path string) bool {
resolved := filepath.Clean(filepath.Join(moduleDir, path))
return strings.HasPrefix(resolved, strings.TrimSuffix(moduleDir, "/")+"/")
}

func replacementWatches(moduleDir, deployDir string) ([]string, error) {
goModPath := filepath.Join(moduleDir, "go.mod")
goModBytes, err := os.ReadFile(goModPath)
if err != nil {
return nil, fmt.Errorf("failed to read %s: %w", goModPath, err)
}
goModFile, err := modfile.Parse(goModPath, goModBytes, nil)
if err != nil {
return nil, fmt.Errorf("failed to parse %s: %w", goModPath, err)
}

replacements := make(map[string]string)
for _, r := range goModFile.Replace {
replacements[r.Old.Path] = r.New.Path
if strings.HasPrefix(r.New.Path, ".") {
relPath, err := filepath.Rel(filepath.Dir(goModPath), filepath.Join(filepath.Dir(goModPath), r.New.Path))
if err != nil {
return nil, err
}
replacements[r.Old.Path] = relPath
}
}

files, err := findReplacedImports(moduleDir, deployDir, replacements)
if err != nil {
return nil, err
}

uniquePatterns := make(map[string]struct{})
for _, file := range files {
pattern := filepath.Join(file, "**/*.go")
uniquePatterns[pattern] = struct{}{}
}

patterns := make([]string, 0, len(uniquePatterns))
for pattern := range uniquePatterns {
patterns = append(patterns, pattern)
}

return patterns, nil
}

// findReplacedImports finds Go files with imports that are specified in the replacements.
func findReplacedImports(moduleDir, deployDir string, replacements map[string]string) ([]string, error) {
var libPaths []string

err := filepath.WalkDir(moduleDir, func(path string, d os.DirEntry, err error) error {
if err != nil {
return err
}
if !d.IsDir() && !strings.Contains(path, deployDir) && strings.HasSuffix(path, ".go") {
imports, err := parseImports(path)
if err != nil {
return err
}

for _, imp := range imports {
for oldPath, newPath := range replacements {
if strings.HasPrefix(imp, oldPath) {
resolvedPath := filepath.Join(newPath, strings.TrimPrefix(imp, oldPath))
libPaths = append(libPaths, resolvedPath)
break // Only add the library path once for each import match
}
}
}
}
return nil
})

return libPaths, err
}

func parseImports(filePath string) ([]string, error) {
fset := token.NewFileSet()
file, err := parser.ParseFile(fset, filePath, nil, parser.ImportsOnly)
if err != nil {
return nil, err
}

var imports []string
for _, imp := range file.Imports {
// Trim the quotes from the import path value
trimmedPath := strings.Trim(imp.Path.Value, `"`)
imports = append(imports, trimmedPath)
}
return imports, nil
}
20 changes: 20 additions & 0 deletions common/moduleconfig/moduleconfig_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
package moduleconfig

import (
"path/filepath"
"reflect"
"testing"
)

func TestParseImportsFromTestData(t *testing.T) {
testFilePath := filepath.Join("testdata", "imports.go")
expectedImports := []string{"fmt", "os"}
imports, err := parseImports(testFilePath)
if err != nil {
t.Fatalf("Failed to parse imports: %v", err)
}

if !reflect.DeepEqual(imports, expectedImports) {
t.Errorf("parseImports() got = %v, want %v", imports, expectedImports)
}
}
11 changes: 11 additions & 0 deletions common/moduleconfig/testdata/imports.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
package main

import (
"fmt"
"os"
)

func main() {
fmt.Println("Hello, world!")
os.Exit(0)
}

0 comments on commit ff1e609

Please sign in to comment.