javascript-obfuscator
Version:
JavaScript obfuscator
174 lines (143 loc) • 5.7 kB
text/typescript
import { inject, injectable, } from 'inversify';
import { ServiceIdentifiers } from '../../container/ServiceIdentifiers';
import * as ESTree from 'estree';
import * as estraverse from 'estraverse';
import { IOptions } from '../../interfaces/options/IOptions';
import { IRandomGenerator } from '../../interfaces/utils/IRandomGenerator';
import { IVisitor } from '../../interfaces/node-transformers/IVisitor';
import { NodeTransformationStage } from '../../enums/node-transformers/NodeTransformationStage';
import { AbstractNodeTransformer } from '../AbstractNodeTransformer';
import { ConditionalCommentObfuscatingGuard } from '../preparing-transformers/obfuscating-guards/ConditionalCommentObfuscatingGuard';
import { NodeGuards } from '../../node/NodeGuards';
()
export class CommentsTransformer extends AbstractNodeTransformer {
/**
* @type {string[]}
*/
private static readonly preservedWords: string[] = [
'@license',
'@preserve'
];
/**
* @param {IRandomGenerator} randomGenerator
* @param {IOptions} options
*/
public constructor (
(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,
(ServiceIdentifiers.IOptions) options: IOptions
) {
super(randomGenerator, options);
}
/**
* @param {NodeTransformationStage} nodeTransformationStage
* @returns {IVisitor | null}
*/
public getVisitor (nodeTransformationStage: NodeTransformationStage): IVisitor | null {
switch (nodeTransformationStage) {
case NodeTransformationStage.Initializing:
return {
leave: (node: ESTree.Node): ESTree.Node | undefined => {
if (NodeGuards.isProgramNode(node)) {
return this.transformNode(node);
}
}
};
case NodeTransformationStage.Finalizing:
return {
leave: (node: ESTree.Node): ESTree.Node | undefined => {
if (NodeGuards.isProgramNode(node)) {
return this.filterCommentsOnFinalizingTraverse(node);
}
}
};
default:
return null;
}
}
/**
* Moves comments to their nodes
*/
public transformNode (rootNode: ESTree.Program): ESTree.Node {
rootNode = this.filterCommentsOnPrimaryTraverse(rootNode);
if (!rootNode.comments || !rootNode.comments.length) {
return rootNode;
}
const comments: ESTree.Comment[] = rootNode.comments.reverse();
if (comments.length === 0) {
return rootNode;
}
if (!rootNode.body.length) {
rootNode.leadingComments = comments;
return rootNode;
}
let isFirstNode: boolean = true;
estraverse.traverse(rootNode, {
enter: (node: ESTree.Node): void => {
if (node === rootNode) {
return;
}
const commentIdx: number = comments.findIndex((comment: ESTree.Comment) =>
comment.range && node.range && comment.range[0] < node.range[0]
);
if (commentIdx >= 0) {
(isFirstNode ? rootNode : node).leadingComments =
comments.splice(commentIdx, comments.length - commentIdx).reverse();
}
isFirstNode = false;
}
});
if (comments.length > 0) {
rootNode.trailingComments = comments.reverse();
}
return rootNode;
}
/**
* Removes all comments from node except comments that contain
* preserved words or `javascript-obfuscator` words
*
* @param {ESTree.Program} rootNode
* @returns {ESTree.Program}
*/
private filterCommentsOnPrimaryTraverse (rootNode: ESTree.Program): ESTree.Program {
rootNode.comments = rootNode.comments?.filter((comment: ESTree.Comment) =>
this.filterComment(comment, true)
);
return rootNode;
}
/**
* Removes all comments from node except comments that contain preserved words
*
* @param {ESTree.Program} rootNode
* @returns {ESTree.Program}
*/
private filterCommentsOnFinalizingTraverse (rootNode: ESTree.Program): ESTree.Program {
estraverse.replace(rootNode, {
enter: (node: ESTree.Node): ESTree.Node => {
if (node.leadingComments) {
node.leadingComments = node.leadingComments?.filter((comment: ESTree.Comment) =>
this.filterComment(comment, false)
);
}
if (node.trailingComments) {
node.trailingComments = node.trailingComments?.filter((comment: ESTree.Comment) =>
this.filterComment(comment, false)
);
}
return node;
}
});
return rootNode;
}
/**
* @param {ESTree.Comment} comment
* @param {boolean} keepConditionalComment
* @returns {boolean}
*/
private filterComment (comment: ESTree.Comment, keepConditionalComment: boolean): boolean {
if (keepConditionalComment && ConditionalCommentObfuscatingGuard.isConditionalComment(comment)) {
return true;
}
return CommentsTransformer.preservedWords
.some((preservedWord: string) => comment.value.includes(preservedWord));
}
}