Skip to content

Commit

Permalink
fix(jenkins): Wrong Job name encoding in query params for Artifacts/P…
Browse files Browse the repository at this point in the history
…roperties (#4722)

* fix(jenkins): Wrong Job name encoding in query params for Artifacts/Properties

* fix(jenkins): Wrong Job name encoding in query params - added Wiremock tests

* fix(jenkins): Wrong Job name encoding in query params - fix Wiremock tests imports
  • Loading branch information
christosarvanitis authored May 9, 2024
1 parent b360ad5 commit b2dba59
Show file tree
Hide file tree
Showing 3 changed files with 242 additions and 2 deletions.
1 change: 1 addition & 0 deletions orca-igor/orca-igor.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@ dependencies {
testImplementation(project(":orca-test-groovy"))
testImplementation("org.springframework:spring-test")
testImplementation("io.spinnaker.fiat:fiat-core:$fiatVersion")
testImplementation("com.github.tomakehurst:wiremock-jre8-standalone")
}

sourceSets {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,7 @@ Map<String, Object> getPropertyFileWithJobAsQueryParam(
@Path("buildNumber") Integer buildNumber,
@Path("fileName") String fileName,
@Path("master") String master,
@Query(value = "job") String job);
@Query(encodeValue = false, value = "job") String job);

@GET("/{repoType}/{projectKey}/{repositorySlug}/compareCommits")
List compareCommits(
Expand All @@ -105,7 +105,7 @@ List<Artifact> getArtifactsWithJobAsQueryParam(
@Path("buildNumber") Integer buildNumber,
@Query("propertyFile") String propertyFile,
@Path("master") String master,
@Query(value = "job") String job);
@Query(value = "job", encodeValue = false) String job);

@POST("/gcb/builds/create/{account}")
GoogleCloudBuild createGoogleCloudBuild(
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,239 @@
/*
* Copyright 2024 Harness, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package com.netflix.spinnaker.orca.igor

import com.fasterxml.jackson.databind.ObjectMapper
import com.github.tomakehurst.wiremock.WireMockServer
import com.jakewharton.retrofit.Ok3Client
import com.netflix.spinnaker.kork.artifacts.model.Artifact
import retrofit.RestAdapter
import retrofit.client.Response
import retrofit.converter.JacksonConverter
import spock.lang.Specification
import spock.lang.Subject

import static com.github.tomakehurst.wiremock.client.WireMock.configureFor
import static com.github.tomakehurst.wiremock.client.WireMock.stubFor
import static com.github.tomakehurst.wiremock.client.WireMock.okJson
import static com.github.tomakehurst.wiremock.client.WireMock.urlEqualTo
import static com.github.tomakehurst.wiremock.client.WireMock.aResponse
import static com.github.tomakehurst.wiremock.client.WireMock.get
import static com.github.tomakehurst.wiremock.client.WireMock.put
import static retrofit.Endpoints.newFixedEndpoint


class BuildServiceSpecMock extends Specification{

static WireMockServer wireMockServer = new WireMockServer(0)

@Subject BuildService buildService

private static final MASTER = 'MASTER'
private static final BUILD_NUMBER = 123
private static final JOB_NAME = "name/with/slashes and spaces"
private static final JOB_NAME_ENCODED = "name/with/slashes%20and%20spaces"
private static final PARAMS = ['key': 'value']
private static final FILENAME = 'build.properties'

def setupSpec() {
wireMockServer.start()
configureFor(wireMockServer.port())
}
def mapper = new ObjectMapper()
IgorService igorService
def setup() {
igorService = new RestAdapter.Builder()
.setEndpoint(newFixedEndpoint(wireMockServer.url("/")))
.setClient(new Ok3Client())
.setConverter(new JacksonConverter(mapper))
.build()
.create(IgorService)
buildService = new BuildService(igorService, new IgorFeatureFlagProperties(jobNameAsQueryParameter: false))
}

def cleanupSpec() {
wireMockServer.stop()
}

def "build starts a Jenkins job"() {
String queryMap = PARAMS.collect { k, v -> "$k=$v" }.join('&')
String uriPath = "/masters/$MASTER/jobs/$JOB_NAME_ENCODED?$queryMap"
stubFor(put(uriPath)
.willReturn(
aResponse()
.withStatus(200)
.withBody(BUILD_NUMBER.toString())
)
)

when:
Response response = buildService.build(MASTER, JOB_NAME, PARAMS)

then:
response.status == 200
}



def "getBuild returns the build details - jobNameAsQueryParameter:false"() {
given:
String uriPath = "/builds/status/$BUILD_NUMBER/$MASTER/$JOB_NAME_ENCODED"
stubFor(get(urlEqualTo(uriPath))
.willReturn(okJson(getBuildResponse())))

when:
Map<String, Object> response = buildService.getBuild(BUILD_NUMBER,MASTER,JOB_NAME)
then:
response.get("name") == JOB_NAME
}

def "getBuild returns the build details - jobNameAsQueryParameter:true"() {
given:
String uriPath = "/builds/status/$BUILD_NUMBER/$MASTER?job=$JOB_NAME_ENCODED"
IgorFeatureFlagProperties igorFeatureFlagProperties = new IgorFeatureFlagProperties()
igorFeatureFlagProperties.setJobNameAsQueryParameter(true)
buildService = new BuildService(igorService, igorFeatureFlagProperties)
stubFor(get(urlEqualTo(uriPath))
.willReturn(okJson(getBuildResponse())))

when:
Map<String, Object> response = buildService.getBuild(BUILD_NUMBER,MASTER,JOB_NAME)
then:
response.get("name") == JOB_NAME
}

def "getPropertyFile returns a property file - jobNameAsQueryParameter:false"() {
given:
String uriPath = "/builds/properties/$BUILD_NUMBER/$FILENAME/$MASTER/$JOB_NAME_ENCODED"
stubFor(get(urlEqualTo(uriPath))
.willReturn(
okJson(getBuildPropertiesResponse())
)
)

when:
Map<String, Object> response = buildService.getPropertyFile(BUILD_NUMBER, FILENAME, MASTER, JOB_NAME)

then:
response.get("randomKey") == "randomValue"
}

def "getPropertyFile returns a property file - jobNameAsQueryParameter:true"() {
given:
String uriPath = "/builds/properties/$BUILD_NUMBER/$FILENAME/$MASTER?job=$JOB_NAME_ENCODED"
IgorFeatureFlagProperties igorFeatureFlagProperties = new IgorFeatureFlagProperties()
igorFeatureFlagProperties.setJobNameAsQueryParameter(true)
buildService = new BuildService(igorService, igorFeatureFlagProperties)
stubFor(get(urlEqualTo(uriPath))
.willReturn(
okJson(getBuildPropertiesResponse())
)
)

when:
Map<String, Object> response = buildService.getPropertyFile(BUILD_NUMBER, FILENAME, MASTER, JOB_NAME)

then:
response.get("randomKey") == "randomValue"

}

def "getArtifacts returns build artifacts - jobNameAsQueryParameter:false"() {
given:
String uriPath = "/builds/artifacts/$BUILD_NUMBER/$MASTER/$JOB_NAME_ENCODED?propertyFile=$FILENAME"
stubFor(get(urlEqualTo(uriPath))
.willReturn(
okJson(getArtifactsResponse())
)
)

when:
List<Artifact> response = buildService.getArtifacts(BUILD_NUMBER, FILENAME, MASTER, JOB_NAME)

then:
response.toString() == "[]"
}

def "getArtifacts returns build artifacts - jobNameAsQueryParameter:true"() {
given:
String uriPath = "/builds/artifacts/$BUILD_NUMBER/$MASTER?propertyFile=$FILENAME&job=$JOB_NAME_ENCODED"
IgorFeatureFlagProperties igorFeatureFlagProperties = new IgorFeatureFlagProperties()
igorFeatureFlagProperties.setJobNameAsQueryParameter(true)
buildService = new BuildService(igorService, igorFeatureFlagProperties)
stubFor(get(urlEqualTo(uriPath))
.willReturn(
okJson(getArtifactsResponse())
)
)

when:
List<Artifact> response = buildService.getArtifacts(BUILD_NUMBER, FILENAME, MASTER, JOB_NAME)

then:
response.toString() == "[]"
}


private static String getBuildResponse() {
return "{\n" +
" \"building\": false,\n" +
" \"fullDisplayName\": \"$JOB_NAME #$BUILD_NUMBER\",\n" +
" \"name\": \"$JOB_NAME\",\n" +
" \"number\": $BUILD_NUMBER,\n" +
" \"duration\": 778,\n" +
" \"timestamp\": \"1714730108196\",\n" +
" \"result\": \"SUCCESS\",\n" +
" \"artifacts\": [\n" +
" {\n" +
" \"fileName\": \"$FILENAME\",\n" +
" \"displayPath\": \"$FILENAME\",\n" +
" \"relativePath\": \"$FILENAME\",\n" +
" \"reference\": \"$FILENAME\",\n" +
" \"name\": \"$JOB_NAME\",\n" +
" \"type\": \"jenkins/file\",\n" +
" \"version\": \"6\",\n" +
" \"decorated\": false\n" +
" }\n" +
" ],\n" +
" \"testResults\": [\n" +
" {\n" +
" \"failCount\": 0,\n" +
" \"skipCount\": 0,\n" +
" \"totalCount\": 0,\n" +
" \"urlName\": null,\n" +
" \"cause\": {\n" +
" \"shortDescription\": \"Started by user Spinnaker\",\n" +
" \"upstreamBuild\": null,\n" +
" \"upstreamProject\": null,\n" +
" \"upstreamUrl\": null\n" +
" }\n" +
" }\n" +
" ],\n" +
" \"url\": \"https://jenkins.somedomain.com/job/$JOB_NAME_ENCODED/$BUILD_NUMBER/\"\n" +
"}"
}

private static String getBuildPropertiesResponse() {
return "{\"randomKey\":\"randomValue\"}"
}

private static String getArtifactsResponse() {
return "[]"
}


}

0 comments on commit b2dba59

Please sign in to comment.