MangledIdentifierNamesGenerator.ts 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. import { inject, injectable } from 'inversify';
  2. import { ServiceIdentifiers } from '../../container/ServiceIdentifiers';
  3. import { TNodeWithLexicalScope } from '../../types/node/TNodeWithLexicalScope';
  4. import { IOptions } from '../../interfaces/options/IOptions';
  5. import { IRandomGenerator } from '../../interfaces/utils/IRandomGenerator';
  6. import { AbstractIdentifierNamesGenerator } from './AbstractIdentifierNamesGenerator';
  7. @injectable()
  8. export class MangledIdentifierNamesGenerator extends AbstractIdentifierNamesGenerator {
  9. /**
  10. * @type {string}
  11. */
  12. private static readonly initMangledNameCharacter: string = '9';
  13. /**
  14. * @type {string[]}
  15. */
  16. private static readonly nameSequence: string[] = '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'.split('');
  17. /**
  18. * Reserved JS words with length of 2-4 symbols that can be possible generated with this replacer
  19. *
  20. * @type {Set<string>}
  21. */
  22. private static readonly reservedNamesSet: Set<string> = new Set([
  23. 'byte', 'case', 'char', 'do', 'else', 'enum', 'eval', 'for', 'goto',
  24. 'if', 'in', 'int', 'let', 'long', 'new', 'null', 'this', 'true', 'try',
  25. 'var', 'void', 'with'
  26. ]);
  27. /**
  28. * @type {string}
  29. */
  30. private previousMangledName: string = MangledIdentifierNamesGenerator.initMangledNameCharacter;
  31. /**
  32. * @param {IRandomGenerator} randomGenerator
  33. * @param {IOptions} options
  34. */
  35. public constructor (
  36. @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,
  37. @inject(ServiceIdentifiers.IOptions) options: IOptions
  38. ) {
  39. super(randomGenerator, options);
  40. }
  41. /**
  42. * We can only ignore limited nameLength, it has no sense here
  43. * @param {number} nameLength
  44. * @returns {string}
  45. */
  46. public generate (nameLength?: number): string {
  47. const identifierName: string = this.generateNewMangledName(this.previousMangledName);
  48. this.previousMangledName = identifierName;
  49. return identifierName;
  50. }
  51. /**
  52. * @param {TNodeWithLexicalScope} lexicalScopeNode
  53. * @param {number} nameLength
  54. * @returns {string}
  55. */
  56. public generateForLexicalScope (lexicalScopeNode: TNodeWithLexicalScope, nameLength?: number): string {
  57. const identifierName: string = this.generateNewMangledName(this.previousMangledName);
  58. if (!this.isValidIdentifierNameInLexicalScope(identifierName, lexicalScopeNode)) {
  59. return this.generateForLexicalScope(lexicalScopeNode, nameLength);
  60. }
  61. this.previousMangledName = identifierName;
  62. return identifierName;
  63. }
  64. /**
  65. * @param {number} nameLength
  66. * @returns {string}
  67. */
  68. public generateWithPrefix (nameLength?: number): string {
  69. const prefix: string = this.options.identifiersPrefix ?
  70. `${this.options.identifiersPrefix}`
  71. : '';
  72. const identifierName: string = this.generate(nameLength);
  73. const identifierNameWithPrefix: string = `${prefix}${identifierName}`;
  74. this.preserveName(identifierNameWithPrefix);
  75. return identifierNameWithPrefix;
  76. }
  77. /**
  78. * @param {string} mangledName
  79. * @returns {boolean}
  80. */
  81. public isValidIdentifierName (mangledName: string): boolean {
  82. return super.isValidIdentifierName(mangledName)
  83. && !MangledIdentifierNamesGenerator.reservedNamesSet.has(mangledName);
  84. }
  85. /**
  86. * @param {string} previousMangledName
  87. * @returns {string}
  88. */
  89. private generateNewMangledName (previousMangledName: string): string {
  90. const generateNewMangledName: (name: string) => string = (name: string): string => {
  91. const nameSequence: string[] = MangledIdentifierNamesGenerator.nameSequence;
  92. const nameSequenceLength: number = nameSequence.length;
  93. const nameLength: number = name.length;
  94. const zeroSequence: (num: number) => string = (num: number): string => {
  95. return '0'.repeat(num);
  96. };
  97. let index: number = nameLength - 1;
  98. do {
  99. const character: string = name[index];
  100. const indexInSequence: number = nameSequence.indexOf(character);
  101. const lastNameSequenceIndex: number = nameSequenceLength - 1;
  102. if (indexInSequence !== lastNameSequenceIndex) {
  103. const previousNamePart: string = name.substring(0, index);
  104. const nextCharacter: string = nameSequence[indexInSequence + 1];
  105. const zeroSequenceLength: number = nameLength - (index + 1);
  106. const zeroSequenceCharacters: string = zeroSequence(zeroSequenceLength);
  107. return previousNamePart + nextCharacter + zeroSequenceCharacters;
  108. }
  109. --index;
  110. } while (index >= 0);
  111. return `a${zeroSequence(nameLength)}`;
  112. };
  113. let newMangledName: string = generateNewMangledName(previousMangledName);
  114. if (!this.isValidIdentifierName(newMangledName)) {
  115. newMangledName = this.generateNewMangledName(newMangledName);
  116. }
  117. return newMangledName;
  118. }
  119. }