yomichan/ext/js/background/profile-conditions-util.js

323 lines
9.9 KiB
JavaScript
Raw Permalink Normal View History

/*
* Copyright (C) 2020-2022 Yomichan Authors
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
/* global
* JsonSchema
*/
/**
* Utility class to help processing profile conditions.
*/
class ProfileConditionsUtil {
/**
* A group of conditions.
* @typedef {object} ProfileConditionGroup
* @property {ProfileCondition[]} conditions The list of conditions for this group.
*/
/**
* A single condition.
* @typedef {object} ProfileCondition
* @property {string} type The type of the condition.
* @property {string} operator The condition operator.
* @property {string} value The value to compare against.
*/
/**
* Creates a new instance.
*/
constructor() {
this._splitPattern = /[,;\s]+/;
this._descriptors = new Map([
[
'popupLevel',
{
operators: new Map([
['equal', this._createSchemaPopupLevelEqual.bind(this)],
['notEqual', this._createSchemaPopupLevelNotEqual.bind(this)],
['lessThan', this._createSchemaPopupLevelLessThan.bind(this)],
['greaterThan', this._createSchemaPopupLevelGreaterThan.bind(this)],
['lessThanOrEqual', this._createSchemaPopupLevelLessThanOrEqual.bind(this)],
['greaterThanOrEqual', this._createSchemaPopupLevelGreaterThanOrEqual.bind(this)]
])
}
],
[
'url',
{
operators: new Map([
['matchDomain', this._createSchemaUrlMatchDomain.bind(this)],
['matchRegExp', this._createSchemaUrlMatchRegExp.bind(this)]
])
}
],
[
'modifierKeys',
{
operators: new Map([
['are', this._createSchemaModifierKeysAre.bind(this)],
['areNot', this._createSchemaModifierKeysAreNot.bind(this)],
['include', this._createSchemaModifierKeysInclude.bind(this)],
['notInclude', this._createSchemaModifierKeysNotInclude.bind(this)]
])
}
],
[
'flags',
{
operators: new Map([
['are', this._createSchemaFlagsAre.bind(this)],
['areNot', this._createSchemaFlagsAreNot.bind(this)],
['include', this._createSchemaFlagsInclude.bind(this)],
['notInclude', this._createSchemaFlagsNotInclude.bind(this)]
])
}
]
]);
}
/**
* Creates a new JSON schema descriptor for the given set of condition groups.
* @param {ProfileConditionGroup[]} conditionGroups An array of condition groups.
* For a profile match, all of the items must return successfully in at least one of the groups.
* @returns {JsonSchema} A new `JsonSchema` object.
*/
createSchema(conditionGroups) {
const anyOf = [];
for (const {conditions} of conditionGroups) {
const allOf = [];
for (const {type, operator, value} of conditions) {
const conditionDescriptor = this._descriptors.get(type);
if (typeof conditionDescriptor === 'undefined') { continue; }
const createSchema = conditionDescriptor.operators.get(operator);
if (typeof createSchema === 'undefined') { continue; }
const schema = createSchema(value);
allOf.push(schema);
}
switch (allOf.length) {
case 0: break;
case 1: anyOf.push(allOf[0]); break;
default: anyOf.push({allOf}); break;
}
}
let schema;
switch (anyOf.length) {
case 0: schema = {}; break;
case 1: schema = anyOf[0]; break;
default: schema = {anyOf}; break;
}
return new JsonSchema(schema);
}
/**
* Creates a normalized version of the context object to test,
* assigning dependent fields as needed.
* @param {object} context A context object which is used during schema validation.
* @returns {object} A normalized context object.
*/
normalizeContext(context) {
const normalizedContext = Object.assign({}, context);
const {url} = normalizedContext;
if (typeof url === 'string') {
try {
normalizedContext.domain = new URL(url).hostname;
} catch (e) {
// NOP
}
}
const {flags} = normalizedContext;
if (!Array.isArray(flags)) {
normalizedContext.flags = [];
}
return normalizedContext;
}
// Private
_split(value) {
return value.split(this._splitPattern);
}
_stringToNumber(value) {
const number = Number.parseFloat(value);
return Number.isFinite(number) ? number : 0;
}
// popupLevel schema creation functions
_createSchemaPopupLevelEqual(value) {
value = this._stringToNumber(value);
return {
required: ['depth'],
properties: {
depth: {const: value}
}
};
}
_createSchemaPopupLevelNotEqual(value) {
return {
not: [this._createSchemaPopupLevelEqual(value)]
};
}
_createSchemaPopupLevelLessThan(value) {
value = this._stringToNumber(value);
return {
required: ['depth'],
properties: {
depth: {type: 'number', exclusiveMaximum: value}
}
};
}
_createSchemaPopupLevelGreaterThan(value) {
value = this._stringToNumber(value);
return {
required: ['depth'],
properties: {
depth: {type: 'number', exclusiveMinimum: value}
}
};
}
_createSchemaPopupLevelLessThanOrEqual(value) {
value = this._stringToNumber(value);
return {
required: ['depth'],
properties: {
depth: {type: 'number', maximum: value}
}
};
}
_createSchemaPopupLevelGreaterThanOrEqual(value) {
value = this._stringToNumber(value);
return {
required: ['depth'],
properties: {
depth: {type: 'number', minimum: value}
}
};
}
// url schema creation functions
_createSchemaUrlMatchDomain(value) {
const oneOf = [];
for (let domain of this._split(value)) {
if (domain.length === 0) { continue; }
domain = domain.toLowerCase();
oneOf.push({const: domain});
}
return {
required: ['domain'],
properties: {
domain: {oneOf}
}
};
}
_createSchemaUrlMatchRegExp(value) {
return {
required: ['url'],
properties: {
url: {type: 'string', pattern: value, patternFlags: 'i'}
}
};
}
// modifierKeys schema creation functions
_createSchemaModifierKeysAre(value) {
return this._createSchemaArrayCheck('modifierKeys', value, true, false);
}
_createSchemaModifierKeysAreNot(value) {
return {
not: [this._createSchemaArrayCheck('modifierKeys', value, true, false)]
};
}
_createSchemaModifierKeysInclude(value) {
return this._createSchemaArrayCheck('modifierKeys', value, false, false);
}
_createSchemaModifierKeysNotInclude(value) {
return this._createSchemaArrayCheck('modifierKeys', value, false, true);
}
// modifierKeys schema creation functions
_createSchemaFlagsAre(value) {
return this._createSchemaArrayCheck('flags', value, true, false);
}
_createSchemaFlagsAreNot(value) {
return {
not: [this._createSchemaArrayCheck('flags', value, true, false)]
};
}
_createSchemaFlagsInclude(value) {
return this._createSchemaArrayCheck('flags', value, false, false);
}
_createSchemaFlagsNotInclude(value) {
return this._createSchemaArrayCheck('flags', value, false, true);
}
// Generic
_createSchemaArrayCheck(key, value, exact, none) {
const containsList = [];
for (const item of this._split(value)) {
if (item.length === 0) { continue; }
containsList.push({
contains: {
const: item
}
});
}
const containsListCount = containsList.length;
const schema = {
type: 'array'
};
if (exact) {
schema.maxItems = containsListCount;
}
if (none) {
if (containsListCount > 0) {
schema.not = containsList;
}
} else {
schema.minItems = containsListCount;
if (containsListCount > 0) {
schema.allOf = containsList;
}
}
return {
required: [key],
properties: {
[key]: schema
}
};
}
}