123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126 |
- import { inject, injectable, } from 'inversify';
- import { ServiceIdentifiers } from '../../../container/ServiceIdentifiers';
- import * as ESTree from 'estree';
- import { TControlFlowCustomNodeFactory } from '../../../types/container/custom-nodes/TControlFlowCustomNodeFactory';
- import { TIdentifierNamesGeneratorFactory } from '../../../types/container/generators/TIdentifierNamesGeneratorFactory';
- import { IControlFlowReplacer } from '../../../interfaces/node-transformers/control-flow-transformers/IControlFlowReplacer';
- import { IControlFlowStorage } from '../../../interfaces/storages/control-flow-transformers/IControlFlowStorage';
- import { ICustomNode } from '../../../interfaces/custom-nodes/ICustomNode';
- import { IIdentifierNamesGenerator } from '../../../interfaces/generators/identifier-names-generators/IIdentifierNamesGenerator';
- import { IOptions } from '../../../interfaces/options/IOptions';
- import { IRandomGenerator } from '../../../interfaces/utils/IRandomGenerator';
- @injectable()
- export abstract class AbstractControlFlowReplacer implements IControlFlowReplacer {
- /**
- * @type {TControlFlowCustomNodeFactory}
- */
- protected readonly controlFlowCustomNodeFactory: TControlFlowCustomNodeFactory;
- /**
- * @type {IIdentifierNamesGenerator}
- */
- protected readonly identifierNamesGenerator: IIdentifierNamesGenerator;
- /**
- * @type {IOptions}
- */
- protected readonly options: IOptions;
- /**
- * @type {IRandomGenerator}
- */
- protected readonly randomGenerator: IRandomGenerator;
- /**
- * @type {Map<string, Map<string, string[]>>}
- */
- protected readonly replacerDataByControlFlowStorageId: Map <string, Map<string, string[]>> = new Map();
- /**
- * @param {TControlFlowCustomNodeFactory} controlFlowCustomNodeFactory
- * @param {TIdentifierNamesGeneratorFactory} identifierNamesGeneratorFactory
- * @param {IRandomGenerator} randomGenerator
- * @param {IOptions} options
- */
- public constructor (
- @inject(ServiceIdentifiers.Factory__IControlFlowCustomNode)
- controlFlowCustomNodeFactory: TControlFlowCustomNodeFactory,
- @inject(ServiceIdentifiers.Factory__IIdentifierNamesGenerator)
- identifierNamesGeneratorFactory: TIdentifierNamesGeneratorFactory,
- @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,
- @inject(ServiceIdentifiers.IOptions) options: IOptions
- ) {
- this.controlFlowCustomNodeFactory = controlFlowCustomNodeFactory;
- this.identifierNamesGenerator = identifierNamesGeneratorFactory(options);
- this.randomGenerator = randomGenerator;
- this.options = options;
- }
- /**
- * Generates storage key with length of 5 characters to prevent collisions and to guarantee that
- * these keys will be added to the string array storage
- *
- * @param {IControlFlowStorage} controlFlowStorage
- * @returns {string}
- */
- public generateStorageKey (controlFlowStorage: IControlFlowStorage): string {
- const key: string = this.randomGenerator.getRandomString(5);
- if (controlFlowStorage.has(key)) {
- return this.generateStorageKey(controlFlowStorage);
- }
- return key;
- }
- /**
- * @param {ICustomNode} customNode
- * @param {IControlFlowStorage} controlFlowStorage
- * @param {string} replacerId
- * @param {number} usingExistingIdentifierChance
- * @returns {string}
- */
- protected insertCustomNodeToControlFlowStorage (
- customNode: ICustomNode,
- controlFlowStorage: IControlFlowStorage,
- replacerId: string,
- usingExistingIdentifierChance: number
- ): string {
- const controlFlowStorageId: string = controlFlowStorage.getStorageId();
- const storageKeysById: Map<string, string[]> = this.replacerDataByControlFlowStorageId.get(controlFlowStorageId)
- ?? new Map <string, string[]>();
- const storageKeysForCurrentId: string[] | null = storageKeysById.get(replacerId) ?? null;
- const shouldPickFromStorageKeysById = this.randomGenerator.getMathRandom() < usingExistingIdentifierChance
- && storageKeysForCurrentId?.length;
- if (shouldPickFromStorageKeysById) {
- return this.randomGenerator.getRandomGenerator().pickone(storageKeysForCurrentId);
- }
- const storageKey: string = this.generateStorageKey(controlFlowStorage);
- storageKeysById.set(replacerId, [storageKey]);
- this.replacerDataByControlFlowStorageId.set(controlFlowStorageId, storageKeysById);
- controlFlowStorage.set(storageKey, customNode);
- return storageKey;
- }
- /**
- * @param {Node} node
- * @param {Node} parentNode
- * @param {TNodeWithLexicalScope} controlFlowStorageLexicalScopeNode
- * @param {IControlFlowStorage} controlFlowStorage
- * @returns {Node}
- */
- public abstract replace (
- node: ESTree.Node,
- parentNode: ESTree.Node,
- controlFlowStorage: IControlFlowStorage
- ): ESTree.Node;
- }
|