Skip to content

Commit 66b0cac

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 7fea494 commit 66b0cac

File tree

4 files changed

+41
-16
lines changed

4 files changed

+41
-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
@@ -571,8 +572,8 @@ func (s *StageExecutor) performCopy(excludes []string, copies ...imagebuilder.Co
571572
// The values for these next two fields are ultimately
572573
// based on command line flags with names that sound
573574
// much more generic.
574-
CertPath: s.executor.systemContext.DockerCertPath,
575-
InsecureSkipTLSVerify: s.executor.systemContext.DockerInsecureSkipTLSVerify,
575+
CertPath: s.systemContext.DockerCertPath,
576+
InsecureSkipTLSVerify: s.systemContext.DockerInsecureSkipTLSVerify,
576577
MaxRetries: s.executor.maxPullPushRetries,
577578
RetryDelay: s.executor.retryPullPushDelay,
578579
}
@@ -827,7 +828,7 @@ func (s *StageExecutor) Run(run imagebuilder.Run, config docker.Config) error {
827828
Stderr: s.executor.err,
828829
Stdin: stdin,
829830
Stdout: s.executor.out,
830-
SystemContext: s.executor.systemContext,
831+
SystemContext: s.systemContext,
831832
Terminal: buildah.WithoutTerminal,
832833
User: config.User,
833834
WorkingDir: config.WorkingDir,
@@ -952,19 +953,21 @@ func (s *StageExecutor) prepare(ctx context.Context, from string, initializeIBCo
952953
}
953954
}
954955

955-
builderSystemContext := s.executor.systemContext
956-
// get platform string from stage
957-
if stage.Builder.Platform != "" {
958-
os, arch, variant, err := parse.Platform(stage.Builder.Platform)
956+
builderSystemContext := s.systemContext
957+
// In a multi-stage build where `FROM --platform=<>` was used then we must
958+
// reset context for new stages so that new stages don't inherit unexpected
959+
// `--platform` from prior stages.
960+
if stage.Builder.Platform != "" || (stage.Position != 0 && builderSystemContext.ArchitectureChoice == "" && builderSystemContext.VariantChoice == "" && builderSystemContext.OSChoice == "") {
961+
imageOS, imageArch, imageVariant, err := parse.Platform(stage.Builder.Platform)
959962
if err != nil {
960963
return nil, fmt.Errorf("unable to parse platform %q: %w", stage.Builder.Platform, err)
961964
}
962-
if arch != "" || variant != "" {
963-
builderSystemContext.ArchitectureChoice = arch
964-
builderSystemContext.VariantChoice = variant
965+
if imageArch != "" || imageVariant != "" {
966+
builderSystemContext.ArchitectureChoice = imageArch
967+
builderSystemContext.VariantChoice = imageVariant
965968
}
966-
if os != "" {
967-
builderSystemContext.OSChoice = os
969+
if imageOS != "" {
970+
builderSystemContext.OSChoice = imageOS
968971
}
969972
}
970973

@@ -2041,7 +2044,7 @@ func (s *StageExecutor) tagExistingImage(ctx context.Context, cacheID, output st
20412044
return "", nil, err
20422045
}
20432046

2044-
policyContext, err := util.GetPolicyContext(s.executor.systemContext)
2047+
policyContext, err := util.GetPolicyContext(s.systemContext)
20452048
if err != nil {
20462049
return "", nil, err
20472050
}
@@ -2154,7 +2157,7 @@ func (s *StageExecutor) pushCache(ctx context.Context, src, cacheKey string) err
21542157
Compression: s.executor.compression,
21552158
SignaturePolicyPath: s.executor.signaturePolicyPath,
21562159
Store: s.executor.store,
2157-
SystemContext: s.executor.systemContext,
2160+
SystemContext: s.systemContext,
21582161
BlobDirectory: s.executor.blobDirectory,
21592162
SignBy: s.executor.signBy,
21602163
MaxRetries: s.executor.maxPullPushRetries,
@@ -2192,7 +2195,7 @@ func (s *StageExecutor) pullCache(ctx context.Context, cacheKey string) (referen
21922195
options := buildah.PullOptions{
21932196
SignaturePolicyPath: s.executor.signaturePolicyPath,
21942197
Store: s.executor.store,
2195-
SystemContext: s.executor.systemContext,
2198+
SystemContext: s.systemContext,
21962199
BlobDirectory: s.executor.blobDirectory,
21972200
MaxRetries: s.executor.maxPullPushRetries,
21982201
RetryDelay: s.executor.retryPullPushDelay,
@@ -2414,7 +2417,7 @@ func (s *StageExecutor) commit(ctx context.Context, createdBy string, emptyLayer
24142417
SignaturePolicyPath: s.executor.signaturePolicyPath,
24152418
ReportWriter: writer,
24162419
PreferredManifestType: s.executor.outputFormat,
2417-
SystemContext: s.executor.systemContext,
2420+
SystemContext: s.systemContext,
24182421
Squash: squash,
24192422
OmitHistory: s.executor.commonBuildOptions.OmitHistory,
24202423
EmptyLayer: emptyLayer,

tests/bud.bats

+12
Original file line numberDiff line numberDiff line change
@@ -6486,6 +6486,18 @@ _EOF
64866486
done
64876487
}
64886488

6489+
@test "build must reset platform for stages if needed" {
6490+
run_buildah info --format '{{.host.arch}}'
6491+
myarch="$output"
6492+
otherarch="arm64"
6493+
# just make sure that other arch is not equivalent to host arch
6494+
if [[ "$otherarch" == "$myarch" ]]; then
6495+
otherarch="amd64"
6496+
fi
6497+
run_buildah build $WITH_POLICY_JSON --build-arg FOREIGNARCH=$otherarch -f $BUDFILES/multiarch/Containerfile.reset-platform $BUDFILES/multiarch
6498+
run_buildah build $WITH_POLICY_JSON --platform linux/$myarch --build-arg FOREIGNARCH=$otherarch -f $BUDFILES/multiarch/Containerfile.reset-platform $BUDFILES/multiarch
6499+
}
6500+
64896501
# * Performs multi-stage build with label1=value1 and verifies
64906502
# * Relabels build with label1=value2 and verifies
64916503
# * 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

0 commit comments

Comments
 (0)