UNPKG

lib-tools

Version:

The lib-tools helps you simplify the build, bundle, test and npm packaging workflows for Typescript, JavaScript, Angular library projects and assets module projects.

155 lines 21.3 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); const webpack = require("webpack"); const webpackDevMiddleware = require("webpack-dev-middleware"); const yargs = require("yargs"); const helpers_1 = require("../helpers"); const utils_1 = require("../utils"); const configs_1 = require("../webpack/configs"); const failure_karma_webpack_plugin_1 = require("../webpack/plugins/failure-karma-webpack-plugin"); const turn_off_watch_webpack_plugin_1 = require("../webpack/plugins/turn-off-watch-webpack-plugin"); let blocked = []; let isBlocked = false; let webpackMiddleware; function requestBlocker() { return (_1, _2, next) => { if (isBlocked) { blocked.push(next); } else { next(); } }; } function fallbackMiddleware() { return (req, res, next) => { if (webpackMiddleware) { const webpackUrl = `/_karma_webpack_${req.url}`; const webpackReq = Object.assign(Object.assign({}, req), { url: webpackUrl }); webpackMiddleware(webpackReq, res, next); } else { next(); } }; } const init = async (config, emitter, customFileHandlers) => { const logger = new utils_1.Logger({ logLevel: 'info' }); config.customContextFile = `${__dirname}/karma-context.html`; config.customDebugFile = `${__dirname}/karma-debug.html`; config.beforeMiddleware = config.beforeMiddleware || []; config.beforeMiddleware.push('lib-tools--blocker'); config.middleware = config.middleware || []; config.middleware.push('lib-tools--fallback'); const webpackMiddlewareConfig = { publicPath: '/_karma_webpack_/' }; const compilationErrorCallback = (_, errors) => { emitter.emit('compile_error', errors); emitter.emit('run_complete', [], { exitCode: 1 }); unblock(); }; let webpackConfig; if (config.webpackConfig) { webpackConfig = config.webpackConfig; } else { const args = process.argv.length > 4 && /(\\|\/)?karma(\.js)?$/i.test(process.argv[1]) ? process.argv.slice(4) : []; const commandOptions = yargs(args) .option('browsers', { type: 'string' }) .option('reporters', { type: 'string' }) .option('codeCoverageExclude', { type: 'string' }).argv; const testConfig = await helpers_1.getTestConfigFromKarma(config, commandOptions); if (!testConfig) { throw new Error('Could not load workflow test config.'); } if (commandOptions.reporters == null && testConfig.reporters != null) { config.reporters = testConfig.reporters; } if (commandOptions.browsers == null && testConfig.browsers != null) { config.browsers = testConfig.browsers; } webpackConfig = await configs_1.getWebpackTestConfig(testConfig, commandOptions); } webpackConfig.plugins = webpackConfig.plugins || []; webpackConfig.plugins.push(new failure_karma_webpack_plugin_1.FailureKarmaWebpackPlugin(compilationErrorCallback)); webpackConfig.watch = !config.singleRun; if (config.singleRun) { webpackConfig.plugins.unshift(new turn_off_watch_webpack_plugin_1.TurnOffWatchWebpackPlugin()); } function unblock() { isBlocked = false; blocked.forEach((cb) => cb()); blocked = []; } let compiler; try { compiler = webpack(webpackConfig); } catch (e) { logger.error(e.stack || e); if (e.details) { logger.error(e.details); } throw e; } function handler(callback) { isBlocked = true; if (typeof callback === 'function') { callback(); } } compiler.hooks.invalid.tap('karma', () => handler()); compiler.hooks.watchRun.tapAsync('karma', (_, callback) => handler(callback)); compiler.hooks.run.tapAsync('karma', (_, callback) => handler(callback)); let lastCompilationHash; compiler.hooks.done.tap('karma', (stats) => { if (stats.compilation.errors.length > 0) { logger.error(stats.toString('errors-only')); lastCompilationHash = undefined; } else if (stats.hash !== lastCompilationHash) { lastCompilationHash = stats.hash; emitter.refreshFiles(); } unblock(); }); webpackMiddleware = webpackDevMiddleware(compiler, webpackMiddlewareConfig); customFileHandlers.push({ urlRegex: /^\/_karma_webpack_\/.*/, handler: (req, res) => { webpackMiddleware(req, res, function () { const alwaysServe = [ '/_karma_webpack_/runtime.js', '/_karma_webpack_/polyfills.js', '/_karma_webpack_/vendor.js' ]; if (alwaysServe.includes(req.url)) { res.statusCode = 200; res.end(); } else { res.statusCode = 404; res.end('Not found'); } }); } }); emitter.on('exit', (done) => { webpackMiddleware.close(); done(); }); }; init.$inject = ['config', 'emitter', 'customFileHandlers']; module.exports = { 'framework:lib-tools': ['factory', init], 'middleware:lib-tools--blocker': ['factory', requestBlocker], 'middleware:lib-tools--fallback': ['factory', fallbackMiddleware] }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"./","sources":["karma-plugin/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as http from 'http';\n\nimport { ConfigOptions as KarmaConfigOptions } from 'karma';\nimport * as webpack from 'webpack';\nimport * as webpackDevMiddleware from 'webpack-dev-middleware';\nimport * as yargs from 'yargs';\n\nimport { getTestConfigFromKarma } from '../helpers';\nimport { Logger } from '../utils';\n\nimport { getWebpackTestConfig } from '../webpack/configs';\nimport { FailureKarmaWebpackPlugin } from '../webpack/plugins/failure-karma-webpack-plugin';\nimport { TurnOffWatchWebpackPlugin } from '../webpack/plugins/turn-off-watch-webpack-plugin';\n\nexport interface PluginOptions extends KarmaConfigOptions {\n    configFile: string;\n    webpackConfig?: webpack.Configuration;\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport type NextHandleFunction = (req: any, res: http.ServerResponse, next: (err?: unknown) => void) => void;\n\nlet blocked: (() => void)[] = [];\nlet isBlocked = false;\nlet webpackMiddleware: webpackDevMiddleware.WebpackDevMiddleware & NextHandleFunction;\n\nfunction requestBlocker() {\n    return (_1: { url: string }, _2: http.ServerResponse, next: () => void) => {\n        if (isBlocked) {\n            blocked.push(next);\n        } else {\n            next();\n        }\n    };\n}\n\nfunction fallbackMiddleware() {\n    return (req: { url: string }, res: http.ServerResponse, next: () => void) => {\n        if (webpackMiddleware) {\n            const webpackUrl = `/_karma_webpack_${req.url}`;\n            const webpackReq = { ...req, url: webpackUrl };\n            webpackMiddleware(webpackReq, res, next);\n        } else {\n            next();\n        }\n    };\n}\n\nconst init = async (\n    config: PluginOptions,\n    emitter: {\n        on: (eventName: string, cb: (doneFn: () => void) => void) => void;\n        emit: (eventName: string, messages: string[], options?: { exitCode: number }) => void;\n        refreshFiles: () => void;\n    },\n    customFileHandlers: { urlRegex: RegExp; handler: NextHandleFunction }[]\n) => {\n    const logger = new Logger({ logLevel: 'info' });\n\n    config.customContextFile = `${__dirname}/karma-context.html`;\n    config.customDebugFile = `${__dirname}/karma-debug.html`;\n\n    config.beforeMiddleware = config.beforeMiddleware || [];\n    config.beforeMiddleware.push('lib-tools--blocker');\n    config.middleware = config.middleware || [];\n    config.middleware.push('lib-tools--fallback');\n\n    const webpackMiddlewareConfig: webpackDevMiddleware.Options = {\n        publicPath: '/_karma_webpack_/'\n    };\n\n    const compilationErrorCallback = (_: string | undefined, errors: string[]) => {\n        emitter.emit('compile_error', errors);\n        emitter.emit('run_complete', [], { exitCode: 1 });\n        unblock();\n    };\n\n    let webpackConfig: webpack.Configuration;\n\n    if (config.webpackConfig) {\n        webpackConfig = config.webpackConfig;\n    } else {\n        const args =\n            process.argv.length > 4 && /(\\\\|\\/)?karma(\\.js)?$/i.test(process.argv[1]) ? process.argv.slice(4) : [];\n\n        const commandOptions = yargs(args)\n            .option('browsers', {\n                type: 'string'\n            })\n            .option('reporters', {\n                type: 'string'\n            })\n            .option('codeCoverageExclude', {\n                type: 'string'\n            }).argv;\n\n        const testConfig = await getTestConfigFromKarma(config, commandOptions);\n        if (!testConfig) {\n            throw new Error('Could not load workflow test config.');\n        }\n\n        if (commandOptions.reporters == null && testConfig.reporters != null) {\n            config.reporters = testConfig.reporters;\n        }\n\n        if (commandOptions.browsers == null && testConfig.browsers != null) {\n            config.browsers = testConfig.browsers;\n        }\n\n        webpackConfig = await getWebpackTestConfig(testConfig, commandOptions);\n    }\n\n    webpackConfig.plugins = webpackConfig.plugins || [];\n    webpackConfig.plugins.push(new FailureKarmaWebpackPlugin(compilationErrorCallback));\n    webpackConfig.watch = !config.singleRun;\n    if (config.singleRun) {\n        webpackConfig.plugins.unshift(new TurnOffWatchWebpackPlugin());\n    }\n\n    function unblock(): void {\n        isBlocked = false;\n        blocked.forEach((cb) => cb());\n        blocked = [];\n    }\n\n    let compiler: webpack.Compiler;\n    try {\n        compiler = webpack(webpackConfig);\n    } catch (e) {\n        logger.error((e as Error).stack || e);\n        if ((e as { details: string }).details) {\n            logger.error((e as { details: string }).details);\n        }\n\n        throw e;\n    }\n\n    function handler(callback?: () => void) {\n        isBlocked = true;\n\n        if (typeof callback === 'function') {\n            callback();\n        }\n    }\n\n    compiler.hooks.invalid.tap('karma', () => handler());\n    compiler.hooks.watchRun.tapAsync('karma', (_, callback: () => void) => handler(callback));\n    compiler.hooks.run.tapAsync('karma', (_, callback: () => void) => handler(callback));\n\n    let lastCompilationHash: string | undefined;\n    compiler.hooks.done.tap('karma', (stats) => {\n        if (stats.compilation.errors.length > 0) {\n            logger.error(stats.toString('errors-only'));\n            lastCompilationHash = undefined;\n        } else if (stats.hash !== lastCompilationHash) {\n            lastCompilationHash = stats.hash;\n            emitter.refreshFiles();\n        }\n\n        unblock();\n    });\n\n    webpackMiddleware = webpackDevMiddleware(compiler, webpackMiddlewareConfig);\n\n    customFileHandlers.push({\n        urlRegex: /^\\/_karma_webpack_\\/.*/,\n        handler: (req: { url: string }, res: http.ServerResponse) => {\n            webpackMiddleware(req, res, function () {\n                const alwaysServe = [\n                    '/_karma_webpack_/runtime.js',\n                    '/_karma_webpack_/polyfills.js',\n                    '/_karma_webpack_/vendor.js'\n                ];\n                if (alwaysServe.includes(req.url)) {\n                    res.statusCode = 200;\n                    res.end();\n                } else {\n                    res.statusCode = 404;\n                    res.end('Not found');\n                }\n            });\n        }\n    });\n\n    emitter.on('exit', (done) => {\n        webpackMiddleware.close();\n        done();\n    });\n};\n\ninit.$inject = ['config', 'emitter', 'customFileHandlers'];\n\nmodule.exports = {\n    'framework:lib-tools': ['factory', init],\n    'middleware:lib-tools--blocker': ['factory', requestBlocker],\n    'middleware:lib-tools--fallback': ['factory', fallbackMiddleware]\n};\n"]}