ObjectExpressionKeysTransformer.ts 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  1. import { inject, injectable } from 'inversify';
  2. import { ServiceIdentifiers } from '../../container/ServiceIdentifiers';
  3. import * as estraverse from 'estraverse';
  4. import * as ESTree from 'estree';
  5. import { TObjectExpressionExtractorFactory } from '../../types/container/node-transformers/TObjectExpressionExtractorFactory';
  6. import { IOptions } from '../../interfaces/options/IOptions';
  7. import { IRandomGenerator } from '../../interfaces/utils/IRandomGenerator';
  8. import { IVisitor } from '../../interfaces/node-transformers/IVisitor';
  9. import { TransformationStage } from '../../enums/node-transformers/TransformationStage';
  10. import { AbstractNodeTransformer } from '../AbstractNodeTransformer';
  11. import { NodeGuards } from '../../node/NodeGuards';
  12. import { NodeStatementUtils } from '../../node/NodeStatementUtils';
  13. import { ObjectExpressionExtractor } from '../../enums/node-transformers/converting-transformers/properties-extractors/ObjectExpressionExtractor';
  14. @injectable()
  15. export class ObjectExpressionKeysTransformer extends AbstractNodeTransformer {
  16. /**
  17. * @type {ObjectExpressionExtractor[]}
  18. */
  19. private static readonly objectExpressionExtractorNames: ObjectExpressionExtractor[] = [
  20. ObjectExpressionExtractor.ObjectExpressionToVariableDeclarationExtractor,
  21. ObjectExpressionExtractor.BasePropertiesExtractor
  22. ];
  23. /**
  24. * @type {TObjectExpressionExtractorFactory}
  25. */
  26. private readonly objectExpressionExtractorFactory: TObjectExpressionExtractorFactory;
  27. /**
  28. * @param {TObjectExpressionExtractorFactory} objectExpressionExtractorFactory
  29. * @param {IRandomGenerator} randomGenerator
  30. * @param {IOptions} options
  31. */
  32. public constructor (
  33. @inject(ServiceIdentifiers.Factory__IObjectExpressionExtractor)
  34. objectExpressionExtractorFactory: TObjectExpressionExtractorFactory,
  35. @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,
  36. @inject(ServiceIdentifiers.IOptions) options: IOptions
  37. ) {
  38. super(randomGenerator, options);
  39. this.objectExpressionExtractorFactory = objectExpressionExtractorFactory;
  40. }
  41. /**
  42. * @param {ObjectExpression} objectExpressionNode
  43. * @param {Statement} hostStatement
  44. * @returns {boolean}
  45. */
  46. private static isProhibitedHostStatement (
  47. objectExpressionNode: ESTree.ObjectExpression,
  48. hostStatement: ESTree.Statement
  49. ): boolean {
  50. return ObjectExpressionKeysTransformer.isReferencedIdentifierName(
  51. objectExpressionNode,
  52. hostStatement
  53. )
  54. || ObjectExpressionKeysTransformer.isProhibitedSequenceExpression(
  55. objectExpressionNode,
  56. hostStatement
  57. );
  58. }
  59. /**
  60. * @param {ObjectExpression} objectExpressionNode
  61. * @param {Node} hostNode
  62. * @returns {boolean}
  63. */
  64. private static isReferencedIdentifierName (
  65. objectExpressionNode: ESTree.ObjectExpression,
  66. hostNode: ESTree.Node,
  67. ): boolean {
  68. const identifierNamesSet: string[] = [];
  69. let isReferencedIdentifierName: boolean = false;
  70. let isCurrentNode: boolean = false;
  71. // should mark node as prohibited if identifier of node is referenced somewhere inside other nodes
  72. estraverse.traverse(hostNode, {
  73. enter: (node: ESTree.Node): void | estraverse.VisitorOption => {
  74. if (node === objectExpressionNode) {
  75. isCurrentNode = true;
  76. }
  77. if (!NodeGuards.isIdentifierNode(node)) {
  78. return;
  79. }
  80. if (!isCurrentNode) {
  81. identifierNamesSet.push(node.name);
  82. return;
  83. }
  84. if (identifierNamesSet.includes(node.name)) {
  85. isReferencedIdentifierName = true;
  86. }
  87. },
  88. leave: (node: ESTree.Node): void | estraverse.VisitorOption => {
  89. if (node === objectExpressionNode) {
  90. isCurrentNode = false;
  91. return estraverse.VisitorOption.Break;
  92. }
  93. }
  94. });
  95. return isReferencedIdentifierName;
  96. }
  97. /**
  98. * @param {ObjectExpression} objectExpressionNode
  99. * @param {Node} hostNode
  100. * @returns {boolean}
  101. */
  102. private static isProhibitedSequenceExpression (
  103. objectExpressionNode: ESTree.ObjectExpression,
  104. hostNode: ESTree.Node,
  105. ): boolean {
  106. return NodeGuards.isExpressionStatementNode(hostNode)
  107. && NodeGuards.isSequenceExpressionNode(hostNode.expression)
  108. && hostNode.expression.expressions.some((expressionNode: ESTree.Expression) =>
  109. NodeGuards.isCallExpressionNode(expressionNode)
  110. && NodeGuards.isSuperNode(expressionNode.callee)
  111. );
  112. }
  113. /**
  114. * @param {TransformationStage} transformationStage
  115. * @returns {IVisitor | null}
  116. */
  117. public getVisitor (transformationStage: TransformationStage): IVisitor | null {
  118. if (!this.options.transformObjectKeys) {
  119. return null;
  120. }
  121. switch (transformationStage) {
  122. case TransformationStage.Converting:
  123. return {
  124. leave: (node: ESTree.Node, parentNode: ESTree.Node | null): ESTree.Node | undefined => {
  125. if (
  126. parentNode
  127. && NodeGuards.isObjectExpressionNode(node)
  128. ) {
  129. return this.transformNode(node, parentNode);
  130. }
  131. }
  132. };
  133. default:
  134. return null;
  135. }
  136. }
  137. /**
  138. * replaces:
  139. * var object = {
  140. * foo: 1,
  141. * bar: 2
  142. * };
  143. *
  144. * on:
  145. * var _0xabc123 = {};
  146. * _0xabc123['foo'] = 1;
  147. * _0xabc123['bar'] = 2;
  148. * var object = _0xabc123;
  149. *
  150. * @param {ObjectExpression} objectExpressionNode
  151. * @param {Node} parentNode
  152. * @returns {NodeGuards}
  153. */
  154. public transformNode (objectExpressionNode: ESTree.ObjectExpression, parentNode: ESTree.Node): ESTree.Node {
  155. if (!objectExpressionNode.properties.length) {
  156. return objectExpressionNode;
  157. }
  158. const hostStatement: ESTree.Statement = NodeStatementUtils.getRootStatementOfNode(objectExpressionNode);
  159. if (ObjectExpressionKeysTransformer.isProhibitedHostStatement(objectExpressionNode, hostStatement)) {
  160. return objectExpressionNode;
  161. }
  162. return this.applyObjectExpressionKeysExtractorsRecursive(
  163. ObjectExpressionKeysTransformer.objectExpressionExtractorNames,
  164. objectExpressionNode,
  165. hostStatement
  166. );
  167. }
  168. /**
  169. * @param {ObjectExpressionExtractor[]} objectExpressionExtractorNames
  170. * @param {ObjectExpression} objectExpressionNode
  171. * @param {Statement} hostStatement
  172. * @returns {Node}
  173. */
  174. private applyObjectExpressionKeysExtractorsRecursive (
  175. objectExpressionExtractorNames: ObjectExpressionExtractor[],
  176. objectExpressionNode: ESTree.ObjectExpression,
  177. hostStatement: ESTree.Statement
  178. ): ESTree.Node {
  179. const newObjectExpressionExtractorNames: ObjectExpressionExtractor[] = [...objectExpressionExtractorNames];
  180. const objectExpressionExtractor: ObjectExpressionExtractor | undefined =
  181. newObjectExpressionExtractorNames.shift();
  182. if (!objectExpressionExtractor) {
  183. return objectExpressionNode;
  184. }
  185. const {
  186. nodeToReplace,
  187. objectExpressionHostStatement: newObjectExpressionHostStatement,
  188. objectExpressionNode: newObjectExpressionNode
  189. } = this.objectExpressionExtractorFactory(objectExpressionExtractor)
  190. .extract(objectExpressionNode, hostStatement);
  191. this.applyObjectExpressionKeysExtractorsRecursive(
  192. newObjectExpressionExtractorNames,
  193. newObjectExpressionNode,
  194. newObjectExpressionHostStatement
  195. );
  196. return nodeToReplace;
  197. }
  198. }