UNPKG

akasharender

Version:

Rendering support for generating static HTML websites or EPUB eBooks

241 lines 26.3 kB
/** * * Copyright 2014-2025 David Herron * * This file is part of AkashaCMS (http://akashacms.com/). * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import { promises as fs } from 'fs'; import path from 'path'; import fastq from 'fastq'; import * as data from '../data.js'; import { documentsCache, assetsCache, layoutsCache, partialsCache } from './file-cache-sqlite.js'; async function renderVPath(config, info) { await data.remove(info.mountPoint, info.vpath); let result = await config.akasha.renderPath(config, info.vpath); console.log(result); } /** * Search for all documents that use the layout which changed, and * rerender them. Obviously this function must only be called in * the handlers for 'change' and 'add' in layouts. * * @param config * @param info */ async function renderForLayout(config, info) { const docs = await documentsCache.search({ layouts: [info.vpath] }); const queue = fastq.promise(async function (item) { try { await data.remove(item.mountPoint, item.vpath); let result = await config.akasha.renderPath(config, item.vpath); console.log(result); return "ok"; } catch (err) { throw new Error(`renderForLayout FAIL to render ${item.vpath} because ${err.stack}`); } }, 10); const waitFor = []; for (let doc of docs) { waitFor.push(queue.push(doc)); } if (waitFor.length > 0) await Promise.all(waitFor); console.log(`FINISH rendering ${waitFor.length} documents from changing template ${info.vpath}`); } async function rebuild(config) { await data.removeAll(); let results = await config.akasha.render(config); for (let result of results) { if (result.error) { console.error(result.error); } else { console.log(result.result); } } return results; } async function unlinkVPath(config, info) { if (!config.renderDestination) { // console.log(`UNLINK ${collection} could not perform unlink (no renderDestination) for `, info); return; } const renderer = config.findRendererPath(info.vpath); let renderPath; if (renderer) { renderPath = renderer.filePath(info.vpath); } else { renderPath = info.vpath; } await fs.unlink(path.join(config.renderDestination, renderPath)); console.log(`UNLINK ${renderPath}`); } export async function watchman(config) { documentsCache .on('change', async (collection, info) => { try { await renderVPath(config, info); } catch (e) { documentsCache.emit('error', { code: 'change', collection: collection, vpath: info.vpath, error: e }); } }) .on('add', async (collection, info) => { try { await renderVPath(config, info); } catch (e) { documentsCache.emit('error', { code: 'add', collection: collection, vpath: info.vpath, error: e }); } }) .on('unlink', async (collection, info) => { try { // console.log(`UNLINK ${config.renderDestination} ${info.renderPath}`); await unlinkVPath(config, info); } catch (e) { documentsCache.emit('error', { code: 'unlink', collection: collection, vpath: info.vpath, error: e }); } }); console.log('... watching documents'); assetsCache .on('change', async (collection, info) => { try { const destFN = path.join(config.renderDestination, info.renderPath); await fs.copyFile(info.fspath, destFN); console.log(`CHANGE ${info.vpath} COPY==> ${info.renderPath}`); } catch (e) { assetsCache.emit('error', { code: 'change', collection: collection, vpath: info.vpath, error: e }); } }) .on('add', async (collection, info) => { try { const destFN = path.join(config.renderDestination, info.renderPath); await fs.copyFile(info.fspath, destFN); console.log(`ADD ${info.vpath} COPY==> ${info.renderPath}`); } catch (e) { assetsCache.emit('error', { code: 'add', collection: collection, vpath: info.vpath, error: e }); } }) .on('unlink', async (collection, info) => { try { await fs.unlink(path.join(config.renderDestination, info.renderPath)); console.log(`UNLINK ${info.renderPath}`); } catch (e) { assetsCache.emit('error', { code: 'unlink', collection: collection, vpath: info.vpath, error: e }); } }); console.log('... watching assets'); layoutsCache .on('change', async (collection, info) => { try { await renderForLayout(config, info); } catch (e) { layoutsCache.emit('error', { code: 'change', collection: collection, vpath: info.vpath, error: e }); } }) .on('add', async (collection, info) => { try { await renderForLayout(config, info); } catch (e) { layoutsCache.emit('error', { code: 'add', collection: collection, vpath: info.vpath, error: e }); } }) .on('unlink', (collection, info) => { // Nothing to do }); console.log('... watching layouts'); partialsCache .on('change', async (collection, info) => { try { await rebuild(config); } catch (e) { partialsCache.emit('error', { code: 'change', collection: collection, vpath: info.vpath, error: e }); } }) .on('add', async (collection, info) => { try { await rebuild(config); } catch (e) { partialsCache.emit('error', { code: 'add', collection: collection, vpath: info.vpath, error: e }); } }) .on('unlink', (collection, info) => { // Nothing to do }); console.log('... watching partials'); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"watchman.js","sourceRoot":"","sources":["../../lib/cache/watchman.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;GAiBG;AAEH,OAAO,EAAE,QAAQ,IAAI,EAAE,EAAE,MAAM,IAAI,CAAC;AACpC,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,KAAK,IAAI,MAAM,YAAY,CAAC;AACnC,OAAO,EACH,cAAc,EAAE,WAAW,EAAE,YAAY,EAAE,aAAa,EAC3D,MAAM,wBAAwB,CAAC;AAEhC,KAAK,UAAU,WAAW,CAAC,MAAM,EAAE,IAAI;IACnC,MAAM,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;IAC/C,IAAI,MAAM,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC,UAAU,CAAC,MAAM,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;IAChE,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;AACxB,CAAC;AAED;;;;;;;GAOG;AACH,KAAK,UAAU,eAAe,CAAC,MAAM,EAAE,IAAI;IACvC,MAAM,IAAI,GAAG,MAAM,cAAc,CAAC,MAAM,CAAC;QACrC,OAAO,EAAE,CAAE,IAAI,CAAC,KAAK,CAAE;KAC1B,CAAC,CAAC;IAEH,MAAM,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,KAAK,WAAU,IAAI;QAC3C,IAAI,CAAC;YACD,MAAM,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;YAC/C,IAAI,MAAM,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC,UAAU,CAAC,MAAM,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;YAChE,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;YACpB,OAAO,IAAI,CAAC;QAChB,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CAAC,kCAAkC,IAAI,CAAC,KAAK,YAAY,GAAG,CAAC,KAAK,EAAE,CAAC,CAAC;QACzF,CAAC;IACL,CAAC,EAAE,EAAE,CAAC,CAAC;IAGP,MAAM,OAAO,GAAG,EAAE,CAAC;IACnB,KAAK,IAAI,GAAG,IAAI,IAAI,EAAE,CAAC;QACnB,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;IAClC,CAAC;IAED,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC;QAAE,MAAM,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IACnD,OAAO,CAAC,GAAG,CAAC,oBAAoB,OAAO,CAAC,MAAM,qCAAqC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;AACrG,CAAC;AAED,KAAK,UAAU,OAAO,CAAC,MAAM;IACzB,MAAM,IAAI,CAAC,SAAS,EAAE,CAAC;IACvB,IAAI,OAAO,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;IACjD,KAAK,IAAI,MAAM,IAAI,OAAO,EAAE,CAAC;QACzB,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAChC,CAAC;aAAM,CAAC;YACJ,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QAC/B,CAAC;IACL,CAAC;IACD,OAAO,OAAO,CAAC;AACnB,CAAC;AAED,KAAK,UAAU,WAAW,CAAC,MAAM,EAAE,IAAI;IACnC,IAAI,CAAC,MAAM,CAAC,iBAAiB,EAAE,CAAC;QAC5B,kGAAkG;QAClG,OAAO;IACX,CAAC;IACD,MAAM,QAAQ,GAAG,MAAM,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACrD,IAAI,UAAU,CAAC;IACf,IAAI,QAAQ,EAAE,CAAC;QACX,UAAU,GAAG,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC/C,CAAC;SAAM,CAAC;QACJ,UAAU,GAAG,IAAI,CAAC,KAAK,CAAC;IAC5B,CAAC;IACD,MAAM,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,iBAAiB,EAAE,UAAU,CAAC,CAAC,CAAC;IACjE,OAAO,CAAC,GAAG,CAAC,UAAU,UAAU,EAAE,CAAC,CAAC;AACxC,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,QAAQ,CAAC,MAAM;IACjC,cAAc;SACb,EAAE,CAAC,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE;QACrC,IAAI,CAAC;YACD,MAAM,WAAW,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;QACpC,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACT,cAAc,CAAC,IAAI,CAAC,OAAO,EAAE;gBACzB,IAAI,EAAE,QAAQ;gBACd,UAAU,EAAE,UAAU;gBACtB,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,KAAK,EAAE,CAAC;aACX,CAAC,CAAC;QACP,CAAC;IACL,CAAC,CAAC;SACD,EAAE,CAAC,KAAK,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE;QAClC,IAAI,CAAC;YACD,MAAM,WAAW,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;QACpC,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACT,cAAc,CAAC,IAAI,CAAC,OAAO,EAAE;gBACzB,IAAI,EAAE,KAAK;gBACX,UAAU,EAAE,UAAU;gBACtB,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,KAAK,EAAE,CAAC;aACX,CAAC,CAAC;QACP,CAAC;IACL,CAAC,CAAC;SACD,EAAE,CAAC,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE;QACrC,IAAI,CAAC;YACD,wEAAwE;YACxE,MAAM,WAAW,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;QACpC,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACT,cAAc,CAAC,IAAI,CAAC,OAAO,EAAE;gBACzB,IAAI,EAAE,QAAQ;gBACd,UAAU,EAAE,UAAU;gBACtB,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,KAAK,EAAE,CAAC;aACX,CAAC,CAAC;QACP,CAAC;IACL,CAAC,CAAC,CAAC;IACH,OAAO,CAAC,GAAG,CAAC,wBAAwB,CAAC,CAAC;IAEtC,WAAW;SACV,EAAE,CAAC,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE;QACrC,IAAI,CAAC;YACD,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,iBAAiB,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;YACpE,MAAM,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;YACvC,OAAO,CAAC,GAAG,CAAC,UAAU,IAAI,CAAC,KAAK,YAAY,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC;QACnE,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACT,WAAW,CAAC,IAAI,CAAC,OAAO,EAAE;gBACtB,IAAI,EAAE,QAAQ;gBACd,UAAU,EAAE,UAAU;gBACtB,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,KAAK,EAAE,CAAC;aACX,CAAC,CAAC;QACP,CAAC;IACL,CAAC,CAAC;SACD,EAAE,CAAC,KAAK,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE;QAClC,IAAI,CAAC;YACD,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,iBAAiB,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;YACpE,MAAM,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;YACvC,OAAO,CAAC,GAAG,CAAC,OAAO,IAAI,CAAC,KAAK,YAAY,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC;QAChE,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACT,WAAW,CAAC,IAAI,CAAC,OAAO,EAAE;gBACtB,IAAI,EAAE,KAAK;gBACX,UAAU,EAAE,UAAU;gBACtB,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,KAAK,EAAE,CAAC;aACX,CAAC,CAAC;QACP,CAAC;IACL,CAAC,CAAC;SACD,EAAE,CAAC,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE;QACrC,IAAI,CAAC;YACD,MAAM,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,iBAAiB,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC;YACtE,OAAO,CAAC,GAAG,CAAC,UAAU,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC;QAC7C,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACT,WAAW,CAAC,IAAI,CAAC,OAAO,EAAE;gBACtB,IAAI,EAAE,QAAQ;gBACd,UAAU,EAAE,UAAU;gBACtB,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,KAAK,EAAE,CAAC;aACX,CAAC,CAAC;QACP,CAAC;IACL,CAAC,CAAC,CAAC;IACH,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,CAAC;IAEnC,YAAY;SACX,EAAE,CAAC,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE;QACrC,IAAI,CAAC;YACD,MAAM,eAAe,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;QACxC,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACT,YAAY,CAAC,IAAI,CAAC,OAAO,EAAE;gBACvB,IAAI,EAAE,QAAQ;gBACd,UAAU,EAAE,UAAU;gBACtB,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,KAAK,EAAE,CAAC;aACX,CAAC,CAAC;QACP,CAAC;IACL,CAAC,CAAC;SACD,EAAE,CAAC,KAAK,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE;QAClC,IAAI,CAAC;YACD,MAAM,eAAe,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;QACxC,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACT,YAAY,CAAC,IAAI,CAAC,OAAO,EAAE;gBACvB,IAAI,EAAE,KAAK;gBACX,UAAU,EAAE,UAAU;gBACtB,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,KAAK,EAAE,CAAC;aACX,CAAC,CAAC;QACP,CAAC;IACL,CAAC,CAAC;SACD,EAAE,CAAC,QAAQ,EAAE,CAAC,UAAU,EAAE,IAAI,EAAE,EAAE;QAC/B,gBAAgB;IACpB,CAAC,CAAC,CAAC;IACH,OAAO,CAAC,GAAG,CAAC,sBAAsB,CAAC,CAAC;IAEpC,aAAa;SACZ,EAAE,CAAC,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE;QACrC,IAAI,CAAC;YACD,MAAM,OAAO,CAAC,MAAM,CAAC,CAAC;QAC1B,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACT,aAAa,CAAC,IAAI,CAAC,OAAO,EAAE;gBACxB,IAAI,EAAE,QAAQ;gBACd,UAAU,EAAE,UAAU;gBACtB,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,KAAK,EAAE,CAAC;aACX,CAAC,CAAC;QACP,CAAC;IACL,CAAC,CAAC;SACD,EAAE,CAAC,KAAK,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE;QAClC,IAAI,CAAC;YACD,MAAM,OAAO,CAAC,MAAM,CAAC,CAAC;QAC1B,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACT,aAAa,CAAC,IAAI,CAAC,OAAO,EAAE;gBACxB,IAAI,EAAE,KAAK;gBACX,UAAU,EAAE,UAAU;gBACtB,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,KAAK,EAAE,CAAC;aACX,CAAC,CAAC;QACP,CAAC;IACL,CAAC,CAAC;SACD,EAAE,CAAC,QAAQ,EAAE,CAAC,UAAU,EAAE,IAAI,EAAE,EAAE;QAC/B,gBAAgB;IACpB,CAAC,CAAC,CAAC;IACH,OAAO,CAAC,GAAG,CAAC,uBAAuB,CAAC,CAAC;AACzC,CAAC","sourcesContent":["/**\n *\n * Copyright 2014-2025 David Herron\n *\n * This file is part of AkashaCMS (http://akashacms.com/).\n *\n *  Licensed under the Apache License, Version 2.0 (the \"License\");\n *  you may not use this file except in compliance with the License.\n *  You may obtain a copy of the License at\n *\n *      http://www.apache.org/licenses/LICENSE-2.0\n *\n *  Unless required by applicable law or agreed to in writing, software\n *  distributed under the License is distributed on an \"AS IS\" BASIS,\n *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n *  See the License for the specific language governing permissions and\n *  limitations under the License.\n */\n\nimport { promises as fs } from 'fs';\nimport path from 'path';\nimport fastq from 'fastq';\nimport * as data from '../data.js';\nimport {\n    documentsCache, assetsCache, layoutsCache, partialsCache\n} from './file-cache-sqlite.js';\n\nasync function renderVPath(config, info) {\n    await data.remove(info.mountPoint, info.vpath);\n    let result = await config.akasha.renderPath(config, info.vpath);\n    console.log(result);\n}\n\n/**\n * Search for all documents that use the layout which changed, and\n * rerender them.  Obviously this function must only be called in\n * the handlers for 'change' and 'add' in layouts.\n * \n * @param config \n * @param info \n */\nasync function renderForLayout(config, info) {\n    const docs = await documentsCache.search({\n        layouts: [ info.vpath ]\n    });\n\n    const queue = fastq.promise(async function(item) {\n        try {\n            await data.remove(item.mountPoint, item.vpath);\n            let result = await config.akasha.renderPath(config, item.vpath);\n            console.log(result);\n            return \"ok\";\n        } catch (err) {\n            throw new Error(`renderForLayout FAIL to render ${item.vpath} because ${err.stack}`);\n        }\n    }, 10);\n\n\n    const waitFor = [];\n    for (let doc of docs) {\n        waitFor.push(queue.push(doc));\n    }\n\n    if (waitFor.length > 0) await Promise.all(waitFor);\n    console.log(`FINISH rendering ${waitFor.length} documents from changing template ${info.vpath}`);\n}\n\nasync function rebuild(config) {\n    await data.removeAll();\n    let results = await config.akasha.render(config);\n    for (let result of results) {\n        if (result.error) {\n            console.error(result.error);\n        } else {\n            console.log(result.result);\n        }\n    }\n    return results;\n}\n\nasync function unlinkVPath(config, info) {\n    if (!config.renderDestination) {\n        // console.log(`UNLINK ${collection} could not perform unlink (no renderDestination) for `, info);\n        return;\n    }\n    const renderer = config.findRendererPath(info.vpath);\n    let renderPath;\n    if (renderer) {\n        renderPath = renderer.filePath(info.vpath);\n    } else {\n        renderPath = info.vpath;\n    }\n    await fs.unlink(path.join(config.renderDestination, renderPath));\n    console.log(`UNLINK ${renderPath}`);\n}\n\nexport async function watchman(config) {\n    documentsCache\n    .on('change', async (collection, info) => {\n        try {\n            await renderVPath(config, info);\n        } catch (e) {\n            documentsCache.emit('error', {\n                code: 'change',\n                collection: collection,\n                vpath: info.vpath,\n                error: e\n            });\n        }\n    })\n    .on('add', async (collection, info) => {\n        try {\n            await renderVPath(config, info);\n        } catch (e) {\n            documentsCache.emit('error', {\n                code: 'add',\n                collection: collection,\n                vpath: info.vpath,\n                error: e\n            });\n        }\n    })\n    .on('unlink', async (collection, info) => {\n        try {\n            // console.log(`UNLINK ${config.renderDestination} ${info.renderPath}`);\n            await unlinkVPath(config, info);\n        } catch (e) {\n            documentsCache.emit('error', {\n                code: 'unlink',\n                collection: collection,\n                vpath: info.vpath,\n                error: e\n            });\n        }\n    });\n    console.log('... watching documents');\n\n    assetsCache\n    .on('change', async (collection, info) => {\n        try {\n            const destFN = path.join(config.renderDestination, info.renderPath);\n            await fs.copyFile(info.fspath, destFN);\n            console.log(`CHANGE ${info.vpath} COPY==> ${info.renderPath}`);\n        } catch (e) {\n            assetsCache.emit('error', {\n                code: 'change',\n                collection: collection,\n                vpath: info.vpath,\n                error: e\n            });\n        }\n    })\n    .on('add', async (collection, info) => {\n        try {\n            const destFN = path.join(config.renderDestination, info.renderPath);\n            await fs.copyFile(info.fspath, destFN);\n            console.log(`ADD ${info.vpath} COPY==> ${info.renderPath}`);\n        } catch (e) {\n            assetsCache.emit('error', {\n                code: 'add',\n                collection: collection,\n                vpath: info.vpath,\n                error: e\n            });\n        }\n    })\n    .on('unlink', async (collection, info) => {\n        try {\n            await fs.unlink(path.join(config.renderDestination, info.renderPath));\n            console.log(`UNLINK ${info.renderPath}`);\n        } catch (e) {\n            assetsCache.emit('error', {\n                code: 'unlink',\n                collection: collection,\n                vpath: info.vpath,\n                error: e\n            });\n        }\n    });\n    console.log('... watching assets');\n\n    layoutsCache\n    .on('change', async (collection, info) => {\n        try {\n            await renderForLayout(config, info);\n        } catch (e) {\n            layoutsCache.emit('error', {\n                code: 'change',\n                collection: collection,\n                vpath: info.vpath,\n                error: e\n            });\n        }\n    })\n    .on('add', async (collection, info) => {\n        try {\n            await renderForLayout(config, info);\n        } catch (e) {\n            layoutsCache.emit('error', {\n                code: 'add',\n                collection: collection,\n                vpath: info.vpath,\n                error: e\n            });\n        }\n    })\n    .on('unlink', (collection, info) => {\n        // Nothing to do\n    });\n    console.log('... watching layouts');\n\n    partialsCache\n    .on('change', async (collection, info) => {\n        try {\n            await rebuild(config);\n        } catch (e) {\n            partialsCache.emit('error', {\n                code: 'change',\n                collection: collection,\n                vpath: info.vpath,\n                error: e\n            });\n        }\n    })\n    .on('add', async (collection, info) => {\n        try {\n            await rebuild(config);\n        } catch (e) {\n            partialsCache.emit('error', {\n                code: 'add',\n                collection: collection,\n                vpath: info.vpath,\n                error: e\n            });\n        }\n    })\n    .on('unlink', (collection, info) => {\n        // Nothing to do\n    });\n    console.log('... watching partials');\n}\n"]}