#!/usr/bin/env node /*! * @license * Copyright © 2005-2025 Hyland Software, Inc. and its affiliates. All rights reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import { argv, exit } from 'node:process'; import { exec } from './exec'; import { Command } from 'commander'; import { logger } from './logger'; import { resolve } from 'path'; const program = new Command(); // eslint-disable-next-line no-shadow enum TARGETS { publish = 'publish', link = 'link' } const DOCKER_FILENAME = 'Dockerfile'; export interface PublishArgs { tag?: string; dryrun?: boolean; verbose?: boolean; loginCheck?: boolean; loginUsername?: string; loginPassword?: string; loginRepo?: string; dockerRepo?: string; buildArgs?: string[]; dockerTags?: string; pathProject: string; fileName: string; sourceTag?: string; } /** * Perform a login * * @param args arguments */ function login(args: PublishArgs) { logger.info(`Perform docker login...${args.loginRepo}`); const loginDockerRes = exec('docker', ['login', `-u=${args.loginUsername}`, `-p=${args.loginPassword}`, `${args.loginRepo}`]); logger.info(loginDockerRes); } /** * Build Docker image * * @param args command arguments * @param tag tag */ function buildImage(args: PublishArgs, tag: string) { logger.info(`Perform docker build...${args.dockerRepo}:${tag}`); const buildArgs = []; if (typeof args.buildArgs === 'string') { buildArgs.push(`--build-arg=${args.buildArgs}`); } else { args.buildArgs.forEach((envVar) => { buildArgs.push(`--build-arg=${envVar}`); }); } if (args.verbose) { logger.info(`Dry-run Perform docker build -t=${args.dockerRepo}:${tag} ${buildArgs} -f=${args.fileName} ${args.pathProject}`); } const response = exec('docker', ['build', `-t=${args.dockerRepo}:${tag}`, ...buildArgs, `-f=${args.fileName}`, args.pathProject], {}); logger.info(response); } /** * Tag Docker image * * @param args command arguments * @param imageTag image tag * @param newTag new image tag */ function tagImage(args: PublishArgs, imageTag: string, newTag: string) { logger.info(`Perform docker tag... ${args.dockerRepo}:${imageTag} on ${args.dockerRepo}:${newTag}`); const response = exec('docker', ['tag', `${args.dockerRepo}:${imageTag}`, `${args.dockerRepo}:${newTag}`], {}); logger.info(response); } /** * Pull Docker image * * @param dockerRepo repository * @param sourceTag tag */ function pullImage(dockerRepo: string, sourceTag: string) { logger.info(`Perform docker pull... ${dockerRepo}:${sourceTag}`); const response = exec('docker', ['pull', `${dockerRepo}:${sourceTag}`], {}); logger.info(response); } /** * Push Docker image * * @param args command arguments * @param tag tag */ function pushImage(args: PublishArgs, tag: string) { if (args.dryrun) { logger.info(`Dry-run Perform docker push... ${args.dockerRepo}:${tag}`); } else { logger.info(`Perform docker push... ${args.dockerRepo}:${tag}`); const response = exec('docker', ['push', `${args.dockerRepo}:${tag}`], {}); logger.info(response); } } /** * Clean Docker image * * @param args command arguments * @param tag tag */ function cleanImage(args: PublishArgs, tag: string) { logger.info(`Perform docker clean on tag:${tag}...`); const response = exec('docker', ['rmi', `-f`, `${args.dockerRepo}:${tag}`], {}); logger.info(response); } /** * Publish to Docker command * * @param args command arguments */ export default function main(args: PublishArgs) { program .version('0.1.0') .description( 'Move in the folder where you have your Dockerfile and run the command:\n\n' + 'adf-cli docker-publish --dockerRepo "${docker_repository}" --dockerTags "${TAGS}"' ) .option('--loginRepo [type]', 'URL registry') .option('--loginPassword [type]', ' password') .option('--loginUsername [type]', ' username') .option('--dryrun [type]', 'dryrun') .option('--verbose [type]', 'verbose') .option('--loginCheck [type]', 'perform login') .option('--pathProject [type]', 'the path build context') .option('--sourceTag [type]', 'sourceTag') .option('--buildArgs [type...]', 'buildArgs') .option('--fileName [type...]', 'Docker file name', DOCKER_FILENAME) .option('--target [type]', 'target: publish or link', TARGETS.publish) .requiredOption('--dockerRepo [type]', 'docker repo') .requiredOption('--dockerTags [type]', ' tags') .parse(argv); if (argv.includes('-h') || argv.includes('--help')) { program.outputHelp(); return; } if (!Object.values(TARGETS).includes(program.opts().target)) { logger.error(`error: invalid --target value. It can be ${Object.values(TARGETS)}`); exit(1); } if (program.opts().target === TARGETS.publish && args.buildArgs === undefined) { logger.error(`error: required option --buildArgs [type] in case the target is ${TARGETS.publish}`); exit(1); } if (program.opts().target === TARGETS.link && args.sourceTag === undefined) { logger.error(`error: required option --sourceTag [type] in case the target is ${TARGETS.link}`); exit(1); } if (args.pathProject === undefined) { args.pathProject = resolve('./'); } if (args.fileName === undefined) { args.fileName = DOCKER_FILENAME; } if (args.loginCheck === true) { login(args); } let mainTag: string; if (args.dockerTags !== '') { args.dockerTags.split(',').forEach((tag, index) => { if (tag) { logger.info(`Analyzing tag:${tag} ... for target ${program.opts().target}`); if (program.opts().target === TARGETS.publish) { if (index === 0) { logger.info(`Build only once`); mainTag = tag; buildImage(args, mainTag); } } else { mainTag = args.sourceTag; pullImage(args.dockerRepo, mainTag); } tagImage(args, mainTag, tag); pushImage(args, tag); } }); cleanImage(args, mainTag); } else { logger.error(`dockerTags cannot be empty ...`); } }