Skip to content

Commit 44b96f9

Browse files
committed
Handle * when resolving paths in watch path
Signed-off-by: Joana Hrotkó <[email protected]>
1 parent 4aacd02 commit 44b96f9

File tree

8 files changed

+131
-51
lines changed

8 files changed

+131
-51
lines changed

loader/loader_test.go

+80-45
Original file line numberDiff line numberDiff line change
@@ -3097,7 +3097,8 @@ services:
30973097
}
30983098

30993099
func TestLoadDevelopConfig(t *testing.T) {
3100-
project, err := Load(buildConfigDetails(`
3100+
t.Run("successfully load watch config", func(t *testing.T) {
3101+
project, err := Load(buildConfigDetails(`
31013102
name: load-develop
31023103
services:
31033104
frontend:
@@ -3112,7 +3113,6 @@ services:
31123113
target: /var/www
31133114
ignore:
31143115
- node_modules/
3115-
31163116
backend:
31173117
image: example/backend
31183118
build: ./backend
@@ -3123,7 +3123,6 @@ services:
31233123
path:
31243124
- ./backend/src
31253125
- ./backend
3126-
31273126
proxy:
31283127
image: example/proxy
31293128
build: ./proxy
@@ -3134,67 +3133,103 @@ services:
31343133
action: sync+restart
31353134
target: /etc/nginx/proxy.conf
31363135
`, nil), func(options *Options) {
3137-
options.ResolvePaths = false
3138-
options.SkipValidation = true
3139-
})
3140-
assert.NilError(t, err)
3141-
frontend, err := project.GetService("frontend")
3142-
assert.NilError(t, err)
3143-
assert.DeepEqual(t, *frontend.Develop, types.DevelopConfig{
3144-
Watch: []types.Trigger{
3145-
{
3146-
Path: []string{"./webapp/html"},
3147-
Action: types.WatchActionSync,
3148-
Target: "/var/www",
3149-
Ignore: []string{"node_modules/"},
3150-
Extensions: types.Extensions{
3151-
"x-initialSync": true,
3136+
options.ResolvePaths = false
3137+
options.SkipValidation = true
3138+
})
3139+
assert.NilError(t, err)
3140+
frontend, err := project.GetService("frontend")
3141+
assert.NilError(t, err)
3142+
assert.DeepEqual(t, *frontend.Develop, types.DevelopConfig{
3143+
Watch: []types.Trigger{
3144+
{
3145+
Path: []string{"./webapp/html"},
3146+
Action: types.WatchActionSync,
3147+
Target: "/var/www",
3148+
Ignore: []string{"node_modules/"},
3149+
Extensions: types.Extensions{
3150+
"x-initialSync": true,
3151+
},
31523152
},
31533153
},
3154-
},
3155-
})
3156-
backend, err := project.GetService("backend")
3157-
assert.NilError(t, err)
3158-
assert.DeepEqual(t, *backend.Develop, types.DevelopConfig{
3159-
Watch: []types.Trigger{
3160-
{
3161-
Path: []string{"./backend/src", "./backend"},
3162-
Action: types.WatchActionRebuild,
3154+
})
3155+
backend, err := project.GetService("backend")
3156+
assert.NilError(t, err)
3157+
assert.DeepEqual(t, *backend.Develop, types.DevelopConfig{
3158+
Watch: []types.Trigger{
3159+
{
3160+
Path: []string{"./backend/src", "./backend"},
3161+
Action: types.WatchActionRebuild,
3162+
},
31633163
},
3164-
},
3165-
})
3166-
proxy, err := project.GetService("proxy")
3167-
assert.NilError(t, err)
3168-
assert.DeepEqual(t, *proxy.Develop, types.DevelopConfig{
3169-
Watch: []types.Trigger{
3170-
{
3171-
Path: []string{"./proxy/proxy.conf"},
3172-
Action: types.WatchActionSyncRestart,
3173-
Target: "/etc/nginx/proxy.conf",
3164+
})
3165+
proxy, err := project.GetService("proxy")
3166+
assert.NilError(t, err)
3167+
assert.DeepEqual(t, *proxy.Develop, types.DevelopConfig{
3168+
Watch: []types.Trigger{
3169+
{
3170+
Path: []string{"./proxy/proxy.conf"},
3171+
Action: types.WatchActionSyncRestart,
3172+
Target: "/etc/nginx/proxy.conf",
3173+
},
31743174
},
3175-
},
3175+
})
31763176
})
3177-
}
31783177

3179-
func TestBadDevelopConfig(t *testing.T) {
3180-
_, err := LoadWithContext(context.TODO(), buildConfigDetails(`
3178+
t.Run("should not load successfully bad watch config", func(t *testing.T) {
3179+
_, err := LoadWithContext(context.TODO(), buildConfigDetails(`
31813180
name: load-develop
31823181
services:
31833182
frontend:
31843183
image: example/webapp
31853184
build: ./webapp
31863185
develop:
31873186
watch:
3188-
# sync static content
3187+
# sync static content
31893188
- path: ./webapp/html
31903189
target: /var/www
31913190
ignore:
31923191
- node_modules/
3193-
31943192
`, nil), func(options *Options) {
3195-
options.ResolvePaths = false
3193+
options.ResolvePaths = false
3194+
})
3195+
assert.ErrorContains(t, err, "validating filename0.yml: services.frontend.develop.watch.0 action is required")
3196+
})
3197+
3198+
t.Run("should return an error when cannot resolve path", func(t *testing.T) {
3199+
b, err := os.ReadFile("testdata/watch/compose-test-watch-star.yaml")
3200+
assert.NilError(t, err)
3201+
3202+
configDetails := types.ConfigDetails{
3203+
WorkingDir: "testdata",
3204+
ConfigFiles: []types.ConfigFile{
3205+
{Filename: "watch/compose-test-watch-star.yaml", Content: b},
3206+
},
3207+
Environment: map[string]string{},
3208+
}
3209+
expServices := types.Services{
3210+
"app": {
3211+
Name: "app",
3212+
Image: "example/app",
3213+
Environment: types.MappingWithEquals{},
3214+
Networks: map[string]*types.ServiceNetworkConfig{"default": nil},
3215+
Develop: &types.DevelopConfig{
3216+
Watch: []types.Trigger{
3217+
{
3218+
Path: []string{
3219+
"testdata/watch/other.txt",
3220+
"testdata/watch/some-text.txt",
3221+
},
3222+
Action: types.WatchActionRebuild,
3223+
},
3224+
},
3225+
},
3226+
},
3227+
}
3228+
3229+
actual, err := Load(configDetails)
3230+
assert.NilError(t, err)
3231+
assert.DeepEqual(t, actual.Services, expServices)
31963232
})
3197-
assert.ErrorContains(t, err, "validating filename0.yml: services.frontend.develop.watch.0 action is required")
31983233
}
31993234

32003235
func TestBadServiceConfig(t *testing.T) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
name: compose-test-watch-star
2+
services:
3+
app:
4+
image: example/app
5+
develop:
6+
watch:
7+
- path: ./watch/*.txt
8+
action: rebuild
9+
# - path: ./watch/*
10+
# target: ./app
11+
# action: sync

loader/testdata/watch/other.txt

Whitespace-only changes.

loader/testdata/watch/some-text.txt

Whitespace-only changes.

loader/validate.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -173,7 +173,7 @@ func checkConsistency(project *types.Project) error {
173173

174174
}
175175
if len(watch.Path) > 1 {
176-
return fmt.Errorf("services.%s.develop.watch: can only use more than one path for actions %s and %s: %w", s.Name, types.WatchActionRebuild, types.WatchActionRestart, errdefs.ErrInvalid)
176+
return fmt.Errorf("services.%s.develop.watch: detected multiple paths %s for action %s. Multiple files are only valid for %s and %s actions: %w", s.Name, watch.Path, watch.Action, types.WatchActionRebuild, types.WatchActionRestart, errdefs.ErrInvalid)
177177
}
178178
}
179179
}

loader/validate_test.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -396,7 +396,7 @@ func TestValidateWatch(t *testing.T) {
396396
},
397397
}
398398
err := checkConsistency(&project)
399-
assert.Error(t, err, "services.myservice.develop.watch: can only use more than one path for actions rebuild and restart: invalid compose project")
399+
assert.ErrorContains(t, err, "services.myservice.develop.watch: detected multiple paths")
400400
})
401401
}
402402
tests = []WatchActionTest{

paths/unix.go

+37-4
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717
package paths
1818

1919
import (
20+
"fmt"
2021
"path"
2122
"path/filepath"
2223

@@ -49,9 +50,41 @@ func (r *relativePathsResolver) absSymbolicLink(value any) (any, error) {
4950
if err != nil {
5051
return nil, err
5152
}
52-
str, ok := abs.(string)
53-
if !ok {
54-
return abs, nil
53+
switch t := abs.(type) {
54+
case string:
55+
// this can return []string if * matches more than one file
56+
return resolveAbsStarPath(t)
57+
case []any:
58+
var res []any
59+
for _, tt := range t {
60+
s, _ := tt.(string)
61+
r, err := resolveAbsStarPath(s)
62+
if err != nil {
63+
return nil, err
64+
}
65+
res = append(res, r...)
66+
}
67+
return res, nil
68+
}
69+
70+
return abs, nil
71+
}
72+
73+
func resolveAbsStarPath(t string) ([]any, error) {
74+
matches, err := filepath.Glob(t)
75+
if err != nil {
76+
return nil, err
77+
}
78+
if len(matches) == 0 {
79+
return nil, fmt.Errorf("could not resolve %s. Please make sure it exists?", t)
80+
}
81+
res := make([]any, len(matches))
82+
for i, m := range matches {
83+
symb, err := utils.ResolveSymbolicLink(m)
84+
if err != nil {
85+
return nil, err
86+
}
87+
res[i] = symb
5588
}
56-
return utils.ResolveSymbolicLink(str)
89+
return res, nil
5790
}

transform/canonical.go

+1
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,7 @@ func init() {
3636
transformers["services.*.networks"] = transformServiceNetworks
3737
transformers["services.*.volumes.*"] = transformVolumeMount
3838
transformers["services.*.dns"] = transformStringOrList
39+
transformers["services.*.develop.watch.*.path"] = transformStringOrList
3940
transformers["services.*.devices.*"] = transformDeviceMapping
4041
transformers["services.*.secrets.*"] = transformFileMount
4142
transformers["services.*.configs.*"] = transformFileMount

0 commit comments

Comments
 (0)