Skip to content

Commit

Permalink
chore: improve logging around lifecycle scripts (#420)
Browse files Browse the repository at this point in the history
  • Loading branch information
johnstcn authored Nov 22, 2024
1 parent b1dc272 commit fafad44
Show file tree
Hide file tree
Showing 2 changed files with 89 additions and 30 deletions.
8 changes: 8 additions & 0 deletions envbuilder.go
Original file line number Diff line number Diff line change
Expand Up @@ -169,6 +169,7 @@ func run(ctx context.Context, opts options.Options, execArgs *execArgsInfo) erro
RemoteEnv: make(map[string]string),
}
if fileExists(opts.Filesystem, workingDir.Image()) {
opts.Logger(log.LevelInfo, "Found magic image file at %s", workingDir.Image())
if err = parseMagicImageFile(opts.Filesystem, workingDir.Image(), &runtimeData); err != nil {
return fmt.Errorf("parse magic image file: %w", err)
}
Expand Down Expand Up @@ -287,6 +288,7 @@ func run(ctx context.Context, opts options.Options, execArgs *execArgsInfo) erro

var buildParams *devcontainer.Compiled
if opts.DockerfilePath == "" {
opts.Logger(log.LevelInfo, "No Dockerfile specified, looking for a devcontainer.json...")
// Only look for a devcontainer if a Dockerfile wasn't specified.
// devcontainer is a standard, so it's reasonable to be the default.
var devcontainerDir string
Expand All @@ -296,6 +298,7 @@ func run(ctx context.Context, opts options.Options, execArgs *execArgsInfo) erro
opts.Logger(log.LevelError, "Failed to locate devcontainer.json: %s", err.Error())
opts.Logger(log.LevelError, "Falling back to the default image...")
} else {
opts.Logger(log.LevelInfo, "Building in Devcontainer mode using %s", strings.TrimPrefix(runtimeData.DevcontainerPath, buildTimeWorkspaceFolder))
// We know a devcontainer exists.
// Let's parse it and use it!
file, err := opts.Filesystem.Open(runtimeData.DevcontainerPath)
Expand Down Expand Up @@ -334,6 +337,7 @@ func run(ctx context.Context, opts options.Options, execArgs *execArgsInfo) erro
} else {
// If a Dockerfile was specified, we use that.
dockerfilePath := filepath.Join(buildTimeWorkspaceFolder, opts.DockerfilePath)
opts.Logger(log.LevelInfo, "Building in Dockerfile-only mode using %s", opts.DockerfilePath)

// If the dockerfilePath is specified and deeper than the base of WorkspaceFolder AND the BuildContextPath is
// not defined, show a warning
Expand Down Expand Up @@ -1406,6 +1410,7 @@ func execOneLifecycleScript(
userInfo userInfo,
) error {
if s.IsEmpty() {
logf(log.LevelInfo, "=== No %s script specified", scriptName)
return nil
}
logf(log.LevelInfo, "=== Running %s as the %q user...", scriptName, userInfo.user.Username)
Expand All @@ -1424,6 +1429,7 @@ func execLifecycleScripts(
userInfo userInfo,
) error {
if options.PostStartScriptPath != "" {
options.Logger(log.LevelDebug, "Removing postStartScriptPath %s", options.PostStartScriptPath)
_ = os.Remove(options.PostStartScriptPath)
}

Expand All @@ -1432,6 +1438,8 @@ func execLifecycleScripts(
// skip remaining lifecycle commands
return nil
}
} else {
options.Logger(log.LevelDebug, "Skipping onCreateCommand for subsequent starts...")
}
if err := execOneLifecycleScript(ctx, options.Logger, scripts.UpdateContentCommand, "updateContentCommand", userInfo); err != nil {
// skip remaining lifecycle commands
Expand Down
111 changes: 81 additions & 30 deletions integration/integration_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -1188,37 +1188,88 @@ func TestBuildSecrets(t *testing.T) {
func TestLifecycleScripts(t *testing.T) {
t.Parallel()

// Ensures that a Git repository with a devcontainer.json is cloned and built.
srv := gittest.CreateGitServer(t, gittest.Options{
Files: map[string]string{
".devcontainer/devcontainer.json": `{
"name": "Test",
"build": {
"dockerfile": "Dockerfile"
},
"onCreateCommand": "echo create > /tmp/out",
"updateContentCommand": ["sh", "-c", "echo update >> /tmp/out"],
"postCreateCommand": "(echo -n postCreate. ; id -un) >> /tmp/out",
"postStartCommand": {
"parallel1": "echo parallel1 > /tmp/parallel1",
"parallel2": ["sh", "-c", "echo parallel2 > /tmp/parallel2"]
}
}`,
".devcontainer/Dockerfile": "FROM " + testImageAlpine + "\nUSER nobody",
for _, tt := range []struct {
name string
files map[string]string
outputCmd string
expectOutput string
}{
{
name: "build",
files: map[string]string{
".devcontainer/devcontainer.json": `{
"name": "Test",
"build": {
"dockerfile": "Dockerfile"
},
"onCreateCommand": "echo create > /tmp/out",
"updateContentCommand": ["sh", "-c", "echo update >> /tmp/out"],
"postCreateCommand": "(echo -n postCreate. ; id -un) >> /tmp/out",
"postStartCommand": {
"parallel1": "echo parallel1 > /tmp/parallel1",
"parallel2": ["sh", "-c", "echo parallel2 > /tmp/parallel2"]
}
}`,
".devcontainer/Dockerfile": "FROM " + testImageAlpine + "\nUSER nobody",
},
outputCmd: "cat /tmp/out /tmp/parallel1 /tmp/parallel2",
expectOutput: "create\nupdate\npostCreate.nobody\nparallel1\nparallel2",
},
})
ctr, err := runEnvbuilder(t, runOpts{env: []string{
envbuilderEnv("GIT_URL", srv.URL),
}})
require.NoError(t, err)

output := execContainer(t, ctr, "cat /tmp/out /tmp/parallel1 /tmp/parallel2")
require.Equal(t,
`create
update
postCreate.nobody
parallel1
parallel2`, strings.TrimSpace(output))
{
name: "image",
files: map[string]string{
".devcontainer/devcontainer.json": fmt.Sprintf(`{
"name": "Test",
"image": %q,
"containerUser": "nobody",
"onCreateCommand": "echo create > /tmp/out",
"updateContentCommand": ["sh", "-c", "echo update >> /tmp/out"],
"postCreateCommand": "(echo -n postCreate. ; id -un) >> /tmp/out",
"postStartCommand": {
"parallel1": "echo parallel1 > /tmp/parallel1",
"parallel2": ["sh", "-c", "echo parallel2 > /tmp/parallel2"]
}
}`, testImageAlpine),
},
outputCmd: "cat /tmp/out /tmp/parallel1 /tmp/parallel2",
expectOutput: "create\nupdate\npostCreate.nobody\nparallel1\nparallel2",
},
{
name: "label",
files: map[string]string{
".devcontainer/Dockerfile": fmt.Sprintf(`FROM %s
LABEL devcontainer.metadata='[{ \
"onCreateCommand": "echo create > /tmp/out", \
"updateContentCommand": ["sh", "-c", "echo update >> /tmp/out"], \
"postCreateCommand": "(echo -n postCreate. ; id -un) >> /tmp/out", \
"postStartCommand": { \
"parallel1": "echo parallel1 > /tmp/parallel1", \
"parallel2": ["sh", "-c", "echo parallel2 > /tmp/parallel2"] \
} \
}]'
USER nobody`, testImageAlpine),
},
outputCmd: "cat /tmp/out /tmp/parallel1 /tmp/parallel2",
expectOutput: "create\nupdate\npostCreate.nobody\nparallel1\nparallel2",
},
} {
tt := tt
t.Run(tt.name, func(t *testing.T) {
srv := gittest.CreateGitServer(t, gittest.Options{
Files: tt.files,
})
env := []string{
envbuilderEnv("GIT_URL", srv.URL),
}
if _, ok := tt.files[".devcontainer/devcontainer.json"]; !ok {
env = append(env, envbuilderEnv("DOCKERFILE_PATH", ".devcontainer/Dockerfile"))
}
ctr, err := runEnvbuilder(t, runOpts{env: env})
require.NoError(t, err, "failed to run envbuilder")
output := execContainer(t, ctr, tt.outputCmd)
require.Equal(t, tt.expectOutput, strings.TrimSpace(output))
})
}
}

func TestPostStartScript(t *testing.T) {
Expand Down

0 comments on commit fafad44

Please sign in to comment.