|
| 1 | +package io.jenkins.plugins.casc; |
| 2 | + |
| 3 | +import static org.hamcrest.MatcherAssert.assertThat; |
| 4 | +import static org.hamcrest.Matchers.containsString; |
| 5 | +import static org.hamcrest.Matchers.is; |
| 6 | + |
| 7 | +import io.jenkins.plugins.casc.misc.JenkinsConfiguredWithCodeRule; |
| 8 | +import io.jenkins.plugins.casc.misc.junit.jupiter.WithJenkinsConfiguredWithCode; |
| 9 | +import java.net.URL; |
| 10 | +import java.nio.charset.StandardCharsets; |
| 11 | +import java.text.MessageFormat; |
| 12 | +import net.sf.json.JSONArray; |
| 13 | +import org.htmlunit.HttpMethod; |
| 14 | +import org.htmlunit.WebClient; |
| 15 | +import org.htmlunit.WebRequest; |
| 16 | +import org.htmlunit.WebResponse; |
| 17 | +import org.junit.jupiter.api.Test; |
| 18 | +import org.jvnet.hudson.test.Issue; |
| 19 | + |
| 20 | +/** |
| 21 | + * Tests for YAML validation in the Configuration-as-Code endpoint. |
| 22 | + */ |
| 23 | +@WithJenkinsConfiguredWithCode |
| 24 | +public class YamlValidationTest { |
| 25 | + |
| 26 | + /** |
| 27 | + * Tests that valid YAML returns an empty array response. |
| 28 | + */ |
| 29 | + @Test |
| 30 | + void validYaml(JenkinsConfiguredWithCodeRule j) throws Exception { |
| 31 | + // Disable CSRF protection to simplify testing |
| 32 | + j.jenkins.setCrumbIssuer(null); |
| 33 | + |
| 34 | + WebResponse response = performYamlValidation(j, "jenkins:\n systemMessage: \"Hello from test\""); |
| 35 | + |
| 36 | + assertThat(response.getStatusCode(), is(200)); |
| 37 | + assertThat(response.getContentType(), is("application/json")); |
| 38 | + |
| 39 | + JSONArray jsonResponse = JSONArray.fromObject(response.getContentAsString()); |
| 40 | + assertThat(jsonResponse.size(), is(0)); |
| 41 | + } |
| 42 | + |
| 43 | + /** |
| 44 | + * Tests that YAML with incorrect indentation returns proper JSON response. |
| 45 | + */ |
| 46 | + @Test |
| 47 | + @Issue("2628") |
| 48 | + void invalidIndentation(JenkinsConfiguredWithCodeRule j) throws Exception { |
| 49 | + j.jenkins.setCrumbIssuer(null); |
| 50 | + |
| 51 | + WebResponse response = performYamlValidation(j, "jenkins:\nsystemMessage: \"Bad indentation\""); |
| 52 | + |
| 53 | + assertThat(response.getContentType(), is("application/json")); |
| 54 | + |
| 55 | + // Verify the response contains valid JSON regardless of status code |
| 56 | + String responseString = response.getContentAsString(); |
| 57 | + JSONArray jsonResponse = JSONArray.fromObject(responseString); |
| 58 | + |
| 59 | + // The response should be a non-empty JSON array |
| 60 | + assertThat("Should return error details", jsonResponse.size() > 0, is(true)); |
| 61 | + } |
| 62 | + |
| 63 | + /** |
| 64 | + * Tests that YAML with invalid structure returns proper JSON response. |
| 65 | + */ |
| 66 | + @Test |
| 67 | + @Issue("2628") |
| 68 | + void invalidStructure(JenkinsConfiguredWithCodeRule j) throws Exception { |
| 69 | + j.jenkins.setCrumbIssuer(null); |
| 70 | + |
| 71 | + WebResponse response = performYamlValidation(j, "jenkins:\n numExecutors: {"); |
| 72 | + |
| 73 | + assertThat(response.getContentType(), is("application/json")); |
| 74 | + |
| 75 | + // Verify the response contains valid JSON regardless of status code |
| 76 | + String responseString = response.getContentAsString(); |
| 77 | + JSONArray jsonResponse = JSONArray.fromObject(responseString); |
| 78 | + |
| 79 | + // The response should be a non-empty JSON array |
| 80 | + assertThat("Should return error details", jsonResponse.size() > 0, is(true)); |
| 81 | + } |
| 82 | + |
| 83 | + /** |
| 84 | + * Tests that YAML with unknown property returns proper JSON response. |
| 85 | + */ |
| 86 | + @Test |
| 87 | + @Issue("2628") |
| 88 | + void unknownProperty(JenkinsConfiguredWithCodeRule j) throws Exception { |
| 89 | + j.jenkins.setCrumbIssuer(null); |
| 90 | + |
| 91 | + WebResponse response = performYamlValidation( |
| 92 | + j, |
| 93 | + "jenkins:\n nonExistentProperty: \"This property doesn't exist\"\n systemMessage: \"Valid property\""); |
| 94 | + |
| 95 | + assertThat(response.getContentType(), is("application/json")); |
| 96 | + |
| 97 | + // Verify the response contains valid JSON regardless of status code |
| 98 | + String responseString = response.getContentAsString(); |
| 99 | + JSONArray jsonResponse = JSONArray.fromObject(responseString); |
| 100 | + |
| 101 | + // The response should be a non-empty JSON array with details about the unknown property |
| 102 | + assertThat("Should return error details", jsonResponse.size() > 0, is(true)); |
| 103 | + assertThat(responseString, containsString("nonExistentProperty")); |
| 104 | + } |
| 105 | + |
| 106 | + /** |
| 107 | + * Helper method to perform YAML validation. |
| 108 | + */ |
| 109 | + private WebResponse performYamlValidation(JenkinsConfiguredWithCodeRule j, String yamlContent) throws Exception { |
| 110 | + URL apiURL = new URL(MessageFormat.format( |
| 111 | + "{0}configuration-as-code/check", j.getURL().toString())); |
| 112 | + WebRequest request = new WebRequest(apiURL, HttpMethod.POST); |
| 113 | + request.setCharset(StandardCharsets.UTF_8); |
| 114 | + request.setRequestBody(yamlContent); |
| 115 | + |
| 116 | + WebClient client = j.createWebClient(); |
| 117 | + // Don't throw exceptions for error status codes |
| 118 | + client.getOptions().setThrowExceptionOnFailingStatusCode(false); |
| 119 | + |
| 120 | + return client.getPage(request).getWebResponse(); |
| 121 | + } |
| 122 | +} |
0 commit comments