renovate/lib/util/git/auth.ts
Tobias 6c6984b220
feat(git): move git autentification via environment variables to git auth (#22821)
Co-authored-by: Michael Kriese <michael.kriese@visualon.de>
2023-06-20 19:30:39 +00:00

214 lines
6.7 KiB
TypeScript

import type { PlatformId } from '../../constants/platforms';
import { logger } from '../../logger';
import type { HostRule } from '../../types';
import { detectPlatform } from '../common';
import { find, getAll } from '../host-rules';
import { regEx } from '../regex';
import { createURLFromHostOrURL, validateUrl } from '../url';
import type { AuthenticationRule } from './types';
import { parseGitUrl } from './url';
const githubApiUrls = new Set([
'github.com',
'api.github.com',
'https://api.github.com',
'https://api.github.com/',
]);
const standardGitAllowedHostTypes = [
// All known git platforms
'azure',
'bitbucket',
'bitbucket-server',
'gitea',
'github',
'gitlab',
] satisfies PlatformId[];
/**
* Add authorization to a Git Url and returns a new environment variables object
* @returns a new NodeJS.ProcessEnv object without modifying any input parameters
*/
export function getGitAuthenticatedEnvironmentVariables(
originalGitUrl: string,
{ token, hostType, matchHost }: HostRule,
environmentVariables?: NodeJS.ProcessEnv
): NodeJS.ProcessEnv {
if (!token) {
logger.warn(
// TODO: types (#7154)
`Could not create environment variable for ${matchHost!} as token was empty`
);
return { ...environmentVariables };
}
// check if the environmentVariables already contain a GIT_CONFIG_COUNT or if the process has one
const gitConfigCountEnvVariable =
environmentVariables?.GIT_CONFIG_COUNT ?? process.env.GIT_CONFIG_COUNT;
let gitConfigCount = 0;
if (gitConfigCountEnvVariable) {
// passthrough the gitConfigCountEnvVariable environment variable as start value of the index count
gitConfigCount = parseInt(gitConfigCountEnvVariable, 10);
if (Number.isNaN(gitConfigCount)) {
logger.warn(
`Found GIT_CONFIG_COUNT env variable, but couldn't parse the value to an integer: ${String(
process.env.GIT_CONFIG_COUNT
)}. Ignoring it.`
);
gitConfigCount = 0;
}
}
const authenticationRules = getAuthenticationRulesWithToken(
originalGitUrl,
hostType,
token
);
// create a shallow copy of the environmentVariables as base so we don't modify the input parameter object
// add the two new config key and value to the returnEnvironmentVariables object
// increase the CONFIG_COUNT by one for each rule and add it to the object
const newEnvironmentVariables = {
...environmentVariables,
};
for (const rule of authenticationRules) {
newEnvironmentVariables[
`GIT_CONFIG_KEY_${gitConfigCount}`
] = `url.${rule.url}.insteadOf`;
newEnvironmentVariables[`GIT_CONFIG_VALUE_${gitConfigCount}`] =
rule.insteadOf;
gitConfigCount++;
}
newEnvironmentVariables['GIT_CONFIG_COUNT'] = gitConfigCount.toString();
return newEnvironmentVariables;
}
function getAuthenticationRulesWithToken(
url: string,
hostType: string | undefined | null,
authToken: string
): AuthenticationRule[] {
let token = authToken;
let type = hostType;
if (!type) {
type = detectPlatform(url);
}
if (type === 'gitlab') {
token = `gitlab-ci-token:${authToken}`;
}
return getAuthenticationRules(url, token);
}
/**
* Generates the authentication rules for later git usage for the given host
* @link https://coolaj86.com/articles/vanilla-devops-git-credentials-cheatsheet/
*/
export function getAuthenticationRules(
gitUrl: string,
token: string
): AuthenticationRule[] {
const authenticationRules = [];
const hasUser = token.split(':').length > 1;
const insteadUrl = parseGitUrl(gitUrl);
const url = { ...insteadUrl };
const protocol = regEx(/^https?$/).test(url.protocol)
? url.protocol
: 'https';
// ssh protocol with user if empty
url.token = hasUser ? token : `ssh:${token}`;
authenticationRules.push({
url: url.toString(protocol),
// only edge case, need to stringify ourself because the exact syntax is not supported by the library
// https://github.com/IonicaBizau/git-url-parse/blob/246c9119fb42c2ea1c280028fe77c53eb34c190c/lib/index.js#L246
insteadOf: `ssh://git@${insteadUrl.resource}${
insteadUrl.port ? `:${insteadUrl.port}` : ''
}/${insteadUrl.full_name}${insteadUrl.git_suffix ? '.git' : ''}`,
});
// alternative ssh protocol with user if empty
url.token = hasUser ? token : `git:${token}`;
authenticationRules.push({
url: url.toString(protocol),
insteadOf: insteadUrl.toString('ssh'),
});
// https protocol with no user as default fallback
url.token = token;
authenticationRules.push({
url: url.toString(protocol),
insteadOf: insteadUrl.toString(protocol),
});
return authenticationRules;
}
export function getGitEnvironmentVariables(
additionalHostTypes: string[] = []
): NodeJS.ProcessEnv {
let environmentVariables: NodeJS.ProcessEnv = {};
// hard-coded logic to use authentication for github.com based on the githubToken for api.github.com
const githubToken = find({
hostType: 'github',
url: 'https://api.github.com/',
});
if (githubToken?.token) {
environmentVariables = getGitAuthenticatedEnvironmentVariables(
'https://github.com/',
githubToken
);
}
// construct the Set of allowed hostTypes consisting of the standard Git provides
// plus additionalHostTypes, which are provided as parameter
const gitAllowedHostTypes = new Set<string>([
...standardGitAllowedHostTypes,
...additionalHostTypes,
]);
// filter rules without `matchHost` and `token` and github api github rules
const hostRules = getAll()
.filter((r) => r.matchHost && r.token)
.filter((r) => !githubToken || !githubApiUrls.has(r.matchHost!));
// for each hostRule without hostType we add additional authentication variables to the environmentVariables
// for each hostRule with hostType we add additional authentication variables to the environmentVariables
for (const hostRule of hostRules) {
if (!hostRule.hostType || gitAllowedHostTypes.has(hostRule.hostType)) {
environmentVariables = addAuthFromHostRule(
hostRule,
environmentVariables
);
}
}
return environmentVariables;
}
function addAuthFromHostRule(
hostRule: HostRule,
env: NodeJS.ProcessEnv
): NodeJS.ProcessEnv {
let environmentVariables = env;
const httpUrl = createURLFromHostOrURL(hostRule.matchHost!)?.toString();
if (validateUrl(httpUrl)) {
logger.trace(
// TODO: types (#7154)
// eslint-disable-next-line @typescript-eslint/restrict-template-expressions
`Adding Git authentication for ${httpUrl} using token auth.`
);
environmentVariables = getGitAuthenticatedEnvironmentVariables(
httpUrl!,
hostRule,
environmentVariables
);
} else {
logger.debug(
`Could not parse registryUrl ${hostRule.matchHost!} or not using http(s). Ignoring`
);
}
return environmentVariables;
}