2020-08-21 14:45:16 +02:00
|
|
|
import * as aws from './aws';
|
2021-06-22 11:09:26 +02:00
|
|
|
import * as core from '@actions/core';
|
2024-07-22 10:41:37 +02:00
|
|
|
|
|
|
|
import {Docker} from '@docker/actions-toolkit/lib/docker/docker';
|
2020-08-20 16:40:33 +02:00
|
|
|
|
2025-01-23 02:47:49 +01:00
|
|
|
export async function login(registry: string, username: string, password: string, ecr: string, http_errors_to_retry: string[], max_attempts: number, retry_timeout: number): Promise<void> {
|
|
|
|
let succeeded: boolean = false;
|
2025-01-23 02:58:22 +01:00
|
|
|
for (let attempt = 1; attempt <= max_attempts && !succeeded; attempt++) {
|
2025-01-23 02:47:49 +01:00
|
|
|
try {
|
|
|
|
if (/true/i.test(ecr) || (ecr == 'auto' && aws.isECR(registry))) {
|
|
|
|
await loginECR(registry, username, password);
|
|
|
|
} else {
|
|
|
|
await loginStandard(registry, username, password);
|
|
|
|
}
|
2025-01-23 02:58:22 +01:00
|
|
|
succeeded = true;
|
2025-01-23 02:47:49 +01:00
|
|
|
} catch (error) {
|
2025-01-23 09:29:11 +01:00
|
|
|
if (attempt < max_attempts && isRetriableError(error.message, http_errors_to_retry)) {
|
2025-01-23 02:53:23 +01:00
|
|
|
core.info(`Attempt ${attempt} out of ${max_attempts} failed, retrying after ${retry_timeout} seconds`);
|
2025-01-23 02:47:49 +01:00
|
|
|
await new Promise(r => setTimeout(r, retry_timeout * 1000));
|
|
|
|
} else {
|
2025-01-23 09:29:11 +01:00
|
|
|
throw error;
|
2025-01-23 02:47:49 +01:00
|
|
|
}
|
|
|
|
}
|
2020-08-20 16:40:33 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function logout(registry: string): Promise<void> {
|
2024-07-22 10:41:37 +02:00
|
|
|
await Docker.getExecOutput(['logout', registry], {
|
2023-02-21 10:06:17 +01:00
|
|
|
ignoreReturnCode: true
|
|
|
|
}).then(res => {
|
|
|
|
if (res.stderr.length > 0 && res.exitCode != 0) {
|
|
|
|
core.warning(res.stderr.trim());
|
|
|
|
}
|
|
|
|
});
|
2020-08-20 16:40:33 +02:00
|
|
|
}
|
|
|
|
|
2025-01-23 02:58:22 +01:00
|
|
|
function isRetriableError(error_message: string, http_errors_to_retry: string[]): boolean {
|
2025-01-23 09:29:11 +01:00
|
|
|
for (const err_code of http_errors_to_retry) {
|
2025-01-23 02:58:22 +01:00
|
|
|
if (error_message.includes('failed with status: ' + err_code)) {
|
2025-01-23 02:47:49 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-08-20 16:40:33 +02:00
|
|
|
export async function loginStandard(registry: string, username: string, password: string): Promise<void> {
|
2024-04-30 21:52:07 +02:00
|
|
|
if (!username && !password) {
|
2020-10-20 14:41:56 +02:00
|
|
|
throw new Error('Username and password required');
|
2020-08-20 16:40:33 +02:00
|
|
|
}
|
2024-04-30 21:52:07 +02:00
|
|
|
if (!username) {
|
|
|
|
throw new Error('Username required');
|
|
|
|
}
|
|
|
|
if (!password) {
|
|
|
|
throw new Error('Password required');
|
|
|
|
}
|
2020-10-20 14:41:56 +02:00
|
|
|
|
2022-03-21 10:57:36 +01:00
|
|
|
const loginArgs: Array<string> = ['login', '--password-stdin'];
|
2020-10-20 14:41:56 +02:00
|
|
|
loginArgs.push('--username', username);
|
2020-08-20 16:40:33 +02:00
|
|
|
loginArgs.push(registry);
|
|
|
|
|
|
|
|
if (registry) {
|
2021-04-28 00:34:26 +02:00
|
|
|
core.info(`Logging into ${registry}...`);
|
2020-08-20 16:40:33 +02:00
|
|
|
} else {
|
2021-04-28 00:34:26 +02:00
|
|
|
core.info(`Logging into Docker Hub...`);
|
2020-08-20 16:40:33 +02:00
|
|
|
}
|
2024-07-22 10:41:37 +02:00
|
|
|
await Docker.getExecOutput(loginArgs, {
|
2023-02-21 10:06:17 +01:00
|
|
|
ignoreReturnCode: true,
|
|
|
|
silent: true,
|
|
|
|
input: Buffer.from(password)
|
|
|
|
}).then(res => {
|
|
|
|
if (res.stderr.length > 0 && res.exitCode != 0) {
|
|
|
|
throw new Error(res.stderr.trim());
|
|
|
|
}
|
|
|
|
core.info(`Login Succeeded!`);
|
|
|
|
});
|
2020-08-20 16:40:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export async function loginECR(registry: string, username: string, password: string): Promise<void> {
|
2021-12-20 10:43:09 +01:00
|
|
|
core.info(`Retrieving registries data through AWS SDK...`);
|
|
|
|
const regDatas = await aws.getRegistriesData(registry, username, password);
|
|
|
|
for (const regData of regDatas) {
|
|
|
|
core.info(`Logging into ${regData.registry}...`);
|
2024-07-22 10:41:37 +02:00
|
|
|
await Docker.getExecOutput(['login', '--password-stdin', '--username', regData.username, regData.registry], {
|
2023-02-21 10:06:17 +01:00
|
|
|
ignoreReturnCode: true,
|
|
|
|
silent: true,
|
|
|
|
input: Buffer.from(regData.password)
|
|
|
|
}).then(res => {
|
|
|
|
if (res.stderr.length > 0 && res.exitCode != 0) {
|
|
|
|
throw new Error(res.stderr.trim());
|
|
|
|
}
|
|
|
|
core.info('Login Succeeded!');
|
|
|
|
});
|
2021-12-20 10:43:09 +01:00
|
|
|
}
|
2020-08-20 16:40:33 +02:00
|
|
|
}
|