nodejs tinypng 压缩

/** 
 * 帮助文档
 * -------
 * 
 * 获取帮助
 * 指令 -h
 * 
 * 获取命令执行文件夹 
 * 指令 -f 
 * 参数 ./
 * 必填,待处理的图片文件夹
 * 
 * 获取是否深度递归处理图片文件夹
 * 指令 -deep
 * 可选,默认不深度递归
 * 
 * 命令行脚本参考示例
 * > node ./tinypng.js -f ./test -deep
 *  */

const fs = require('fs');
const path = require('path');
const https = require('https');
const URL = require('url').URL;
const EventEmitter = require('events');
const err = msg => new EventEmitter().emit('error', msg);

if (getHelp()) return false;

const conf = {
    files: [],
    EntryFolder: getEntryFolder(),
    DeepLoop: getDeepLoop(),
    Exts: ['.jpg', '.png'],
    Max: 5200000, // 5MB == 5242848.754299136
}

fileFilter(conf.EntryFolder)

console.log("本次执行脚本的配置:", conf);
console.log("等待处理文件的数量:", conf.files.length)

conf.files.forEach(img => fileUpload(img));

 工具函数

/**
 * 获取帮助命令
 * 指令 -h
 */
function getHelp() {
    let i = process.argv.findIndex(i => i === "-h");
    if (i !== -1) {
        // console.log(
        // * 帮助文档
        // * -------
        // * 
        // * 获取帮助
        // * 指令 -h
        // * 
        // * 获取命令执行文件夹 
        // * 指令 -f 
        // * 参数 ./
        // * 必填,待处理的图片文件夹
        // * 
        // * 获取是否深度递归处理图片文件夹
        // * 指令 -deep
        // * 可选,默认不深度递归
        // * 
        // * > node ./tinypng.js -f ./test -deep
        // )
        return true;
    }
}

/**
 * 获取命令执行文件夹 
 * 指令 -f 
 * 参数 ./
 * 必填,待处理的图片文件夹
 */
function getEntryFolder() {
    let i = process.argv.findIndex(i => i === "-f");
    if (i === -1 || !process.argv[i + 1]) return err('获取命令执行文件夹:失败');
    return process.argv[i + 1];
}

/**
 * 获取是否深度递归处理图片文件夹
 * 指令 -deep
 * 可选,默认不深度递归
 */
function getDeepLoop() {
    return process.argv.findIndex(i => i === "-deep") !== -1;
}

/**
 * 过滤待处理文件夹,得到待处理文件列表
 * @param {*} folder 待处理文件夹
 * @param {*} files 待处理文件列表
 */
function fileFilter(folder) {
    // 读取文件夹
    fs.readdirSync(folder).forEach(file => {
        let fullFilePath = path.join(folder, file)
        // 读取文件信息
        let fileStat = fs.statSync(fullFilePath);
        // 过滤文件安全性/大小限制/后缀名
        if (fileStat.size <= conf.Max && fileStat.isFile() && conf.Exts.includes(path.extname(file))) conf.files.push(fullFilePath);
        // 是都要深度递归处理文件夹
        else if (conf.DeepLoop && fileStat.isDirectory()) fileFilter(fullFilePath);
    });
}

/**
 * TinyPng 远程压缩 HTTPS 请求的配置生成方法
 */

function getAjaxOptions() {
    return {
        method: 'POST',
        hostname: 'tinypng.com',
        path: '/web/shrink',
        headers: {
            rejectUnauthorized: false,
            "X-Forwarded-For": Array(4).fill(1).map(() => parseInt(Math.random() * 254 + 1)).join('.'),
            'Postman-Token': Date.now(),
            'Cache-Control': 'no-cache',
            'Content-Type': 'application/x-www-form-urlencoded',
            'User-Agent': 'Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/56.0.2924.87 Safari/537.36'
        }
    }
}

