The author has been working on a micro-frontend project for more than a year. A team manages ten micro-applications. After switching to docker image deployment, the release operation has changed from 1 minute when the original script directly connects to the server to dozens of minutes, especially when uploading each application to its respective Alibaba Cloud warehouse. Here we will write another script to package the Docker image with one click and upload it to Alibaba Cloud. This article only talks about how to make a script to help reduce the burden on developers. For the configuration of docker-compose, see: Deploy micro-frontend projects comfortably using various postures (Part 1: Packaging and uploading) RenderingDirectly on the code/** * @name docker image packaging and upload script * @author weilan * @time 2021.02.22 */ const fs = require('fs'); const path = require('path'); const util = require('util'); const { log } = require('../utils/log'); const exec = util.promisify(require('child_process').exec); const sub_app_ath = path.resolve(); let sub_apps = fs.readdirSync(sub_app_ath).filter(i => /^subapp|master/.test(i)); const inquirer = require('inquirer'); // For command line interaction/** * @name command line interaction configuration item*/ const question = [ { type: 'confirm', name: 'dist', message: 'Do you need to package the front-end static resources? ', }, { type: 'confirm', name: 'env', message: 'Please select whether you need to package it for offline intranet deployment', when: function (answers) { // The current question will be asked only when answer is true return answers.dist } }, { type: 'checkbox', name: 'apps', message: 'Please select the module to be published', choices: sub_apps, validate: function (val) { if (val.length) { // check return true; } return "The selection cannot be empty"; } }, ] /** * @name Perform logical processing based on command interaction configuration results*/ inquirer.prompt(question).then(async (answer) => { let subApps = answer.apps; let buildScript = answer.env ? 'yarn build --Intranet' : 'yarn build'; let needDist = answer.dist; let now = +new Date(); // Log in to Alibaba Cloud const { error: loginError } = await exec('docker login --username=哈哈哈--password=嘿嘿registry.cn-zhangjiakou.aliyuncs.com'); if (loginError) { log.red(loginError, 'Login to mirror center failed') return; } console.log(`Start processing ${JSON.stringify(subApps)} in sequence ......`); subApps.reduce((chain, item) => { return chain.then(() => publishIamge(item, now, needDist, buildScript)) }, Promise.resolve()) }); /** * @name Package the image and push it to Alibaba Cloud* @param {String} moduleName Module name* @param {String} now Current version timestamp* @param {Boolean} needDist Whether to package the front-end static resources* @param {String} buildScript Front-end static resource packaging command*/ async function publishIamge(moduleName, now, needDist, buildScript) { //Package front-end static resources if (needDist) { console.log('Start packaging front-end static resources' + moduleName); const { error } = await exec(buildScript, { cwd: path.resolve(moduleName) }); if (error) { log.red(moduleName, 'Front-end code packaging error:', error) return; } log.green(moduleName + 'Front-end code packaged successfully') } // Packaging image console.log(`Start packaging image ${moduleName} ......`); const { stdout: buildStdout, error: buildError } = await exec('docker-compose build ' + moduleName); if (buildError) { log.red(buildError, 'Mirror packaging error') return; } log.cyan(buildStdout) log.green('Image packaging completed, start making image tags') // Update image tag const imageName = 'ibp2fe_' + moduleName; const { error: tagError } = await exec(`docker tag ${imageName} registry.cn-zhangjiakou.aliyuncs.com/futureweb/${imageName}:${now}`); if (tagError) { log.red(tagError, 'Mirror tag exception') return; } log.green('Mirror version tag updated, start updating last tag') // Update the last version of the image tag const { error: tagLastError } = await exec(`docker tag ${imageName} registry.cn-zhangjiakou.aliyuncs.com/futureweb/${imageName}`); if (tagLastError) { log.red(tagError, 'Mirror last tag exception') return; } log.green('Mirror last tag updated, start uploading') const { stdout: pushStdout, error: pushError } = await exec('docker push registry.cn-zhangjiakou.aliyuncs.com/futureweb/' + imageName); if (pushError) { log.red(pushError, 'Image upload failed') return; } log.cyan(pushStdout) log.green('Image upload successful') } process.on('unhandledRejection', (reason, p) => { console.log('Unhandled Rejection at: Promise', p, 'reason:', reason); // application specific logging, throwing an error, or other logic here }); Implementation ideas and precautions
There is one point that needs attention here. Because it is handled by tool functions, the path of each module is read by node, so your module name should be consistent with the service name and container name in docker-compose.yml. Also, please note that after your image is packaged, the underscore of the root directory outside your docker-compose.yml is connected to your docker-compose service name, so your Alibaba Cloud image repository should be named consistent with the combined image name. The above is to facilitate the tool function to universally handle the images of various modules.
You may also be interested in:
|
<<: HTML code example: detailed explanation of hyperlinks
>>: Detailed explanation of semiotics in Html/CSS
Diversifying website layouts is our front-end spe...
After MySQL is installed, you can verify whether ...
Judgment symbols are often used in MySQL, and not...
This article uses examples to describe common bas...
Table of contents 1. What is recursion? 2. Solve ...
There are two ways to export csv in win10. The fi...
Preface Workbench is installed on one computer, a...
In MySQL, database garbled characters can general...
There are three main ways to use CSS in a page: ad...
Every time after installing the system, I have to...
What is an inode? To understand inode, we must st...
React is different from Vue. It implements route ...
How can you find the location of the configuratio...
Table of contents Preface Generate SVG Introducti...
Preface The server used by the blogger was purcha...