Commit cade38e5 authored by Matt Kasa's avatar Matt Kasa

Show variable expansion info when adding CI variable

Changelog: fixed

Closes https://gitlab.com/gitlab-org/gitlab/-/issues/297519
parent 32613a4b
...@@ -24,6 +24,7 @@ import { ...@@ -24,6 +24,7 @@ import {
ADD_CI_VARIABLE_MODAL_ID, ADD_CI_VARIABLE_MODAL_ID,
AWS_TIP_DISMISSED_COOKIE_NAME, AWS_TIP_DISMISSED_COOKIE_NAME,
AWS_TIP_MESSAGE, AWS_TIP_MESSAGE,
CONTAINS_VARIABLE_REFERENCE_MESSAGE,
} from '../constants'; } from '../constants';
import CiEnvironmentsDropdown from './ci_environments_dropdown.vue'; import CiEnvironmentsDropdown from './ci_environments_dropdown.vue';
import { awsTokens, awsTokenList } from './ci_variable_autocomplete_tokens'; import { awsTokens, awsTokenList } from './ci_variable_autocomplete_tokens';
...@@ -33,6 +34,7 @@ export default { ...@@ -33,6 +34,7 @@ export default {
tokens: awsTokens, tokens: awsTokens,
tokenList: awsTokenList, tokenList: awsTokenList,
awsTipMessage: AWS_TIP_MESSAGE, awsTipMessage: AWS_TIP_MESSAGE,
containsVariableReferenceMessage: CONTAINS_VARIABLE_REFERENCE_MESSAGE,
components: { components: {
CiEnvironmentsDropdown, CiEnvironmentsDropdown,
GlAlert, GlAlert,
...@@ -70,6 +72,7 @@ export default { ...@@ -70,6 +72,7 @@ export default {
'awsTipDeployLink', 'awsTipDeployLink',
'awsTipCommandsLink', 'awsTipCommandsLink',
'awsTipLearnLink', 'awsTipLearnLink',
'containsVariableReferenceLink',
'protectedEnvironmentVariablesLink', 'protectedEnvironmentVariablesLink',
'maskedEnvironmentVariablesLink', 'maskedEnvironmentVariablesLink',
]), ]),
...@@ -99,6 +102,10 @@ export default { ...@@ -99,6 +102,10 @@ export default {
const regex = RegExp(this.maskableRegex); const regex = RegExp(this.maskableRegex);
return regex.test(this.variable.secret_value); return regex.test(this.variable.secret_value);
}, },
containsVariableReference() {
const regex = RegExp(/\$/);
return regex.test(this.variable.secret_value);
},
displayMaskedError() { displayMaskedError() {
return !this.canMask && this.variable.masked; return !this.canMask && this.variable.masked;
}, },
...@@ -328,6 +335,22 @@ export default { ...@@ -328,6 +335,22 @@ export default {
</div> </div>
</gl-alert> </gl-alert>
</gl-collapse> </gl-collapse>
<gl-alert
v-if="containsVariableReference"
:title="__('Value may contain a variable reference')"
:dismissible="false"
variant="warning"
data-testid="contains-variable-reference"
>
<gl-sprintf :message="$options.containsVariableReferenceMessage">
<template #code="{ content }">
<code>{{ content }}</code>
</template>
<template #docsLink="{ content }">
<gl-link :href="containsVariableReferenceLink" target="_blank">{{ content }}</gl-link>
</template>
</gl-sprintf>
</gl-alert>
<template #modal-footer> <template #modal-footer>
<gl-button @click="hideModal">{{ __('Cancel') }}</gl-button> <gl-button @click="hideModal">{{ __('Cancel') }}</gl-button>
<gl-button <gl-button
......
...@@ -24,3 +24,7 @@ export const AWS_ACCESS_KEY_ID = 'AWS_ACCESS_KEY_ID'; ...@@ -24,3 +24,7 @@ export const AWS_ACCESS_KEY_ID = 'AWS_ACCESS_KEY_ID';
export const AWS_DEFAULT_REGION = 'AWS_DEFAULT_REGION'; export const AWS_DEFAULT_REGION = 'AWS_DEFAULT_REGION';
export const AWS_SECRET_ACCESS_KEY = 'AWS_SECRET_ACCESS_KEY'; export const AWS_SECRET_ACCESS_KEY = 'AWS_SECRET_ACCESS_KEY';
export const AWS_TOKEN_CONSTANTS = [AWS_ACCESS_KEY_ID, AWS_DEFAULT_REGION, AWS_SECRET_ACCESS_KEY]; export const AWS_TOKEN_CONSTANTS = [AWS_ACCESS_KEY_ID, AWS_DEFAULT_REGION, AWS_SECRET_ACCESS_KEY];
export const CONTAINS_VARIABLE_REFERENCE_MESSAGE = __(
'Variable references indicated by %{codeStart}$%{codeEnd} may be expanded. If this is not what you want, consider %{docsLinkStart}using a workaround to prevent expansion%{docsLinkEnd}.',
);
...@@ -14,6 +14,7 @@ const mountCiVariableListApp = (containerEl) => { ...@@ -14,6 +14,7 @@ const mountCiVariableListApp = (containerEl) => {
awsTipDeployLink, awsTipDeployLink,
awsTipCommandsLink, awsTipCommandsLink,
awsTipLearnLink, awsTipLearnLink,
containsVariableReferenceLink,
protectedEnvironmentVariablesLink, protectedEnvironmentVariablesLink,
maskedEnvironmentVariablesLink, maskedEnvironmentVariablesLink,
} = containerEl.dataset; } = containerEl.dataset;
...@@ -30,6 +31,7 @@ const mountCiVariableListApp = (containerEl) => { ...@@ -30,6 +31,7 @@ const mountCiVariableListApp = (containerEl) => {
awsTipDeployLink, awsTipDeployLink,
awsTipCommandsLink, awsTipCommandsLink,
awsTipLearnLink, awsTipLearnLink,
containsVariableReferenceLink,
protectedEnvironmentVariablesLink, protectedEnvironmentVariablesLink,
maskedEnvironmentVariablesLink, maskedEnvironmentVariablesLink,
}); });
......
...@@ -16,6 +16,7 @@ ...@@ -16,6 +16,7 @@
aws_tip_deploy_link: help_page_path('ci/cloud_deployment/index.md', anchor: 'deploy-your-application-to-the-aws-elastic-container-service-ecs'), aws_tip_deploy_link: help_page_path('ci/cloud_deployment/index.md', anchor: 'deploy-your-application-to-the-aws-elastic-container-service-ecs'),
aws_tip_commands_link: help_page_path('ci/cloud_deployment/index.md', anchor: 'run-aws-commands-from-gitlab-cicd'), aws_tip_commands_link: help_page_path('ci/cloud_deployment/index.md', anchor: 'run-aws-commands-from-gitlab-cicd'),
aws_tip_learn_link: help_page_path('ci/cloud_deployment/index.md', anchor: 'aws'), aws_tip_learn_link: help_page_path('ci/cloud_deployment/index.md', anchor: 'aws'),
contains_variable_reference_link: help_page_path('ci/variables/index', anchor: 'troubleshooting-variables-containing-references'),
protected_environment_variables_link: help_page_path('ci/variables/index', anchor: 'protect-a-cicd-variable'), protected_environment_variables_link: help_page_path('ci/variables/index', anchor: 'protect-a-cicd-variable'),
masked_environment_variables_link: help_page_path('ci/variables/index', anchor: 'mask-a-cicd-variable'), masked_environment_variables_link: help_page_path('ci/variables/index', anchor: 'mask-a-cicd-variable'),
} } } }
......
...@@ -368,6 +368,26 @@ WARNING: ...@@ -368,6 +368,26 @@ WARNING:
When you store credentials, there are [security implications](#cicd-variable-security). When you store credentials, there are [security implications](#cicd-variable-security).
If you use AWS keys for example, follow the [Best practices for managing AWS access keys](https://docs.aws.amazon.com/general/latest/gr/aws-access-keys-best-practices.html). If you use AWS keys for example, follow the [Best practices for managing AWS access keys](https://docs.aws.amazon.com/general/latest/gr/aws-access-keys-best-practices.html).
### Troubleshooting variables containing references
When a variable value contains a reference indicated by `$`, it may be expanded
which can lead to unexpected values. Use `$$` to ignore a variable name inside
another variable:
```plaintext
SOME$$VALUE
```
Another workaround is to add a new variable set to the one that contains a
reference:
```yaml
variables:
NEWVAR: $MYVAR
script:
- echo $NEWVAR # outputs SOME$VALUE
```
## Use CI/CD variables in job scripts ## Use CI/CD variables in job scripts
All CI/CD variables are set as environment variables in the job's environment. All CI/CD variables are set as environment variables in the job's environment.
......
...@@ -36355,6 +36355,9 @@ msgstr "" ...@@ -36355,6 +36355,9 @@ msgstr ""
msgid "Value Stream Analytics gives an overview of how much time it takes to go from idea to production in your project." msgid "Value Stream Analytics gives an overview of how much time it takes to go from idea to production in your project."
msgstr "" msgstr ""
msgid "Value may contain a variable reference"
msgstr ""
msgid "Value stream" msgid "Value stream"
msgstr "" msgstr ""
...@@ -36415,6 +36418,9 @@ msgstr "" ...@@ -36415,6 +36418,9 @@ msgstr ""
msgid "Variable" msgid "Variable"
msgstr "" msgstr ""
msgid "Variable references indicated by %{codeStart}$%{codeEnd} may be expanded. If this is not what you want, consider %{docsLinkStart}using a workaround to prevent expansion%{docsLinkEnd}."
msgstr ""
msgid "Variable will be masked in job logs." msgid "Variable will be masked in job logs."
msgstr "" msgstr ""
......
...@@ -123,6 +123,29 @@ describe('Ci variable modal', () => { ...@@ -123,6 +123,29 @@ describe('Ci variable modal', () => {
}); });
}); });
describe.each`
value | secret | rendered
${'value'} | ${'secret_value'} | ${false}
${'dollar$ign'} | ${'dollar$ign'} | ${true}
`('Adding a new variable', ({ value, secret, rendered }) => {
beforeEach(() => {
const [variable] = mockData.mockVariables;
const invalidKeyVariable = {
...variable,
key: 'key',
value,
secret_value: secret,
};
createComponent(mount);
store.state.variable = invalidKeyVariable;
});
it(`${rendered ? 'renders' : 'does not render'} the variable reference warning`, () => {
const warning = wrapper.find(`[data-testid='contains-variable-reference']`);
expect(warning.exists()).toBe(rendered);
});
});
describe('Editing a variable', () => { describe('Editing a variable', () => {
beforeEach(() => { beforeEach(() => {
const [variable] = mockData.mockVariables; const [variable] = mockData.mockVariables;
......
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