/**
 * TinyPng 远程压缩 HTTPS 请求
 * @param {string} img 待处理的文件
 * @success {
 *              "input": { "size": 887, "type": "image/png" },
 *              "output": { "size": 785, "type": "image/png", "width": 81, "height": 81, "ratio": 0.885, "url": "https://tinypng.com/web/output/7aztz90nq5p9545zch8gjzqg5ubdatd6" }
 *           }
 * @error  {"error": "Bad request", "message" : "Request is invalid"}
 */
function fileUpload(imgPath) {
    let req = https.request(getAjaxOptions(), (res) => {
        res.on('data', buf => {
            let obj = JSON.parse(buf.toString());
            if (obj.error) console.log(`压缩失败!\n 当前文件:${imgPath} \n ${obj.message}`);
            else fileUpdate(imgPath, obj);
        });
    });

    req.write(fs.readFileSync(imgPath), 'binary');
    req.on('error', e => console.error(`请求错误! \n 当前文件:${imgPath} \n`, e));
    req.end();
}

// 该方法被循环调用,请求图片数据
function fileUpdate(entryImgPath, obj) {
    let options = new URL(obj.output.url);
    let req = https.request(options, res => {
        let body = '';
        res.setEncoding('binary');
        res.on('data', (data) => body += data);
        res.on('end', () => {
            fs.writeFile(entryImgPath, body, 'binary', err => {
                if (err) return console.error(err);
                let log = `压缩成功 ,`;
                log += `优化比例: ${ (( 1 - obj.output.ratio) * 100).toFixed(2) }%` ,
                log += `原始大小: ${ (obj.input.size / 1024).toFixed(2) }KB` ,
                log += `压缩大小: ${ (obj.output.size / 1024).toFixed(2) }KB` ,
                log += `文件:${entryImgPath}`
                console.log(log);
            });
        });
    });
    req.on('error', e => console.error(e));
    req.end();
}

// node ./tinypng.js -f ./static/smp/m/course_task

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要在Node.js中进行文件压缩和解压缩操作,你可以使用一些第三方库来实现。首先,你需要安装这些库。你可以使用以下命令来安装uglify-js和compressing库: ``` npm install -g uglify-js npm install compressing ``` 接下来,你可以使用uglify-js库来压缩JavaScript文件。你可以使用以下命令来压缩一个JavaScript文件: ``` uglifyjs file.js -c -m -o file-min.js ``` 这将压缩file.js文件并将结果保存为file-min.js文件。 对于文件的解压缩,你可以使用compressing库。以下是一个解压无密码文件的示例代码: ```javascript var compressing = require("compressing"); compressing.zip.uncompress('a2.zip', '') .then(() => console.log('success')) .catch(err => console.error(err)); ``` 这将解压名为a2.zip的文件。 如果你想进行无密码压缩,你可以使用archiver库。以下是一个示例代码,展示了如何使用archiver库进行无密码压缩: ```javascript var fs = require('fs'); var path = require('path'); var archiver = require('archiver'); zip("zipout", "test.zip", (s) => console.log(s)); function zip(url, name, cb) { // init var output = fs.createWriteStream(name); output.on('close', () => cb('finish')); // zip var archive = archiver('zip', { zlib: { level: 9 } }); archive.on('error', err => cb(err)); archive.pipe(output); if (fs.statSync(url).isFile()) archive.file(url, { name: path.basename(url) }); // 文件 else archive.directory(url, url); // 文件夹 archive.finalize(); } ``` 这将压缩名为zipout的文件夹,并将结果保存为test.zip文件。 希望这些信息能够帮助到你。如果你还有其他问题,请随时提问。 #### 引用[.reference_title] - *1* [NodeJs使用UglifyJs压缩js/css文件](https://blog.csdn.net/SE_JW/article/details/125447915)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [【nodeJS】文件压缩](https://blog.csdn.net/liuzhicheng1845/article/details/118471037)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值