2016-05-26 11:09:41 +00:00
|
|
|
/* global hexo */
|
2018-10-25 09:35:59 +00:00
|
|
|
'use strict'
|
2019-04-23 08:16:56 +00:00
|
|
|
const Htmlminifier = require('html-minifier').minify
|
2018-10-26 04:57:23 +00:00
|
|
|
const CleanCSS = require('clean-css')
|
2019-04-10 09:54:02 +00:00
|
|
|
const Terser = require('terser')
|
2019-04-23 08:16:56 +00:00
|
|
|
const svgo = require('svgo')
|
2018-10-26 04:57:23 +00:00
|
|
|
const zlib = require('zlib')
|
|
|
|
const br = require('iltorb')
|
2019-04-23 08:16:56 +00:00
|
|
|
const nanomatch = require('nanomatch')
|
2016-05-26 11:09:41 +00:00
|
|
|
|
2018-10-25 09:35:59 +00:00
|
|
|
function logicHtml (str, data) {
|
2018-10-26 04:57:23 +00:00
|
|
|
const hexo = this
|
|
|
|
const options = hexo.config.neat_html
|
2018-10-25 09:35:59 +00:00
|
|
|
// Return if disabled.
|
|
|
|
if (options.enable === false) return
|
|
|
|
|
2018-10-26 04:57:23 +00:00
|
|
|
let path = data.path
|
|
|
|
let exclude = options.exclude
|
2018-10-25 09:35:59 +00:00
|
|
|
|
2019-04-23 04:05:59 +00:00
|
|
|
// Return if a path matches exclusion pattern
|
2018-10-25 09:35:59 +00:00
|
|
|
if (path && exclude && exclude.length) {
|
2019-04-23 04:05:59 +00:00
|
|
|
if (nanomatch.some(path, exclude, { matchBase: true })) return str
|
2018-10-25 09:35:59 +00:00
|
|
|
}
|
|
|
|
|
2018-10-26 04:57:23 +00:00
|
|
|
let result = Htmlminifier(str, options)
|
|
|
|
let saved = ((str.length - result.length) / str.length * 100).toFixed(2)
|
2018-10-25 09:35:59 +00:00
|
|
|
if (options.logger) {
|
2018-10-26 04:57:23 +00:00
|
|
|
let log = hexo.log || console.log
|
2018-10-25 09:35:59 +00:00
|
|
|
log.log('Minify the html: %s [%s saved]', path, saved + '%')
|
|
|
|
}
|
|
|
|
return result
|
2018-09-29 05:38:45 +00:00
|
|
|
}
|
2016-05-26 11:09:41 +00:00
|
|
|
|
2018-10-25 09:35:59 +00:00
|
|
|
function logicCss (str, data) {
|
2018-10-26 04:57:23 +00:00
|
|
|
const hexo = this
|
|
|
|
const options = hexo.config.neat_css
|
2018-10-25 09:35:59 +00:00
|
|
|
// Return if disabled.
|
|
|
|
if (options.enable === false) return
|
2016-05-26 11:09:41 +00:00
|
|
|
|
2018-10-26 04:57:23 +00:00
|
|
|
let path = data.path
|
|
|
|
let exclude = options.exclude
|
2016-06-22 11:15:47 +00:00
|
|
|
|
2019-04-23 04:05:59 +00:00
|
|
|
// Return if a path matches exclusion pattern
|
2018-10-25 09:35:59 +00:00
|
|
|
if (path && exclude && exclude.length) {
|
2019-04-23 04:05:59 +00:00
|
|
|
if (nanomatch.some(path, exclude, { matchBase: true })) return str
|
2018-10-25 09:35:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return new Promise(function (resolve, reject) {
|
2019-04-23 03:56:43 +00:00
|
|
|
new CleanCSS(options).minify(str, function (err, result) {
|
2018-10-25 09:35:59 +00:00
|
|
|
if (err) return reject(err)
|
2018-10-26 04:57:23 +00:00
|
|
|
let saved = ((str.length - result.styles.length) / str.length * 100).toFixed(2)
|
2018-10-25 09:35:59 +00:00
|
|
|
resolve(result.styles)
|
|
|
|
if (options.logger) {
|
2018-10-26 04:57:23 +00:00
|
|
|
let log = hexo.log || console.log
|
2018-10-25 09:35:59 +00:00
|
|
|
log.log('Minify the css: %s [%s saved]', path, saved + '%')
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
2016-05-26 11:09:41 +00:00
|
|
|
}
|
|
|
|
|
2018-10-25 09:35:59 +00:00
|
|
|
function logicJs (str, data) {
|
2018-10-26 04:57:23 +00:00
|
|
|
const hexo = this
|
|
|
|
const options = hexo.config.neat_js
|
2018-10-25 09:35:59 +00:00
|
|
|
// Return if disabled.
|
|
|
|
if (options.enable === false) return
|
2016-05-26 11:09:41 +00:00
|
|
|
|
2018-10-26 04:57:23 +00:00
|
|
|
let path = data.path
|
|
|
|
let exclude = options.exclude
|
2016-05-26 11:09:41 +00:00
|
|
|
|
2019-04-23 04:05:59 +00:00
|
|
|
// Return if a path matches exclusion pattern
|
2018-10-25 09:35:59 +00:00
|
|
|
if (path && exclude && exclude.length) {
|
2019-04-23 04:05:59 +00:00
|
|
|
if (nanomatch.some(path, exclude, { matchBase: true })) return str
|
2018-10-25 09:35:59 +00:00
|
|
|
}
|
|
|
|
|
2019-04-10 09:54:02 +00:00
|
|
|
// Terser doesn't like unsupported options
|
2019-04-23 04:05:59 +00:00
|
|
|
const jsOptions = Object.assign({}, options)
|
2019-04-23 03:00:47 +00:00
|
|
|
delete jsOptions.enable
|
|
|
|
delete jsOptions.exclude
|
|
|
|
delete jsOptions.logger
|
2018-10-25 09:35:59 +00:00
|
|
|
|
2019-04-23 03:00:47 +00:00
|
|
|
let result = Terser.minify(str, jsOptions)
|
2018-10-26 04:57:23 +00:00
|
|
|
let saved = ((str.length - result.code.length) / str.length * 100).toFixed(2)
|
2019-04-23 03:00:47 +00:00
|
|
|
if (options.logger) {
|
2018-10-26 04:57:23 +00:00
|
|
|
let log = hexo.log || console.log
|
2018-10-25 09:35:59 +00:00
|
|
|
log.log('Minify the js: %s [%s saved]', path, saved + '%')
|
|
|
|
}
|
|
|
|
return result.code
|
2016-05-26 11:09:41 +00:00
|
|
|
}
|
|
|
|
|
2019-04-23 07:59:35 +00:00
|
|
|
function logicSvg () {
|
|
|
|
const hexo = this
|
|
|
|
const options = hexo.config.neat_svg
|
|
|
|
// Return if disabled.
|
|
|
|
if (options.enable === false) return
|
|
|
|
|
|
|
|
let route = hexo.route
|
|
|
|
let routeList = route.list()
|
|
|
|
let include = options.include
|
|
|
|
|
|
|
|
return Promise.all((nanomatch(routeList, include, { matchBase: true })).map(path => {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
// Grab all assets using hexo router
|
|
|
|
let assetPath = route.get(path)
|
|
|
|
let assetTxt = ''
|
|
|
|
// Extract the content
|
|
|
|
assetPath.on('data', (chunk) => (assetTxt += chunk))
|
|
|
|
assetPath.on('end', () => {
|
|
|
|
if (assetTxt.length) {
|
2019-04-23 08:27:52 +00:00
|
|
|
// Minify using svgo
|
2019-05-02 06:25:43 +00:00
|
|
|
new svgo(options).optimize(assetTxt).then(function (result) {
|
|
|
|
// Replace the original file with the minified.
|
|
|
|
route.set(path, result.data)
|
|
|
|
// Logging
|
|
|
|
let saved = ((assetTxt.length - result.data.length) / assetTxt.length * 100).toFixed(2)
|
|
|
|
if (options.logger) {
|
|
|
|
let log = hexo.log || console.log
|
|
|
|
log.log('Minify the svg: %s [%s saved]', path, saved + '%')
|
|
|
|
}
|
|
|
|
resolve(assetTxt)
|
2019-04-23 07:59:35 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
2018-10-25 09:35:59 +00:00
|
|
|
function logicGzip () {
|
2018-10-26 04:57:23 +00:00
|
|
|
const hexo = this
|
|
|
|
const options = hexo.config.neat_gzip
|
2018-10-25 09:35:59 +00:00
|
|
|
// Return if disabled.
|
|
|
|
if (options.enable === false) return
|
|
|
|
|
2018-10-26 04:57:23 +00:00
|
|
|
let route = hexo.route
|
|
|
|
let routeList = route.list()
|
2019-04-23 05:40:43 +00:00
|
|
|
let include = options.include
|
2018-10-25 09:35:59 +00:00
|
|
|
|
2019-04-23 05:40:43 +00:00
|
|
|
return Promise.all((nanomatch(routeList, include, { matchBase: true })).map(path => {
|
2018-10-25 09:35:59 +00:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
// Grab all assets using hexo router
|
2018-10-26 04:57:23 +00:00
|
|
|
let assetPath = route.get(path)
|
2018-10-25 09:35:59 +00:00
|
|
|
let assetTxt = ''
|
|
|
|
// Extract the content
|
|
|
|
assetPath.on('data', (chunk) => (assetTxt += chunk))
|
|
|
|
assetPath.on('end', () => {
|
|
|
|
if (assetTxt.length) {
|
|
|
|
// gzip compress using highest level
|
2019-04-22 10:38:50 +00:00
|
|
|
zlib.gzip(assetTxt, { level: zlib.constants.Z_BEST_COMPRESSION }, (err, Input) => {
|
2018-10-25 09:35:59 +00:00
|
|
|
if (!err) {
|
|
|
|
// Save the compressed file to .gz
|
|
|
|
route.set(path + '.gz', Input)
|
|
|
|
// Logging
|
2018-10-26 04:57:23 +00:00
|
|
|
let saved = ((assetTxt.length - Input.toString().length) / assetTxt.length * 100).toFixed(2)
|
2018-10-25 09:35:59 +00:00
|
|
|
if (options.logger) {
|
2018-10-26 04:57:23 +00:00
|
|
|
let log = hexo.log || console.log
|
2018-10-25 09:35:59 +00:00
|
|
|
log.log('Gzip-compressed %s [%s saved]', path, saved + '%')
|
2018-09-30 07:30:32 +00:00
|
|
|
}
|
2018-10-25 09:35:59 +00:00
|
|
|
resolve(assetTxt)
|
|
|
|
} else {
|
|
|
|
reject(err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}))
|
2018-09-30 07:30:32 +00:00
|
|
|
}
|
|
|
|
|
2018-10-25 09:35:59 +00:00
|
|
|
function logicBrotli () {
|
2018-10-26 04:57:23 +00:00
|
|
|
const hexo = this
|
|
|
|
const options = hexo.config.neat_brotli
|
2018-10-25 09:35:59 +00:00
|
|
|
// Return if disabled.
|
|
|
|
if (options.enable === false) return
|
|
|
|
|
2018-10-26 04:57:23 +00:00
|
|
|
let route = hexo.route
|
|
|
|
let routeList = route.list()
|
2019-04-23 05:40:43 +00:00
|
|
|
let include = options.include
|
2018-10-25 09:35:59 +00:00
|
|
|
|
2019-04-23 05:40:43 +00:00
|
|
|
return Promise.all((nanomatch(routeList, include, { matchBase: true })).map(path => {
|
2018-10-25 09:35:59 +00:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
// Grab all assets using hexo router
|
2018-10-26 04:57:23 +00:00
|
|
|
let assetPath = route.get(path)
|
2018-10-25 09:35:59 +00:00
|
|
|
let assetTxt = ''
|
|
|
|
// Extract the content
|
|
|
|
assetPath.on('data', (chunk) => (assetTxt += chunk))
|
|
|
|
assetPath.on('end', () => {
|
|
|
|
if (assetTxt.length) {
|
|
|
|
// Input has to be buffer for brotli
|
2019-05-02 07:04:11 +00:00
|
|
|
let input = Buffer.from(assetTxt, 'utf-8')
|
2019-04-22 08:58:00 +00:00
|
|
|
// brotli defaults to max compression level
|
|
|
|
br.compress(input, (err, output) => {
|
2018-10-25 09:35:59 +00:00
|
|
|
if (!err) {
|
|
|
|
// Save the compressed file to .br
|
|
|
|
route.set(path + '.br', output)
|
|
|
|
// Logging
|
2018-10-26 04:57:23 +00:00
|
|
|
let saved = ((input.length - output.toString().length) / input.length * 100).toFixed(2)
|
2018-10-25 09:35:59 +00:00
|
|
|
if (options.logger) {
|
2018-10-26 04:57:23 +00:00
|
|
|
let log = hexo.log || console.log
|
2018-10-25 09:35:59 +00:00
|
|
|
log.log('Brotli-compressed %s [%s saved]', path, saved + '%')
|
2018-09-29 05:38:45 +00:00
|
|
|
}
|
2018-10-25 09:35:59 +00:00
|
|
|
resolve(assetTxt)
|
|
|
|
} else {
|
|
|
|
reject(err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}))
|
2018-09-28 07:43:54 +00:00
|
|
|
}
|
|
|
|
|
2016-05-26 11:09:41 +00:00
|
|
|
module.exports = {
|
2018-10-25 09:35:59 +00:00
|
|
|
logicHtml: logicHtml,
|
|
|
|
logicCss: logicCss,
|
|
|
|
logicJs: logicJs,
|
2019-04-23 07:59:35 +00:00
|
|
|
logicSvg: logicSvg,
|
2018-10-25 09:35:59 +00:00
|
|
|
logicGzip: logicGzip,
|
|
|
|
logicBrotli: logicBrotli
|
|
|
|
}
|