Insert tips as comments in the YAML snippet

Include a few tips as code comments inserted in generated YAML snippets.
parent 4db43ced
...@@ -18,6 +18,7 @@ import DynamicFields from '../../components/dynamic_fields.vue'; ...@@ -18,6 +18,7 @@ import DynamicFields from '../../components/dynamic_fields.vue';
import FormInput from '../../components/form_input.vue'; import FormInput from '../../components/form_input.vue';
import { SCAN_MODES, CONFIGURATION_SNIPPET_MODAL_ID } from '../constants'; import { SCAN_MODES, CONFIGURATION_SNIPPET_MODAL_ID } from '../constants';
import apiFuzzingCiConfigurationCreate from '../graphql/api_fuzzing_ci_configuration_create.mutation.graphql'; import apiFuzzingCiConfigurationCreate from '../graphql/api_fuzzing_ci_configuration_create.mutation.graphql';
import { insertTips } from '../utils';
import ConfigurationSnippetModal from './configuration_snippet_modal.vue'; import ConfigurationSnippetModal from './configuration_snippet_modal.vue';
export default { export default {
...@@ -155,6 +156,30 @@ export default { ...@@ -155,6 +156,30 @@ export default {
} }
return fields.some(({ value }) => isEmptyValue(value)); return fields.some(({ value }) => isEmptyValue(value));
}, },
configurationYamlWithTips() {
if (!this.configurationYaml) {
return '';
}
return insertTips(this.configurationYaml, [
{
tip: s__('APIFuzzing|Tip: Insert this part below all stages'),
// eslint-disable-next-line @gitlab/require-i18n-strings
token: 'stages:',
},
{
tip: s__('APIFuzzing|Tip: Insert this part below all include'),
// eslint-disable-next-line @gitlab/require-i18n-strings
token: 'include:',
},
{
tip: s__(
'APIFuzzing|Tip: Insert the following variables anywhere below stages and include',
),
// eslint-disable-next-line @gitlab/require-i18n-strings
token: 'variables:',
},
]);
},
}, },
methods: { methods: {
async onSubmit() { async onSubmit() {
...@@ -303,7 +328,7 @@ export default { ...@@ -303,7 +328,7 @@ export default {
<configuration-snippet-modal <configuration-snippet-modal
:ref="$options.CONFIGURATION_SNIPPET_MODAL_ID" :ref="$options.CONFIGURATION_SNIPPET_MODAL_ID"
:ci-yaml-edit-url="ciYamlEditPath" :ci-yaml-edit-url="ciYamlEditPath"
:yaml="configurationYaml" :yaml="configurationYamlWithTips"
/> />
</form> </form>
</template> </template>
/* eslint-disable @gitlab/require-i18n-strings */
import { isString } from 'lodash';
export const insertTip = ({ snippet, tip, token }) => {
if (!isString(snippet)) {
throw new Error('snippet must be a string');
}
if (!isString(tip)) {
throw new Error('tip must be a string');
}
if (!isString(token)) {
throw new Error('token must be a string');
}
return snippet.replace(token, `# ${tip}\n${token}`);
};
export const insertTips = (snippet, tips = []) =>
tips.reduce(
(snippetWithTips, { tip, token }) => insertTip({ snippet: snippetWithTips, tip, token }),
snippet,
);
---
title: Insert tips as comments in the YAML snippet
merge_request: 58656
author:
type: changed
...@@ -241,9 +241,16 @@ describe('EE - ApiFuzzingConfigurationForm', () => { ...@@ -241,9 +241,16 @@ describe('EE - ApiFuzzingConfigurationForm', () => {
ciYamlEditUrl: ciYamlEditUrl:
createApiFuzzingConfigurationMutationResponse.data.apiFuzzingCiConfigurationCreate createApiFuzzingConfigurationMutationResponse.data.apiFuzzingCiConfigurationCreate
.gitlabCiYamlEditPath, .gitlabCiYamlEditPath,
yaml: yaml: `---
createApiFuzzingConfigurationMutationResponse.data.apiFuzzingCiConfigurationCreate # Tip: Insert this part below all stages
.configurationYaml, stages:
- fuzz
# Tip: Insert this part below all include
include:
- template: template.gitlab-ci.yml
# Tip: Insert the following variables anywhere below stages and include
variables:
- FOO: bar`,
}); });
}); });
......
...@@ -43,7 +43,13 @@ export const apiFuzzingConfigurationQueryResponse = { ...@@ -43,7 +43,13 @@ export const apiFuzzingConfigurationQueryResponse = {
export const createApiFuzzingConfigurationMutationResponse = { export const createApiFuzzingConfigurationMutationResponse = {
data: { data: {
apiFuzzingCiConfigurationCreate: { apiFuzzingCiConfigurationCreate: {
configurationYaml: 'yaml snippet', configurationYaml: `---
stages:
- fuzz
include:
- template: template.gitlab-ci.yml
variables:
- FOO: bar`,
gitlabCiYamlEditPath: '/ci/editor', gitlabCiYamlEditPath: '/ci/editor',
errors: [], errors: [],
__typename: 'ApiFuzzingCiConfiguration', __typename: 'ApiFuzzingCiConfiguration',
......
import { insertTip, insertTips } from 'ee/security_configuration/api_fuzzing/utils';
const nonStringValues = [1, {}, null];
describe('insertTip', () => {
describe.each(['snippet', 'tip', 'token'])('throws when %s is', (arg) => {
const validValues = {
snippet: 'snippet',
tip: 'tip',
token: 'token',
};
it.each(nonStringValues)('%s', (value) => {
expect(() => {
insertTip({ ...validValues, [arg]: value });
}).toThrowError(`${arg} must be a string`);
});
});
it('returns snippet as is if token can not be found', () => {
const snippet = 'some code snippet';
expect(
insertTip({
snippet,
token: 'ghost',
tip: 'a very helpful tip',
}),
).toBe(snippet);
});
const tip = 'a very helpful tip';
it.each`
snippet | token | expected
${'some code snippet'} | ${'code'} | ${`some # ${tip}\ncode snippet`}
${'some code snippet'} | ${'some'} | ${`# ${tip}\nsome code snippet`}
${'some code snippet'} | ${'e'} | ${`som# ${tip}\ne code snippet`}
`('inserts the tip on the line before the first found token', ({ snippet, token, expected }) => {
expect(
insertTip({
snippet,
token,
tip,
}),
).toBe(expected);
});
});
describe('insertTips', () => {
const validTips = [
{
tip: 'Tip 1',
token: 'stages:',
},
{
tip: 'Tip 2',
token: 'variables:',
},
];
it.each(nonStringValues)('throws if snippet is not a string', (snippet) => {
expect(() => {
insertTips(snippet, validTips);
}).toThrowError('snippet must be a string');
});
describe.each(['tip', 'token'])('throws if %s', (prop) => {
it.each(nonStringValues)('is %s', (value) => {
expect(() => {
insertTips('some code snippet', [
{
...validTips[0],
[prop]: value,
},
]);
}).toThrowError(`${prop} must be a string`);
});
});
it('returns snippet as is if token can not be found', () => {
const snippet = 'some code snippet';
expect(insertTips(snippet, validTips)).toBe(snippet);
});
it('returns the snippet with properly inserted tips', () => {
const snippet = `
---
stages:
- fuzz
include:
- template: template.gitlab-cy.yml
variables:
- FOO: bar`;
const expected = `
---
# Tip 1
stages:
- fuzz
include:
- template: template.gitlab-cy.yml
# Tip 2
variables:
- FOO: bar`;
expect(insertTips(snippet, validTips)).toBe(expected);
});
});
...@@ -1560,6 +1560,15 @@ msgstr "" ...@@ -1560,6 +1560,15 @@ msgstr ""
msgid "APIFuzzing|There are two ways to perform scans." msgid "APIFuzzing|There are two ways to perform scans."
msgstr "" msgstr ""
msgid "APIFuzzing|Tip: Insert the following variables anywhere below stages and include"
msgstr ""
msgid "APIFuzzing|Tip: Insert this part below all include"
msgstr ""
msgid "APIFuzzing|Tip: Insert this part below all stages"
msgstr ""
msgid "APIFuzzing|To prevent a security leak, authentication info must be added as a %{ciVariablesLinkStart}CI variable%{ciVariablesLinkEnd}. A user with maintainer access rights can manage CI variables in the %{ciSettingsLinkStart}Settings%{ciSettingsLinkEnd} area. We detected that you are not a maintainer. Commit your changes and assign them to a maintainer to update the credentials before merging." msgid "APIFuzzing|To prevent a security leak, authentication info must be added as a %{ciVariablesLinkStart}CI variable%{ciVariablesLinkEnd}. A user with maintainer access rights can manage CI variables in the %{ciSettingsLinkStart}Settings%{ciSettingsLinkEnd} area. We detected that you are not a maintainer. Commit your changes and assign them to a maintainer to update the credentials before merging."
msgstr "" msgstr ""
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment