mirror of https://github.com/renovatebot/renovate
256 lines
7.2 KiB
TypeScript
256 lines
7.2 KiB
TypeScript
import is from '@sindresorhus/is';
|
|
import { findLocalSiblingOrParent, readLocalFile } from '../../../../util/fs';
|
|
import { newlineRegex, regEx } from '../../../../util/regex';
|
|
import { get as getVersioning } from '../../../versioning';
|
|
import type { UpdateArtifactsResult } from '../../types';
|
|
import type {
|
|
LineNumbers,
|
|
ProviderLock,
|
|
ProviderLockUpdate,
|
|
ProviderSlice,
|
|
} from './types';
|
|
|
|
const providerStartLineRegex = regEx(
|
|
`^provider "(?<registryUrl>[^/]*)/(?<namespace>[^/]*)/(?<depName>[^/]*)"`,
|
|
);
|
|
const versionLineRegex = regEx(
|
|
`^(?<prefix>[\\s]*version[\\s]*=[\\s]*")(?<version>[^"']+)(?<suffix>".*)$`,
|
|
);
|
|
const constraintLineRegex = regEx(
|
|
`^(?<prefix>[\\s]*constraints[\\s]*=[\\s]*")(?<constraint>[^"']+)(?<suffix>".*)$`,
|
|
);
|
|
const hashLineRegex = regEx(`^(?<prefix>\\s*")(?<hash>[^"]+)(?<suffix>",.*)$`);
|
|
|
|
const lockFile = '.terraform.lock.hcl';
|
|
|
|
export function findLockFile(packageFilePath: string): Promise<string | null> {
|
|
return findLocalSiblingOrParent(packageFilePath, lockFile);
|
|
}
|
|
|
|
export function readLockFile(lockFilePath: string): Promise<string | null> {
|
|
return readLocalFile(lockFilePath, 'utf8');
|
|
}
|
|
|
|
export function extractLocks(lockFileContent: string): ProviderLock[] | null {
|
|
const lines = lockFileContent.split(newlineRegex);
|
|
const blockStarts: number[] = [];
|
|
// get first lines of blocks
|
|
lines.forEach((line, index) => {
|
|
if (line.startsWith('provider "')) {
|
|
blockStarts.push(index);
|
|
}
|
|
});
|
|
|
|
// sort ascending
|
|
const sortedStarts = blockStarts.sort((a, b) => a - b);
|
|
const contentSlices = sortedStarts.map((start, index, array) => {
|
|
let end: number;
|
|
if (index < array.length - 1) {
|
|
end = array[index + 1];
|
|
} else {
|
|
end = lines.length - 1;
|
|
}
|
|
const slice: ProviderSlice = {
|
|
lines: lines.slice(start, end),
|
|
block: {
|
|
start,
|
|
end,
|
|
},
|
|
};
|
|
return slice;
|
|
});
|
|
|
|
// generate Lock objects from slices
|
|
const locks = contentSlices.map((slice) => {
|
|
let packageName = '';
|
|
let registryUrl = '';
|
|
let version = '';
|
|
let constraints = '';
|
|
const relativeLineNumbers: LineNumbers = {
|
|
block: slice.block,
|
|
hashes: {
|
|
start: -1,
|
|
end: -1,
|
|
},
|
|
};
|
|
const hashes: string[] = [];
|
|
|
|
slice.lines.forEach((line, index) => {
|
|
const hashLineResult = hashLineRegex.exec(line);
|
|
if (hashLineResult?.groups) {
|
|
hashes.push(hashLineResult.groups.hash);
|
|
relativeLineNumbers.hashes.start =
|
|
relativeLineNumbers.hashes.start === -1
|
|
? index
|
|
: relativeLineNumbers.hashes.start;
|
|
relativeLineNumbers.hashes.end = index;
|
|
return;
|
|
}
|
|
|
|
const providerStartLineResult = providerStartLineRegex.exec(line);
|
|
if (providerStartLineResult?.groups) {
|
|
packageName = `${providerStartLineResult.groups.namespace}/${providerStartLineResult.groups.depName}`;
|
|
registryUrl = providerStartLineResult.groups.registryUrl;
|
|
return;
|
|
}
|
|
|
|
const versionLineResult = versionLineRegex.exec(line);
|
|
if (versionLineResult?.groups) {
|
|
version = versionLineResult.groups.version;
|
|
relativeLineNumbers.version = index;
|
|
return;
|
|
}
|
|
|
|
const constraintLineResult = constraintLineRegex.exec(line);
|
|
if (constraintLineResult?.groups) {
|
|
constraints = constraintLineResult.groups.constraint;
|
|
relativeLineNumbers.constraint = index;
|
|
}
|
|
});
|
|
|
|
const lock: ProviderLock = {
|
|
packageName,
|
|
registryUrl: `https://${registryUrl}`,
|
|
version,
|
|
constraints,
|
|
hashes,
|
|
lineNumbers: relativeLineNumbers,
|
|
};
|
|
return lock;
|
|
});
|
|
|
|
if (locks.length === 0) {
|
|
return null;
|
|
}
|
|
return locks;
|
|
}
|
|
|
|
export function isPinnedVersion(value: string | undefined): boolean {
|
|
const versioning = getVersioning('hashicorp');
|
|
return !!value && !!versioning.isSingleVersion(value);
|
|
}
|
|
|
|
export function writeLockUpdates(
|
|
updates: ProviderLockUpdate[],
|
|
lockFilePath: string,
|
|
oldLockFileContent: string,
|
|
): UpdateArtifactsResult {
|
|
const lines = oldLockFileContent.split(newlineRegex);
|
|
|
|
const sections: string[][] = [];
|
|
|
|
// sort updates in order of appearance in the lockfile
|
|
// TODO #22198
|
|
updates.sort(
|
|
(a, b) => a.lineNumbers.block!.start - b.lineNumbers.block!.start,
|
|
);
|
|
updates.forEach((update, index, array) => {
|
|
// re add leading whitespace
|
|
let startWhitespace: number | undefined;
|
|
if (index > 0) {
|
|
// get end of the
|
|
// TODO #22198
|
|
startWhitespace = array[index - 1].lineNumbers.block!.end;
|
|
}
|
|
const leadingNonRelevantLines = lines.slice(
|
|
startWhitespace,
|
|
// TODO #22198
|
|
update.lineNumbers.block!.start,
|
|
);
|
|
sections.push(leadingNonRelevantLines);
|
|
|
|
const providerBlockLines = lines.slice(
|
|
// TODO #22198
|
|
update.lineNumbers.block!.start,
|
|
update.lineNumbers.block!.end,
|
|
);
|
|
const newProviderBlockLines: string[] = [];
|
|
let hashLinePrefix = '';
|
|
let hashLineSuffix = '';
|
|
providerBlockLines.forEach((providerBlockLine, providerBlockIndex) => {
|
|
const versionLine = providerBlockLine.replace(
|
|
versionLineRegex,
|
|
`$<prefix>${update.newVersion}$<suffix>`,
|
|
);
|
|
if (versionLine !== providerBlockLine) {
|
|
newProviderBlockLines.push(versionLine);
|
|
return;
|
|
}
|
|
|
|
const constraintLine = providerBlockLine.replace(
|
|
constraintLineRegex,
|
|
`$<prefix>${update.newConstraint}$<suffix>`,
|
|
);
|
|
if (constraintLine !== providerBlockLine) {
|
|
newProviderBlockLines.push(constraintLine);
|
|
return;
|
|
}
|
|
|
|
const hashLineRegexResult = hashLineRegex.exec(providerBlockLine);
|
|
if (hashLineRegexResult?.groups) {
|
|
// skip hash line but safe the whitespace
|
|
hashLinePrefix = hashLineRegexResult.groups.prefix;
|
|
hashLineSuffix = hashLineRegexResult.groups.suffix;
|
|
return;
|
|
}
|
|
newProviderBlockLines.push(providerBlockLine);
|
|
});
|
|
|
|
const hashesWithWhitespace = update.newHashes.map(
|
|
(value) => `${hashLinePrefix}${value}${hashLineSuffix}`,
|
|
);
|
|
newProviderBlockLines.splice(
|
|
// TODO #22198
|
|
update.lineNumbers.hashes.start!,
|
|
0,
|
|
...hashesWithWhitespace,
|
|
);
|
|
sections.push(newProviderBlockLines);
|
|
});
|
|
|
|
const trailingNotUpdatedLines = lines.slice(
|
|
updates[updates.length - 1].lineNumbers.block?.end,
|
|
);
|
|
sections.push(trailingNotUpdatedLines);
|
|
|
|
const newLines = sections.reduce((previousValue, currentValue) =>
|
|
previousValue.concat(currentValue),
|
|
);
|
|
const newContent = newLines.join('\n');
|
|
|
|
return {
|
|
file: {
|
|
type: 'addition',
|
|
path: lockFilePath,
|
|
contents: newContent,
|
|
},
|
|
};
|
|
}
|
|
|
|
export function massageNewValue(value: string | undefined): string | undefined {
|
|
if (is.nullOrUndefined(value)) {
|
|
return value;
|
|
}
|
|
|
|
const elements = value.split(',');
|
|
const massagedElements: string[] = [];
|
|
for (const element of elements) {
|
|
// these constraints are allowed to miss precision
|
|
if (element.includes('~>')) {
|
|
massagedElements.push(element);
|
|
continue;
|
|
}
|
|
|
|
const missing0s = 3 - element.split('.').length;
|
|
|
|
let massagedElement = element;
|
|
|
|
for (let i = 0; i < missing0s; i++) {
|
|
massagedElement = `${massagedElement}.0`;
|
|
}
|
|
massagedElements.push(massagedElement);
|
|
}
|
|
|
|
return massagedElements.join(',');
|
|
}
|