From cc14776993ce1c10fa9b5060f9b687c768119eec Mon Sep 17 00:00:00 2001 From: Mahmood Ali Date: Wed, 14 Jul 2021 17:28:36 -0400 Subject: [PATCH] Merge pull request #10875 from hashicorp/b-namespace-flag-override cli: `-namespace` should override job namespace --- .changelog/10875.txt | 3 ++ command/agent/job_endpoint.go | 28 ++++++++++-- command/agent/job_endpoint_test.go | 71 ++++++++++++++++++++++++++++++ 3 files changed, 99 insertions(+), 3 deletions(-) create mode 100644 .changelog/10875.txt diff --git a/.changelog/10875.txt b/.changelog/10875.txt new file mode 100644 index 00000000000..67f6fab0dd5 --- /dev/null +++ b/.changelog/10875.txt @@ -0,0 +1,3 @@ +```release-note:bug +cli: Fixed a bug where `-namespace` flag was not respected for `job run` and `job plan` commands. +``` diff --git a/command/agent/job_endpoint.go b/command/agent/job_endpoint.go index c34426f998d..5feb8ac17c4 100644 --- a/command/agent/job_endpoint.go +++ b/command/agent/job_endpoint.go @@ -707,10 +707,9 @@ func (s *HTTPServer) apiJobAndRequestToStructs(job *api.Job, req *http.Request, AuthToken: apiReq.SecretID, } - queryRegion := req.URL.Query().Get("region") s.parseToken(req, &writeReq.AuthToken) - parseNamespace(req, &writeReq.Namespace) + queryRegion := req.URL.Query().Get("region") requestRegion, jobRegion := regionForJob( job, queryRegion, writeReq.Region, s.agent.config.Region, ) @@ -718,7 +717,11 @@ func (s *HTTPServer) apiJobAndRequestToStructs(job *api.Job, req *http.Request, sJob := ApiJobToStructJob(job) sJob.Region = jobRegion writeReq.Region = requestRegion - writeReq.Namespace = sJob.Namespace + + queryNamespace := req.URL.Query().Get("namespace") + namespace := namespaceForJob(job.Namespace, queryNamespace, writeReq.Namespace) + sJob.Namespace = namespace + writeReq.Namespace = namespace return sJob, writeReq } @@ -775,6 +778,25 @@ func regionForJob(job *api.Job, queryRegion, apiRegion, agentRegion string) (str return requestRegion, jobRegion } +func namespaceForJob(jobNamespace *string, queryNamespace, apiNamespace string) string { + + // Namespace in query param (-namespace flag) takes precedence. + if queryNamespace != "" { + return queryNamespace + } + + // Next the request body... + if apiNamespace != "" { + return apiNamespace + } + + if jobNamespace != nil && *jobNamespace != "" { + return *jobNamespace + } + + return structs.DefaultNamespace +} + func ApiJobToStructJob(job *api.Job) *structs.Job { job.Canonicalize() diff --git a/command/agent/job_endpoint_test.go b/command/agent/job_endpoint_test.go index 546b9600fcd..fe1b7155c56 100644 --- a/command/agent/job_endpoint_test.go +++ b/command/agent/job_endpoint_test.go @@ -1845,6 +1845,77 @@ func TestJobs_RegionForJob(t *testing.T) { } } +func TestJobs_NamespaceForJob(t *testing.T) { + t.Parallel() + + // test namespace for pointer inputs + ns := "dev" + + cases := []struct { + name string + job *api.Job + queryNamespace string + apiNamespace string + expected string + }{ + { + name: "no namespace provided", + job: &api.Job{}, + expected: structs.DefaultNamespace, + }, + + { + name: "jobspec has namespace", + job: &api.Job{Namespace: &ns}, + expected: "dev", + }, + + { + name: "-namespace flag overrides empty job namespace", + job: &api.Job{}, + queryNamespace: "prod", + expected: "prod", + }, + + { + name: "-namespace flag overrides job namespace", + job: &api.Job{Namespace: &ns}, + queryNamespace: "prod", + expected: "prod", + }, + + { + name: "-namespace flag overrides job namespace even if default", + job: &api.Job{Namespace: &ns}, + queryNamespace: structs.DefaultNamespace, + expected: structs.DefaultNamespace, + }, + + { + name: "API param overrides empty job namespace", + job: &api.Job{}, + apiNamespace: "prod", + expected: "prod", + }, + + { + name: "-namespace flag overrides API param", + job: &api.Job{Namespace: &ns}, + queryNamespace: "prod", + apiNamespace: "whatever", + expected: "prod", + }, + } + + for _, tc := range cases { + t.Run(tc.name, func(t *testing.T) { + require.Equal(t, tc.expected, + namespaceForJob(tc.job.Namespace, tc.queryNamespace, tc.apiNamespace), + ) + }) + } +} + func TestJobs_ApiJobToStructsJob(t *testing.T) { apiJob := &api.Job{ Stop: helper.BoolToPtr(true),