Skip to content

Commit 8ec6b2b

Browse files
committed
stage_executor: reset platform in systemcontext for stages
Every stage now has its own copy of systemcontext. On processing of every stage platform spec in systemcontext must be correctly reset. Closes: #5968 Signed-off-by: flouthoc <[email protected]>
1 parent 898fbb2 commit 8ec6b2b

File tree

5 files changed

+47
-16
lines changed

5 files changed

+47
-16
lines changed

imagebuildah/executor.go

+3
Original file line numberDiff line numberDiff line change
@@ -372,9 +372,12 @@ func newExecutor(logger *logrus.Logger, logPrefix string, store storage.Store, o
372372
// startStage creates a new stage executor that will be referenced whenever a
373373
// COPY or ADD statement uses a --from=NAME flag.
374374
func (b *Executor) startStage(ctx context.Context, stage *imagebuilder.Stage, stages imagebuilder.Stages, output string) *StageExecutor {
375+
// create a copy of systemContext for each stage executor.
376+
systemContext := *b.systemContext
375377
stageExec := &StageExecutor{
376378
ctx: ctx,
377379
executor: b,
380+
systemContext: &systemContext,
378381
log: b.log,
379382
index: stage.Position,
380383
stages: stages,

imagebuildah/stage_executor.go

+19-16
Original file line numberDiff line numberDiff line change
@@ -59,6 +59,7 @@ import (
5959
// name to the image that it produces.
6060
type StageExecutor struct {
6161
ctx context.Context
62+
systemContext *types.SystemContext
6263
executor *Executor
6364
log func(format string, args ...interface{})
6465
index int
@@ -584,8 +585,8 @@ func (s *StageExecutor) performCopy(excludes []string, copies ...imagebuilder.Co
584585
// The values for these next two fields are ultimately
585586
// based on command line flags with names that sound
586587
// much more generic.
587-
CertPath: s.executor.systemContext.DockerCertPath,
588-
InsecureSkipTLSVerify: s.executor.systemContext.DockerInsecureSkipTLSVerify,
588+
CertPath: s.systemContext.DockerCertPath,
589+
InsecureSkipTLSVerify: s.systemContext.DockerInsecureSkipTLSVerify,
589590
MaxRetries: s.executor.maxPullPushRetries,
590591
RetryDelay: s.executor.retryPullPushDelay,
591592
Parents: copy.Parents,
@@ -841,7 +842,7 @@ func (s *StageExecutor) Run(run imagebuilder.Run, config docker.Config) error {
841842
Stderr: s.executor.err,
842843
Stdin: stdin,
843844
Stdout: s.executor.out,
844-
SystemContext: s.executor.systemContext,
845+
SystemContext: s.systemContext,
845846
Terminal: buildah.WithoutTerminal,
846847
User: config.User,
847848
WorkingDir: config.WorkingDir,
@@ -966,19 +967,21 @@ func (s *StageExecutor) prepare(ctx context.Context, from string, initializeIBCo
966967
}
967968
}
968969

969-
builderSystemContext := s.executor.systemContext
970-
// get platform string from stage
971-
if stage.Builder.Platform != "" {
972-
os, arch, variant, err := parse.Platform(stage.Builder.Platform)
970+
builderSystemContext := s.systemContext
971+
// In a multi-stage build where `FROM --platform=<>` was used then we must
972+
// reset context for new stages so that new stages don't inherit unexpected
973+
// `--platform` from prior stages.
974+
if stage.Builder.Platform != "" || (len(s.stages) > 1 && ((builderSystemContext.ArchitectureChoice == "" && builderSystemContext.VariantChoice == "" && builderSystemContext.OSChoice == "") || (s.executor.skipUnusedStages == types.OptionalBoolFalse))) {
975+
imageOS, imageArch, imageVariant, err := parse.Platform(stage.Builder.Platform)
973976
if err != nil {
974977
return nil, fmt.Errorf("unable to parse platform %q: %w", stage.Builder.Platform, err)
975978
}
976-
if arch != "" || variant != "" {
977-
builderSystemContext.ArchitectureChoice = arch
978-
builderSystemContext.VariantChoice = variant
979+
if imageArch != "" || imageVariant != "" {
980+
builderSystemContext.ArchitectureChoice = imageArch
981+
builderSystemContext.VariantChoice = imageVariant
979982
}
980-
if os != "" {
981-
builderSystemContext.OSChoice = os
983+
if imageOS != "" {
984+
builderSystemContext.OSChoice = imageOS
982985
}
983986
}
984987

@@ -2058,7 +2061,7 @@ func (s *StageExecutor) tagExistingImage(ctx context.Context, cacheID, output st
20582061
return "", nil, err
20592062
}
20602063

2061-
policyContext, err := util.GetPolicyContext(s.executor.systemContext)
2064+
policyContext, err := util.GetPolicyContext(s.systemContext)
20622065
if err != nil {
20632066
return "", nil, err
20642067
}
@@ -2171,7 +2174,7 @@ func (s *StageExecutor) pushCache(ctx context.Context, src, cacheKey string) err
21712174
Compression: s.executor.compression,
21722175
SignaturePolicyPath: s.executor.signaturePolicyPath,
21732176
Store: s.executor.store,
2174-
SystemContext: s.executor.systemContext,
2177+
SystemContext: s.systemContext,
21752178
BlobDirectory: s.executor.blobDirectory,
21762179
SignBy: s.executor.signBy,
21772180
MaxRetries: s.executor.maxPullPushRetries,
@@ -2209,7 +2212,7 @@ func (s *StageExecutor) pullCache(ctx context.Context, cacheKey string) (referen
22092212
options := buildah.PullOptions{
22102213
SignaturePolicyPath: s.executor.signaturePolicyPath,
22112214
Store: s.executor.store,
2212-
SystemContext: s.executor.systemContext,
2215+
SystemContext: s.systemContext,
22132216
BlobDirectory: s.executor.blobDirectory,
22142217
MaxRetries: s.executor.maxPullPushRetries,
22152218
RetryDelay: s.executor.retryPullPushDelay,
@@ -2431,7 +2434,7 @@ func (s *StageExecutor) commit(ctx context.Context, createdBy string, emptyLayer
24312434
SignaturePolicyPath: s.executor.signaturePolicyPath,
24322435
ReportWriter: writer,
24332436
PreferredManifestType: s.executor.outputFormat,
2434-
SystemContext: s.executor.systemContext,
2437+
SystemContext: s.systemContext,
24352438
Squash: squash,
24362439
OmitHistory: s.executor.commonBuildOptions.OmitHistory,
24372440
EmptyLayer: emptyLayer,

tests/bud.bats

+15
Original file line numberDiff line numberDiff line change
@@ -6573,6 +6573,21 @@ _EOF
65736573
done
65746574
}
65756575

6576+
@test "build must reset platform for stages if needed" {
6577+
run_buildah info --format '{{.host.arch}}'
6578+
myarch="$output"
6579+
otherarch="arm64"
6580+
# just make sure that other arch is not equivalent to host arch
6581+
if [[ "$otherarch" == "$myarch" ]]; then
6582+
otherarch="amd64"
6583+
fi
6584+
run_buildah build $WITH_POLICY_JSON --build-arg FOREIGNARCH=$otherarch -f $BUDFILES/multiarch/Containerfile.reset-platform $BUDFILES/multiarch
6585+
run_buildah build $WITH_POLICY_JSON --build-arg TARGETPLATFORM=linux/$myarch --build-arg FOREIGNARCH=$otherarch -f $BUDFILES/multiarch/Containerfile.reset-platform $BUDFILES/multiarch
6586+
# Following line here attempts to cover edge case discussed here https://github.com/containers/buildah/pull/5971/files#r1970277844 but it's hard to reproduce this in a certain
6587+
# way since it can happen only when unused stage is picked in a particular order, following line ensures if this thing breaks then we luckily catch it in our CI.
6588+
run_buildah build $WITH_POLICY_JSON --jobs=2 --skip-unused-stages=true --build-arg FOREIGNARCH=$otherarch -f $BUDFILES/multiarch/Containerfile.unused-stage $BUDFILES/multiarch
6589+
}
6590+
65766591
# * Performs multi-stage build with label1=value1 and verifies
65776592
# * Relabels build with label1=value2 and verifies
65786593
# * Rebuild with label1=value1 and makes sure everything is used from cache
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
ARG FOREIGNARCH
2+
FROM --platform=linux/$FOREIGNARCH busybox AS foreign
3+
FROM busybox
4+
COPY --from=foreign /bin/busybox /bin/busybox.foreign
5+
RUN arch
6+
RUN ls -l /bin/busybox /bin/busybox.foreign
7+
RUN ! cmp /bin/busybox /bin/busybox.foreign
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
FROM busybox
2+
RUN arch
3+
FROM --platform=linux/$FOREIGNARCH busybox AS foreign

0 commit comments

Comments
 (0)