@adguard/agtree
Version:
Tool set for working with adblock filter lists
137 lines (134 loc) • 5.58 kB
JavaScript
/*
* AGTree v3.2.2 (build date: Tue, 08 Jul 2025 13:39:47 GMT)
* (c) 2025 Adguard Software Ltd.
* Released under the MIT license
* https://github.com/AdguardTeam/tsurlfilter/tree/master/packages/agtree#readme
*/
import { NULL } from '../../utils/constants.js';
import { isUndefined } from '../../utils/type-guards.js';
import { BaseSerializer } from '../base-serializer.js';
import { KNOWN_VARIABLES_SERIALIZATION_MAP, VariableNodeBinaryPropMarshallingMap, ParenthesisNodeBinaryPropMarshallingMap, OperatorNodeBinaryPropMarshallingMap, LOGICAL_EXPRESSION_OPERATOR_SERIALISATION_MAP } from '../../marshalling-utils/misc/logical-expression-common.js';
import { BinaryTypeMarshallingMap } from '../../marshalling-utils/misc/binary-type-common.js';
/**
* Possible node types in the logical expression.
*/
const NodeType = {
Variable: 'Variable',
Operator: 'Operator',
Parenthesis: 'Parenthesis',
};
/**
* `LogicalExpressionSerializer` is responsible for serializing logical expressions.
*
* @example
* From the following rule:
* ```adblock
* !#if (adguard_ext_android_cb || adguard_ext_safari)
* ```
* this parser will parse the expression `(adguard_ext_android_cb || adguard_ext_safari)`.
*/
// TODO: Refactor this class
class LogicalExpressionSerializer extends BaseSerializer {
/**
* Serializes a variable node to binary format.
*
* @param node Node to serialize.
* @param buffer ByteBuffer for writing binary data.
*/
// TODO: create a common serialize / deserialize interface for such nodes (Variable, Value, Parameter, etc.)
static serializeVariableNode(node, buffer) {
buffer.writeUint8(BinaryTypeMarshallingMap.ExpressionVariableNode);
const frequentName = KNOWN_VARIABLES_SERIALIZATION_MAP.get(node.name);
if (!isUndefined(frequentName)) {
buffer.writeUint8(VariableNodeBinaryPropMarshallingMap.FrequentName);
buffer.writeUint8(frequentName);
}
else {
buffer.writeUint8(VariableNodeBinaryPropMarshallingMap.Name);
buffer.writeString(node.name);
}
if (!isUndefined(node.start)) {
buffer.writeUint8(VariableNodeBinaryPropMarshallingMap.Start);
buffer.writeUint32(node.start);
}
if (!isUndefined(node.end)) {
buffer.writeUint8(VariableNodeBinaryPropMarshallingMap.End);
buffer.writeUint32(node.end);
}
buffer.writeUint8(NULL);
}
/**
* Serializes a parenthesis node to binary format.
*
* @param node Node to serialize.
* @param buffer ByteBuffer for writing binary data.
*/
static serializeParenthesisNode(node, buffer) {
buffer.writeUint8(BinaryTypeMarshallingMap.ExpressionParenthesisNode);
buffer.writeUint8(ParenthesisNodeBinaryPropMarshallingMap.Expression);
LogicalExpressionSerializer.serialize(node.expression, buffer);
if (!isUndefined(node.start)) {
buffer.writeUint8(ParenthesisNodeBinaryPropMarshallingMap.Start);
buffer.writeUint32(node.start);
}
if (!isUndefined(node.end)) {
buffer.writeUint8(ParenthesisNodeBinaryPropMarshallingMap.End);
buffer.writeUint32(node.end);
}
buffer.writeUint8(NULL);
}
/**
* Serializes an operator node to binary format.
*
* @param node Node to serialize.
* @param buffer ByteBuffer for writing binary data.
*/
static serializeOperatorNode(node, buffer) {
buffer.writeUint8(BinaryTypeMarshallingMap.ExpressionOperatorNode);
buffer.writeUint8(OperatorNodeBinaryPropMarshallingMap.Operator);
const operatorBinary = LOGICAL_EXPRESSION_OPERATOR_SERIALISATION_MAP.get(node.operator);
if (isUndefined(operatorBinary)) {
throw new Error(`Unknown operator: ${node.operator}`);
}
buffer.writeUint8(operatorBinary);
buffer.writeUint8(OperatorNodeBinaryPropMarshallingMap.Left);
LogicalExpressionSerializer.serialize(node.left, buffer);
if (node.right) {
buffer.writeUint8(OperatorNodeBinaryPropMarshallingMap.Right);
LogicalExpressionSerializer.serialize(node.right, buffer);
}
if (!isUndefined(node.start)) {
buffer.writeUint8(OperatorNodeBinaryPropMarshallingMap.Start);
buffer.writeUint32(node.start);
}
if (!isUndefined(node.end)) {
buffer.writeUint8(OperatorNodeBinaryPropMarshallingMap.End);
buffer.writeUint32(node.end);
}
buffer.writeUint8(NULL);
}
/**
* Serializes a logical expression node to binary format.
*
* @param node Node to serialize.
* @param buffer ByteBuffer for writing binary data.
*/
static serialize(node, buffer) {
switch (node.type) {
case NodeType.Variable:
LogicalExpressionSerializer.serializeVariableNode(node, buffer);
break;
case NodeType.Operator:
LogicalExpressionSerializer.serializeOperatorNode(node, buffer);
break;
case NodeType.Parenthesis:
LogicalExpressionSerializer.serializeParenthesisNode(node, buffer);
break;
default:
// eslint-disable-next-line @typescript-eslint/no-explicit-any
throw new Error(`Unexpected node type: ${node.type}`);
}
buffer.writeUint8(NULL);
}
}
export { LogicalExpressionSerializer, NodeType };