Files
vue-ts-example/vite.config.plugin.archiver.ts
mini2024 051244dde6
Some checks failed
/ surge (push) Successful in 2m32s
/ depcheck (push) Successful in 2m43s
/ build-and-deploy-to-vercel (push) Successful in 2m51s
/ lint-build-and-check (push) Successful in 3m5s
/ playwright (push) Has been cancelled
refactor(archiver): 优化打包插件逻辑并支持更多压缩格式
重构打包插件以支持多种压缩格式(zip, tar, tgz),并增加时间戳选项。同时,将插件调用移至VS Code终端检测逻辑中,确保仅在VS Code环境中执行
2025-03-23 00:46:56 +08:00

120 lines
3.7 KiB
TypeScript

import type { Plugin } from 'vite';
import archiver from 'archiver';
import fs from 'node:fs';
import path from 'node:path';
import { createLogger } from 'vite';
interface ArchiverOptions {
/**
* 是否添加时间戳到输出文件名
* @default false
*/
addTimestamp?: boolean;
/**
* 压缩格式,支持 'zip', 'tar', 'tgz'
* @default 'zip'
*/
format?: 'tar' | 'tgz' | 'zip';
/**
* 输出目录
* @default ''
*/
outputDir?: string;
/**
* 输出的文件名(不含扩展名)
* @default 'dist'
*/
outputFileName?: string;
/**
* 要打包的源目录,可以是单个目录或目录数组
* @default 'dist'
*/
sourceDir?: string | string[];
}
/**
* 用于将构建输出打包成压缩文件的Vite插件
*/
export function viteArchiverPlugin(options: ArchiverOptions = {}): Plugin {
const { addTimestamp = false, format = 'zip', outputDir = '', outputFileName = 'dist', sourceDir = 'dist' } = options;
// 创建Vite日志记录器
const logger = createLogger();
return {
apply: 'build',
closeBundle: async () => {
// 处理单个目录或目录数组
const sourceDirs = Array.isArray(sourceDir) ? sourceDir : [sourceDir];
const sourcePaths = sourceDirs.map((dir) => path.resolve(process.cwd(), dir));
// 检查所有源目录是否存在
const nonExistentDirs = sourcePaths.filter((dir) => !fs.existsSync(dir));
if (nonExistentDirs.length > 0) {
logger.error(`Source directories do not exist: ${nonExistentDirs.join(', ')}`);
return;
}
logger.info(`Creating archive from: ${sourceDirs.join(', ')}`);
// 如果输出目录不存在,则创建它
const outputPath = path.resolve(process.cwd(), outputDir);
if (outputDir && !fs.existsSync(outputPath)) {
fs.mkdirSync(outputPath, { recursive: true });
}
// 生成文件名,可选添加时间戳
let finalFileName = outputFileName;
if (addTimestamp) {
const now = new Date();
const timestamp = `${now.getFullYear()}${String(now.getMonth() + 1).padStart(2, '0')}${String(now.getDate()).padStart(2, '0')}-${String(now.getHours()).padStart(2, '0')}${String(now.getMinutes()).padStart(2, '0')}${String(now.getSeconds()).padStart(2, '0')}`;
finalFileName = `${outputFileName}-${timestamp}`;
}
// 根据格式确定文件扩展名
const fileExtension = format === 'zip' ? 'zip' : format === 'tar' ? 'tar' : 'tar.gz';
const outputFilePath = path.join(outputPath, `${finalFileName}.${fileExtension}`);
const output = fs.createWriteStream(outputFilePath);
// 创建适当格式的归档器
const archive = archiver(format === 'tgz' ? 'tar' : format, {
gzip: format === 'tgz',
...(format === 'zip' ? { zlib: { level: 9 } } : {}),
});
// 监听所有归档数据写入完成
output.on('close', () => {
logger.info(`Archive created: ${outputFilePath}`);
});
// 错误处理
archive.on('warning', (err) => {
if (err.code === 'ENOENT') {
logger.warn(`Warning during archiving: ${err.message}`);
} else {
logger.error(`Error during archiving: ${err.message}`);
throw err;
}
});
archive.on('error', (err) => {
logger.error(`Fatal error during archiving: ${err.message}`);
throw err;
});
// 将归档数据通过管道传输到文件
archive.pipe(output);
// 添加文件到归档
for (const sourcePath of sourcePaths) {
archive.directory(sourcePath, false);
}
// 完成归档
await archive.finalize();
},
name: 'vite-plugin-archiver',
};
}