1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
|
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
import { XPCOMUtils } from "resource://gre/modules/XPCOMUtils.sys.mjs";
const lazy = {};
ChromeUtils.defineESModuleGetters(lazy, {
PreferenceFilters:
"resource://gre/modules/components-utils/PreferenceFilters.sys.mjs",
Sampling: "resource://gre/modules/components-utils/Sampling.sys.mjs",
});
ChromeUtils.defineModuleGetter(
lazy,
"mozjexl",
"resource://gre/modules/components-utils/mozjexl.js"
);
XPCOMUtils.defineLazyGetter(lazy, "jexl", () => {
const jexl = new lazy.mozjexl.Jexl();
jexl.addTransforms({
date: dateString => new Date(dateString),
stableSample: lazy.Sampling.stableSample,
bucketSample: lazy.Sampling.bucketSample,
preferenceValue: lazy.PreferenceFilters.preferenceValue,
preferenceIsUserSet: lazy.PreferenceFilters.preferenceIsUserSet,
preferenceExists: lazy.PreferenceFilters.preferenceExists,
keys,
length,
mapToProperty,
regExpMatch,
versionCompare,
});
jexl.addBinaryOp("intersect", 40, operatorIntersect);
return jexl;
});
export var FilterExpressions = {
getAvailableTransforms() {
return Object.keys(lazy.jexl._transforms);
},
eval(expr, context = {}) {
const onelineExpr = expr.replace(/[\t\n\r]/g, " ");
return lazy.jexl.eval(onelineExpr, context);
},
};
/**
* Return an array of the given object's own keys (specifically, its enumerable
* properties), or undefined if the argument isn't an object.
* @param {Object} obj
* @return {Array[String]|undefined}
*/
function keys(obj) {
if (typeof obj !== "object" || obj === null) {
return undefined;
}
return Object.keys(obj);
}
/**
* Return the length of an array
* @param {Array} arr
* @return {number}
*/
function length(arr) {
return Array.isArray(arr) ? arr.length : undefined;
}
/**
* Given an input array and property name, return an array with each element of
* the original array replaced with the given property of that element.
* @param {Array} arr
* @param {string} prop
* @return {Array}
*/
function mapToProperty(arr, prop) {
return Array.isArray(arr) ? arr.map(elem => elem[prop]) : undefined;
}
/**
* Find all the values that are present in both lists. Returns undefined if
* the arguments are not both Arrays.
* @param {Array} listA
* @param {Array} listB
* @return {Array|undefined}
*/
function operatorIntersect(listA, listB) {
if (!Array.isArray(listA) || !Array.isArray(listB)) {
return undefined;
}
return listA.filter(item => listB.includes(item));
}
/**
* Matches a string against a regular expression. Returns null if there are
* no matches or an Array of matches.
* @param {string} str
* @param {string} pattern
* @param {string} flags
* @return {Array|null}
*/
function regExpMatch(str, pattern, flags) {
const re = new RegExp(pattern, flags);
return str.match(re);
}
/**
* Compares v1 to v2 and returns 0 if they are equal, a negative number if
* v1 < v2 or a positive number if v1 > v2.
* @param {string} v1
* @param {string} v2
* @return {number}
*/
function versionCompare(v1, v2) {
return Services.vc.compare(v1, v2);
}
|