2021-03-22 14:51:38 +00:00
|
|
|
import is from '@sindresorhus/is';
|
|
|
|
import {
|
|
|
|
CONFIG_SECRETS_INVALID,
|
|
|
|
CONFIG_VALIDATION,
|
|
|
|
} from '../constants/error-messages';
|
|
|
|
import { logger } from '../logger';
|
|
|
|
import { regEx } from '../util/regex';
|
2022-01-26 09:57:21 +00:00
|
|
|
import { addSecretForSanitizing } from '../util/sanitize';
|
2022-02-07 06:37:17 +00:00
|
|
|
import type { AllConfig, RenovateConfig } from './types';
|
2021-03-22 14:51:38 +00:00
|
|
|
|
|
|
|
const secretNamePattern = '[A-Za-z][A-Za-z0-9_]*';
|
|
|
|
|
|
|
|
const secretNameRegex = regEx(`^${secretNamePattern}$`);
|
|
|
|
const secretTemplateRegex = regEx(`{{ secrets\\.(${secretNamePattern}) }}`);
|
|
|
|
|
|
|
|
function validateSecrets(secrets_: unknown): void {
|
|
|
|
if (!secrets_) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const validationErrors: string[] = [];
|
|
|
|
if (is.plainObject(secrets_)) {
|
|
|
|
for (const [secretName, secretValue] of Object.entries(secrets_)) {
|
|
|
|
if (!secretNameRegex.test(secretName)) {
|
|
|
|
validationErrors.push(`Invalid secret name "${secretName}"`);
|
|
|
|
}
|
|
|
|
if (!is.string(secretValue)) {
|
|
|
|
validationErrors.push(
|
|
|
|
`Secret values must be strings. Found type ${typeof secretValue} for secret ${secretName}`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
validationErrors.push(
|
|
|
|
`Config secrets must be a plain object. Found: ${typeof secrets_}`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (validationErrors.length) {
|
|
|
|
logger.error({ validationErrors }, 'Invalid secrets configured');
|
|
|
|
throw new Error(CONFIG_SECRETS_INVALID);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-02 09:25:10 +00:00
|
|
|
export function validateConfigSecrets(config: AllConfig): void {
|
2021-03-22 14:51:38 +00:00
|
|
|
validateSecrets(config.secrets);
|
|
|
|
if (config.repositories) {
|
|
|
|
for (const repository of config.repositories) {
|
|
|
|
if (is.plainObject(repository)) {
|
|
|
|
validateSecrets(repository.secrets);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function replaceSecretsInString(
|
|
|
|
key: string,
|
|
|
|
value: string,
|
|
|
|
secrets: Record<string, string>
|
|
|
|
): string {
|
|
|
|
// do nothing if no secret template found
|
|
|
|
if (!secretTemplateRegex.test(value)) {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
const disallowedPrefixes = ['branch', 'commit', 'group', 'pr', 'semantic'];
|
|
|
|
if (disallowedPrefixes.some((prefix) => key.startsWith(prefix))) {
|
|
|
|
const error = new Error(CONFIG_VALIDATION);
|
2021-05-17 07:40:54 +00:00
|
|
|
error.validationSource = 'config';
|
2021-03-22 14:51:38 +00:00
|
|
|
error.validationError = 'Disallowed secret substitution';
|
|
|
|
error.validationMessage = `The field ${key} may not use secret substitution`;
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
return value.replace(secretTemplateRegex, (_, secretName) => {
|
2022-01-28 13:49:21 +00:00
|
|
|
if (secrets?.[secretName]) {
|
2021-03-22 14:51:38 +00:00
|
|
|
return secrets[secretName];
|
|
|
|
}
|
|
|
|
const error = new Error(CONFIG_VALIDATION);
|
2021-05-17 07:40:54 +00:00
|
|
|
error.validationSource = 'config';
|
2021-03-22 14:51:38 +00:00
|
|
|
error.validationError = 'Unknown secret name';
|
|
|
|
error.validationMessage = `The following secret name was not found in config: ${String(
|
|
|
|
secretName
|
|
|
|
)}`;
|
|
|
|
throw error;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-02-21 16:11:40 +00:00
|
|
|
function replaceSecretsInObject(
|
2021-03-22 14:51:38 +00:00
|
|
|
config_: RenovateConfig,
|
2022-01-28 13:49:21 +00:00
|
|
|
secrets: Record<string, string>,
|
|
|
|
deleteSecrets: boolean
|
2021-03-22 14:51:38 +00:00
|
|
|
): RenovateConfig {
|
|
|
|
const config = { ...config_ };
|
2022-01-28 13:49:21 +00:00
|
|
|
if (deleteSecrets) {
|
|
|
|
delete config.secrets;
|
|
|
|
}
|
2021-03-22 14:51:38 +00:00
|
|
|
for (const [key, value] of Object.entries(config)) {
|
|
|
|
if (is.plainObject(value)) {
|
2022-02-21 16:11:40 +00:00
|
|
|
config[key] = replaceSecretsInObject(value, secrets, deleteSecrets);
|
2021-03-22 14:51:38 +00:00
|
|
|
}
|
|
|
|
if (is.string(value)) {
|
|
|
|
config[key] = replaceSecretsInString(key, value, secrets);
|
|
|
|
}
|
|
|
|
if (is.array(value)) {
|
|
|
|
for (const [arrayIndex, arrayItem] of value.entries()) {
|
|
|
|
if (is.plainObject(arrayItem)) {
|
2022-02-21 16:11:40 +00:00
|
|
|
value[arrayIndex] = replaceSecretsInObject(
|
2022-01-28 13:49:21 +00:00
|
|
|
arrayItem,
|
|
|
|
secrets,
|
|
|
|
deleteSecrets
|
|
|
|
);
|
2021-03-22 14:51:38 +00:00
|
|
|
} else if (is.string(arrayItem)) {
|
2022-02-11 10:02:30 +00:00
|
|
|
value[arrayIndex] = replaceSecretsInString(key, arrayItem, secrets);
|
2021-03-22 14:51:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return config;
|
|
|
|
}
|
|
|
|
|
2021-06-16 14:02:07 +00:00
|
|
|
export function applySecretsToConfig(
|
|
|
|
config: RenovateConfig,
|
2022-01-28 13:49:21 +00:00
|
|
|
secrets = config.secrets,
|
|
|
|
deleteSecrets = true
|
2021-06-16 14:02:07 +00:00
|
|
|
): RenovateConfig {
|
2021-03-22 14:51:38 +00:00
|
|
|
// Add all secrets to be sanitized
|
2021-06-16 14:02:07 +00:00
|
|
|
if (is.plainObject(secrets)) {
|
|
|
|
for (const secret of Object.values(secrets)) {
|
2023-01-03 12:29:07 +00:00
|
|
|
addSecretForSanitizing(secret);
|
2021-03-22 14:51:38 +00:00
|
|
|
}
|
|
|
|
}
|
2022-02-11 10:02:30 +00:00
|
|
|
// TODO: fix types (#9610)
|
2022-02-21 16:11:40 +00:00
|
|
|
return replaceSecretsInObject(config, secrets as never, deleteSecrets);
|
2021-03-22 14:51:38 +00:00
|
|
|
}
|