nodejs 全自動使用 Tinypng (免費版,無需任何配置)壓縮圖片

前端一直比較喜歡tinypng壓縮圖片,可是每次都要用手拖來拖去,很累.
最近不忙了,node擼了一段100行的代碼,一個命令全自動壓縮文件夾(包含子文件夾)裏全部大小小於5MBjpgpng圖片,親測成功,再次感謝tinypng前端

大致思路:node

  • 遞歸獲取本地文件夾裏的文件
  • 過濾文件,格式必須是.jpg .png,大小小於5MB.(文件夾遞歸)
  • 每次只處理一個文件(能夠繞過20個的數量限制)
  • 處理返回數據拿到遠程優化圖片地址
  • 取回圖片更新本地圖片
  • 純node實現不依賴任何其餘代碼片斷

上代碼:web

const fs = require('fs');
const path = require('path');
const https = require('https');
const crypto = require('crypto');
const { URL } = require('url');

const root = './',
  exts = ['.jpg', '.png'],
  max = 5200000; // 5MB == 5242848.754299136

const options = {
  method: 'POST',
  hostname: 'tinypng.com',
  path: '/web/shrink',
  headers: {
    rejectUnauthorized: false,
    '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'
  }
};

fileList(root);

// 獲取文件列表
function fileList(folder) {
  fs.readdir(folder, (err, files) => {
    if (err) console.error(err);
    files.forEach(file => {
      fileFilter(folder + file);
    });
  });
}

// 過濾文件格式,返回全部jpg,png圖片
function fileFilter(file) {
  fs.stat(file, (err, stats) => {
    if (err) return console.error(err);
    if (
      // 必須是文件,小於5MB,後綴 jpg||png
      stats.size <= max &&
      stats.isFile() &&
      exts.includes(path.extname(file))
    ) {
      fileUpload(file); // console.log('能夠壓縮:' + file);
    }
    if (stats.isDirectory()) fileList(file + '/');
  });
}
// 異步API,壓縮圖片
// {"error":"Bad request","message":"Request is invalid"}
// {"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" }}
function fileUpload(img) {
  var req = https.request(options, function(res) {
    res.on('data', buf => {
      let obj = JSON.parse(buf.toString());
      if (obj.error) {
        console.log(`[${img}]:壓縮失敗!報錯:${obj.message}`);
      } else {
        fileUpdate(img, obj);
      }
    });
  });

  req.write(fs.readFileSync(img), 'binary');
  req.on('error', e => {
    console.error(e);
  });
  req.end();
}
// 該方法被循環調用,請求圖片數據
function fileUpdate(imgpath, obj) {
  let options = new URL(obj.output.url);
  let req = https.request(options, res => {
    let body = '';
    res.setEncoding('binary');
    res.on('data', function(data) {
      body += data;
    });

    res.on('end', function() {
      fs.writeFile(imgpath, body, 'binary', err => {
        if (err) return console.error(err);
        console.log(
          `[${imgpath}] \n 壓縮成功,原始大小-${obj.input.size},壓縮大小-${
            obj.output.size
          },優化比例-${obj.output.ratio}`
        );
      });
    });
  });
  req.on('error', e => {
    console.error(e);
  });
  req.end();
}
相關文章
相關標籤/搜索