From f9ae03ff83b6f6a180c175623b7b9257834ef86c Mon Sep 17 00:00:00 2001 From: James Rasell Date: Mon, 24 Apr 2023 08:04:34 +0000 Subject: [PATCH] backport of commit f5898100f37a9f9dbb67eef3d62eb7dbb7093105 --- nomad/job_endpoint.go | 6 ++++++ nomad/job_endpoint_test.go | 28 ++++++++++++++++++++++++++++ 2 files changed, 34 insertions(+) diff --git a/nomad/job_endpoint.go b/nomad/job_endpoint.go index 9a41eb88b1e..1e9e3befec5 100644 --- a/nomad/job_endpoint.go +++ b/nomad/job_endpoint.go @@ -1049,9 +1049,15 @@ func (j *Job) Scale(args *structs.JobScaleRequest, reply *structs.JobRegisterRes return err } + // Perform validation on the job to ensure we have something that can + // actually be scaled. This logic can only exist here, as we need access + // to the job object. if job == nil { return structs.NewErrRPCCoded(404, fmt.Sprintf("job %q not found", args.JobID)) } + if job.Type == structs.JobTypeSystem { + return structs.NewErrRPCCoded(http.StatusBadRequest, `cannot scale jobs of type "system"`) + } // Since job is going to be mutated we must copy it since state store methods // return a shared pointer. diff --git a/nomad/job_endpoint_test.go b/nomad/job_endpoint_test.go index a9cd91f53c7..49beaa03209 100644 --- a/nomad/job_endpoint_test.go +++ b/nomad/job_endpoint_test.go @@ -7635,6 +7635,34 @@ func TestJobEndpoint_Scale_Priority(t *testing.T) { requireAssertion.NotZero(eval.ModifyTime) } +func TestJobEndpoint_Scale_SystemJob(t *testing.T) { + ci.Parallel(t) + + testServer, testServerCleanup := TestServer(t, nil) + defer testServerCleanup() + codec := rpcClient(t, testServer) + testutil.WaitForLeader(t, testServer.RPC) + state := testServer.fsm.State() + + mockSystemJob := mock.SystemJob() + must.NoError(t, state.UpsertJob(structs.MsgTypeTestSetup, 10, nil, mockSystemJob)) + + scaleReq := &structs.JobScaleRequest{ + JobID: mockSystemJob.ID, + Target: map[string]string{ + structs.ScalingTargetGroup: mockSystemJob.TaskGroups[0].Name, + }, + Count: pointer.Of(int64(13)), + WriteRequest: structs.WriteRequest{ + Region: DefaultRegion, + Namespace: mockSystemJob.Namespace, + }, + } + var resp structs.JobRegisterResponse + must.ErrorContains(t, msgpackrpc.CallWithCodec(codec, "Job.Scale", scaleReq, &resp), + `400,cannot scale jobs of type "system"`) +} + func TestJobEndpoint_InvalidCount(t *testing.T) { ci.Parallel(t) require := require.New(t)