184 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			184 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
/*
 | 
						|
	MIT License http://www.opensource.org/licenses/mit-license.php
 | 
						|
	Author Tobias Koppers @sokra
 | 
						|
*/
 | 
						|
"use strict";
 | 
						|
 | 
						|
const util = require("util");
 | 
						|
 | 
						|
const deprecateContext = util.deprecate(
 | 
						|
	() => {},
 | 
						|
	"Hook.context is deprecated and will be removed"
 | 
						|
);
 | 
						|
 | 
						|
function CALL_DELEGATE(...args) {
 | 
						|
	this.call = this._createCall("sync");
 | 
						|
	return this.call(...args);
 | 
						|
}
 | 
						|
 | 
						|
function CALL_ASYNC_DELEGATE(...args) {
 | 
						|
	this.callAsync = this._createCall("async");
 | 
						|
	return this.callAsync(...args);
 | 
						|
}
 | 
						|
 | 
						|
function PROMISE_DELEGATE(...args) {
 | 
						|
	this.promise = this._createCall("promise");
 | 
						|
	return this.promise(...args);
 | 
						|
}
 | 
						|
 | 
						|
class Hook {
 | 
						|
	constructor(args = [], name = undefined) {
 | 
						|
		this._args = args;
 | 
						|
		this.name = name;
 | 
						|
		this.taps = [];
 | 
						|
		this.interceptors = [];
 | 
						|
		this._call = CALL_DELEGATE;
 | 
						|
		this.call = CALL_DELEGATE;
 | 
						|
		this._callAsync = CALL_ASYNC_DELEGATE;
 | 
						|
		this.callAsync = CALL_ASYNC_DELEGATE;
 | 
						|
		this._promise = PROMISE_DELEGATE;
 | 
						|
		this.promise = PROMISE_DELEGATE;
 | 
						|
		this._x = undefined;
 | 
						|
 | 
						|
		// eslint-disable-next-line no-self-assign
 | 
						|
		this.compile = this.compile;
 | 
						|
		// eslint-disable-next-line no-self-assign
 | 
						|
		this.tap = this.tap;
 | 
						|
		// eslint-disable-next-line no-self-assign
 | 
						|
		this.tapAsync = this.tapAsync;
 | 
						|
		// eslint-disable-next-line no-self-assign
 | 
						|
		this.tapPromise = this.tapPromise;
 | 
						|
	}
 | 
						|
 | 
						|
	compile(_options) {
 | 
						|
		throw new Error("Abstract: should be overridden");
 | 
						|
	}
 | 
						|
 | 
						|
	_createCall(type) {
 | 
						|
		return this.compile({
 | 
						|
			taps: this.taps,
 | 
						|
			interceptors: this.interceptors,
 | 
						|
			args: this._args,
 | 
						|
			type
 | 
						|
		});
 | 
						|
	}
 | 
						|
 | 
						|
	_tap(type, options, fn) {
 | 
						|
		if (typeof options === "string") {
 | 
						|
			options = {
 | 
						|
				name: options.trim()
 | 
						|
			};
 | 
						|
		} else if (typeof options !== "object" || options === null) {
 | 
						|
			throw new Error("Invalid tap options");
 | 
						|
		}
 | 
						|
		if (typeof options.name !== "string" || options.name === "") {
 | 
						|
			throw new Error("Missing name for tap");
 | 
						|
		}
 | 
						|
		if (typeof options.context !== "undefined") {
 | 
						|
			deprecateContext();
 | 
						|
		}
 | 
						|
		options = Object.assign({ type, fn }, options);
 | 
						|
		options = this._runRegisterInterceptors(options);
 | 
						|
		this._insert(options);
 | 
						|
	}
 | 
						|
 | 
						|
	tap(options, fn) {
 | 
						|
		this._tap("sync", options, fn);
 | 
						|
	}
 | 
						|
 | 
						|
	tapAsync(options, fn) {
 | 
						|
		this._tap("async", options, fn);
 | 
						|
	}
 | 
						|
 | 
						|
	tapPromise(options, fn) {
 | 
						|
		this._tap("promise", options, fn);
 | 
						|
	}
 | 
						|
 | 
						|
	_runRegisterInterceptors(options) {
 | 
						|
		for (const interceptor of this.interceptors) {
 | 
						|
			if (interceptor.register) {
 | 
						|
				const newOptions = interceptor.register(options);
 | 
						|
				if (newOptions !== undefined) {
 | 
						|
					options = newOptions;
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		return options;
 | 
						|
	}
 | 
						|
 | 
						|
	withOptions(options) {
 | 
						|
		const mergeOptions = (opt) =>
 | 
						|
			Object.assign({}, options, typeof opt === "string" ? { name: opt } : opt);
 | 
						|
 | 
						|
		return {
 | 
						|
			name: this.name,
 | 
						|
			tap: (opt, fn) => this.tap(mergeOptions(opt), fn),
 | 
						|
			tapAsync: (opt, fn) => this.tapAsync(mergeOptions(opt), fn),
 | 
						|
			tapPromise: (opt, fn) => this.tapPromise(mergeOptions(opt), fn),
 | 
						|
			intercept: (interceptor) => this.intercept(interceptor),
 | 
						|
			isUsed: () => this.isUsed(),
 | 
						|
			withOptions: (opt) => this.withOptions(mergeOptions(opt))
 | 
						|
		};
 | 
						|
	}
 | 
						|
 | 
						|
	isUsed() {
 | 
						|
		return this.taps.length > 0 || this.interceptors.length > 0;
 | 
						|
	}
 | 
						|
 | 
						|
	intercept(interceptor) {
 | 
						|
		this._resetCompilation();
 | 
						|
		this.interceptors.push(Object.assign({}, interceptor));
 | 
						|
		if (interceptor.register) {
 | 
						|
			for (let i = 0; i < this.taps.length; i++) {
 | 
						|
				this.taps[i] = interceptor.register(this.taps[i]);
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	_resetCompilation() {
 | 
						|
		this.call = this._call;
 | 
						|
		this.callAsync = this._callAsync;
 | 
						|
		this.promise = this._promise;
 | 
						|
	}
 | 
						|
 | 
						|
	_insert(item) {
 | 
						|
		this._resetCompilation();
 | 
						|
		let before;
 | 
						|
		if (typeof item.before === "string") {
 | 
						|
			before = new Set([item.before]);
 | 
						|
		} else if (Array.isArray(item.before)) {
 | 
						|
			before = new Set(item.before);
 | 
						|
		}
 | 
						|
		let stage = 0;
 | 
						|
		if (typeof item.stage === "number") {
 | 
						|
			stage = item.stage;
 | 
						|
		}
 | 
						|
		let i = this.taps.length;
 | 
						|
		while (i > 0) {
 | 
						|
			i--;
 | 
						|
			const tap = this.taps[i];
 | 
						|
			this.taps[i + 1] = tap;
 | 
						|
			const xStage = tap.stage || 0;
 | 
						|
			if (before) {
 | 
						|
				if (before.has(tap.name)) {
 | 
						|
					before.delete(tap.name);
 | 
						|
					continue;
 | 
						|
				}
 | 
						|
				if (before.size > 0) {
 | 
						|
					continue;
 | 
						|
				}
 | 
						|
			}
 | 
						|
			if (xStage > stage) {
 | 
						|
				continue;
 | 
						|
			}
 | 
						|
			i++;
 | 
						|
			break;
 | 
						|
		}
 | 
						|
		this.taps[i] = item;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
Object.setPrototypeOf(Hook.prototype, null);
 | 
						|
 | 
						|
module.exports = Hook;
 |