mirror of https://github.com/curbengh/hexo-yam
215 lines
6.4 KiB
JavaScript
215 lines
6.4 KiB
JavaScript
'use strict'
|
|
|
|
const Htmlminifier = require('html-minifier').minify
|
|
const CleanCSS = require('clean-css')
|
|
const Terser = require('terser')
|
|
const Svgo = require('svgo')
|
|
const zlib = require('zlib')
|
|
const { promisify } = require('util')
|
|
const gzip = promisify(zlib.gzip)
|
|
const br = require('zlib').brotliCompress ? promisify(require('zlib').brotliCompress) : require('iltorb').compress
|
|
const micromatch = require('micromatch')
|
|
|
|
const isMatch = (path, patterns, options) => {
|
|
if (path && patterns && patterns.length) {
|
|
return micromatch.isMatch(path, patterns, options)
|
|
} else {
|
|
return false
|
|
}
|
|
}
|
|
|
|
function verbose (original, minified, path, ext) {
|
|
const saved = ((original.length - minified.length) / original.length * 100).toFixed(2)
|
|
const log = this.log || console
|
|
log.log(`${ext}: ${path} [${saved}% saved]`)
|
|
}
|
|
|
|
function error (msg) {
|
|
const log = this.log || console
|
|
log.error(msg)
|
|
}
|
|
|
|
function minifyHtml (str, data) {
|
|
const hexo = this
|
|
const options = hexo.config.minify.html
|
|
if (options.enable === false) return
|
|
|
|
const { path } = data
|
|
const { exclude, globOptions } = options
|
|
|
|
let excludeString = exclude || ''
|
|
if (Array.isArray(exclude)) excludeString = exclude.join('')
|
|
if (excludeString.includes('/')) globOptions.basename = false
|
|
|
|
// Return if a path matches exclusion pattern
|
|
if (isMatch(path, exclude, globOptions)) return str
|
|
|
|
const result = Htmlminifier(str, options)
|
|
if (options.logger) verbose.call(this, str, result, path, 'html')
|
|
|
|
return result
|
|
}
|
|
|
|
async function minifyCss (str, data) {
|
|
const hexo = this
|
|
const options = hexo.config.minify.css
|
|
if (options.enable === false) return
|
|
|
|
const { path } = data
|
|
const { exclude, globOptions } = options
|
|
|
|
let excludeString = exclude || ''
|
|
if (exclude && Array.isArray(exclude)) excludeString = exclude.join('')
|
|
if (excludeString && excludeString.includes('/')) globOptions.basename = false
|
|
|
|
if (isMatch(path, exclude, globOptions)) return str
|
|
|
|
try {
|
|
const result = await new CleanCSS(options).minify(str)
|
|
if (options.logger) verbose.call(this, str, result.styles, path, 'css')
|
|
return result.styles
|
|
} catch (err) {
|
|
error(err)
|
|
}
|
|
}
|
|
|
|
function minifyJs (str, data) {
|
|
const hexo = this
|
|
const options = hexo.config.minify.js
|
|
if (options.enable === false) return
|
|
|
|
const { path } = data
|
|
const { exclude, globOptions } = options
|
|
|
|
let excludeString = exclude || ''
|
|
if (exclude && Array.isArray(exclude)) excludeString = exclude.join('')
|
|
if (excludeString && excludeString.includes('/')) globOptions.basename = false
|
|
|
|
if (isMatch(path, exclude, globOptions)) return str
|
|
|
|
// Terser doesn't like unsupported options
|
|
const jsOptions = Object.assign({}, options)
|
|
delete jsOptions.enable
|
|
delete jsOptions.exclude
|
|
delete jsOptions.logger
|
|
delete jsOptions.globOptions
|
|
|
|
const result = Terser.minify(str, jsOptions)
|
|
if (options.logger) verbose.call(this, str, result.code, path, 'js')
|
|
|
|
return result.code
|
|
}
|
|
|
|
function minifySvg () {
|
|
const hexo = this
|
|
const options = hexo.config.minify.svg
|
|
if (options.enable === false) return
|
|
|
|
const { route } = hexo
|
|
const routeList = route.list()
|
|
const { globOptions, include } = options
|
|
|
|
let includeString = include || ''
|
|
if (include && Array.isArray(include)) includeString = include.join('')
|
|
if (includeString && includeString.includes('/')) globOptions.basename = false
|
|
|
|
return Promise.all((micromatch(routeList, include, globOptions)).map((path) => {
|
|
return new Promise((resolve, reject) => {
|
|
const assetPath = route.get(path)
|
|
const assetTxt = []
|
|
assetPath.on('data', (chunk) => (assetTxt.push(chunk)))
|
|
assetPath.on('end', async () => {
|
|
if (assetTxt.length) {
|
|
try {
|
|
const result = await new Svgo(options).optimize(assetTxt)
|
|
if (options.logger) verbose.call(this, assetTxt.toString(), result.data, path, 'svg')
|
|
resolve(route.set(path, result.data))
|
|
} catch (err) {
|
|
error.call(this, err)
|
|
reject(err)
|
|
}
|
|
}
|
|
})
|
|
})
|
|
}))
|
|
}
|
|
|
|
function gzipFn () {
|
|
const hexo = this
|
|
const options = hexo.config.minify.gzip
|
|
if (options.enable === false) return
|
|
|
|
const { route } = hexo
|
|
const routeList = route.list()
|
|
const { globOptions, include } = options
|
|
|
|
let includeString = include || ''
|
|
if (include && Array.isArray(include)) includeString = include.join('')
|
|
if (includeString && includeString.includes('/')) globOptions.basename = false
|
|
|
|
return Promise.all((micromatch(routeList, include, globOptions)).map((path) => {
|
|
return new Promise((resolve, reject) => {
|
|
const assetPath = route.get(path)
|
|
const assetTxt = []
|
|
assetPath.on('data', (chunk) => (assetTxt.push(chunk)))
|
|
assetPath.on('end', async () => {
|
|
if (assetTxt.length) {
|
|
try {
|
|
const input = Buffer.from(assetTxt[0], 'utf-8')
|
|
const result = await gzip(input, { level: zlib.constants.Z_BEST_COMPRESSION })
|
|
if (options.logger) verbose.call(this, input, result.toString(), path, 'gzip')
|
|
resolve(route.set(path + '.gz', result))
|
|
} catch (err) {
|
|
error(err)
|
|
reject(err)
|
|
}
|
|
}
|
|
})
|
|
})
|
|
}))
|
|
}
|
|
|
|
function brotliFn () {
|
|
const hexo = this
|
|
const options = hexo.config.minify.brotli
|
|
if (options.enable === false) return
|
|
|
|
const { route } = hexo
|
|
const routeList = route.list()
|
|
const { globOptions, include } = options
|
|
|
|
let includeString = include || ''
|
|
if (include && Array.isArray(include)) includeString = include.join('')
|
|
if (includeString && includeString.includes('/')) globOptions.basename = false
|
|
|
|
return Promise.all((micromatch(routeList, include, globOptions)).map((path) => {
|
|
return new Promise((resolve, reject) => {
|
|
const assetPath = route.get(path)
|
|
const assetTxt = []
|
|
assetPath.on('data', (chunk) => (assetTxt.push(chunk)))
|
|
assetPath.on('end', async () => {
|
|
if (assetTxt.length) {
|
|
try {
|
|
const input = Buffer.from(assetTxt[0], 'utf-8')
|
|
const result = await br(input)
|
|
if (options.logger) verbose.call(this, input, result.toString(), path, 'brotli')
|
|
resolve(route.set(path + '.br', result))
|
|
} catch (err) {
|
|
error(err)
|
|
reject(err)
|
|
}
|
|
}
|
|
})
|
|
})
|
|
}))
|
|
}
|
|
|
|
module.exports = {
|
|
minifyHtml: minifyHtml,
|
|
minifyCss: minifyCss,
|
|
minifyJs: minifyJs,
|
|
minifySvg: minifySvg,
|
|
gzipFn: gzipFn,
|
|
brotliFn: brotliFn
|
|
}
|