mirror of
https://github.com/vrtmrz/obsidian-livesync.git
synced 2026-06-17 20:00:13 +00:00
24bc8404fe
- Fetch chunks on demand now respects network conditions. ### Misc - Prettified
378 lines
14 KiB
JavaScript
378 lines
14 KiB
JavaScript
import * as acorn from "acorn";
|
|
import fs from "fs";
|
|
|
|
// Parse command line arguments
|
|
const args = process.argv.slice(2);
|
|
let file = "main.js";
|
|
let target = 2018;
|
|
let ios = null;
|
|
|
|
// Help menu
|
|
if (args.includes("--help") || args.includes("-h")) {
|
|
console.log(`Usage: node utils/check-compatibility.js [options]
|
|
Options:
|
|
--file <path> Path to the bundle file to check (default: main.js)
|
|
--target <year> Target ECMAScript version (default: 2018)
|
|
--ios <version> Target iOS version (e.g. 14, 15, 16.4). Sets defaults automatically.
|
|
--[no-]allow-dynamic-import Allow dynamic import() expressions
|
|
--[no-]allow-bigint Allow BigInt literals
|
|
--[no-]allow-numeric-separator Allow numeric separators (e.g. 1_000)
|
|
--[no-]allow-class-fields Allow public/private/static class fields
|
|
--[no-]allow-class-static-blocks Allow class static initialization blocks
|
|
--[no-]allow-regexp-lookbehind Allow RegExp lookbehind assertions ((?<=...) / (?<!...))
|
|
--[no-]allow-regexp-indices Allow RegExp 'd' (indices) flag
|
|
--[no-]allow-regexp-v-flag Allow RegExp 'v' (Unicode properties) flag
|
|
`);
|
|
process.exit(0);
|
|
}
|
|
|
|
for (let i = 0; i < args.length; i++) {
|
|
if (args[i] === "--file" && args[i + 1]) {
|
|
file = args[i + 1];
|
|
i++;
|
|
} else if (args[i] === "--target" && args[i + 1]) {
|
|
target = parseInt(args[i + 1], 10);
|
|
i++;
|
|
} else if (args[i] === "--ios" && args[i + 1]) {
|
|
ios = parseFloat(args[i + 1]);
|
|
i++;
|
|
}
|
|
}
|
|
|
|
// Default feature flags based on target ECMA version
|
|
let allowDynamicImport = target >= 2020;
|
|
let allowBigInt = target >= 2020;
|
|
let allowNumericSeparator = target >= 2021;
|
|
let allowClassFields = target >= 2022;
|
|
let allowClassStaticBlocks = target >= 2022;
|
|
let allowRegexpLookbehind = target >= 2023;
|
|
let allowRegexpIndices = target >= 2022;
|
|
let allowRegexpVFlag = target >= 2024;
|
|
|
|
// Override feature flags if target iOS version is specified
|
|
if (ios !== null) {
|
|
// Determine a general baseline ECMA version for parser reports
|
|
if (ios >= 16.4) target = 2022;
|
|
else if (ios >= 15.0) target = 2021;
|
|
else if (ios >= 14.0) target = 2020;
|
|
else target = 2018;
|
|
|
|
allowDynamicImport = ios >= 11.3;
|
|
allowBigInt = ios >= 14.0;
|
|
allowNumericSeparator = ios >= 14.0;
|
|
allowClassFields = ios >= 14.5;
|
|
allowRegexpIndices = ios >= 15.0;
|
|
allowClassStaticBlocks = ios >= 16.4;
|
|
allowRegexpLookbehind = ios >= 16.4;
|
|
allowRegexpVFlag = ios >= 17.0;
|
|
}
|
|
|
|
// Override defaults with explicit command line options if specified
|
|
for (let i = 0; i < args.length; i++) {
|
|
if (args[i] === "--allow-dynamic-import") allowDynamicImport = true;
|
|
else if (args[i] === "--no-allow-dynamic-import") allowDynamicImport = false;
|
|
else if (args[i] === "--allow-bigint") allowBigInt = true;
|
|
else if (args[i] === "--no-allow-bigint") allowBigInt = false;
|
|
else if (args[i] === "--allow-numeric-separator") allowNumericSeparator = true;
|
|
else if (args[i] === "--no-allow-numeric-separator") allowNumericSeparator = false;
|
|
else if (args[i] === "--allow-class-fields") allowClassFields = true;
|
|
else if (args[i] === "--no-allow-class-fields") allowClassFields = false;
|
|
else if (args[i] === "--allow-class-static-blocks") allowClassStaticBlocks = true;
|
|
else if (args[i] === "--no-allow-class-static-blocks") allowClassStaticBlocks = false;
|
|
else if (args[i] === "--allow-regexp-lookbehind") allowRegexpLookbehind = true;
|
|
else if (args[i] === "--no-allow-regexp-lookbehind") allowRegexpLookbehind = false;
|
|
else if (args[i] === "--allow-regexp-indices") allowRegexpIndices = true;
|
|
else if (args[i] === "--no-allow-regexp-indices") allowRegexpIndices = false;
|
|
else if (args[i] === "--allow-regexp-v-flag") allowRegexpVFlag = true;
|
|
else if (args[i] === "--no-allow-regexp-v-flag") allowRegexpVFlag = false;
|
|
}
|
|
|
|
if (!fs.existsSync(file)) {
|
|
console.error(`Error: File '${file}' does not exist.`);
|
|
process.exit(1);
|
|
}
|
|
|
|
const code = fs.readFileSync(file, "utf8");
|
|
let ast;
|
|
|
|
const targetInfo = ios !== null ? `iOS ${ios}` : `ES${target}`;
|
|
console.log(`Parsing '${file}' to inspect compatibility (target ${targetInfo})...`);
|
|
console.log(`Rules:
|
|
Dynamic Import: ${allowDynamicImport ? "Allowed" : "Prohibited"}
|
|
BigInt: ${allowBigInt ? "Allowed" : "Prohibited"}
|
|
Numeric Separators: ${allowNumericSeparator ? "Allowed" : "Prohibited"}
|
|
Class Fields: ${allowClassFields ? "Allowed" : "Prohibited"}
|
|
Class Static Block: ${allowClassStaticBlocks ? "Allowed" : "Prohibited"}
|
|
RegExp Lookbehind: ${allowRegexpLookbehind ? "Allowed" : "Prohibited"}
|
|
RegExp Indices (d): ${allowRegexpIndices ? "Allowed" : "Prohibited"}
|
|
RegExp Unicode (v): ${allowRegexpVFlag ? "Allowed" : "Prohibited"}
|
|
`);
|
|
|
|
try {
|
|
ast = acorn.parse(code, { ecmaVersion: "latest", sourceType: "script" });
|
|
} catch (err) {
|
|
console.error(`Syntax Error: Failed to parse '${file}' due to a syntax issue:`);
|
|
console.error(err.message);
|
|
if (err.pos !== undefined) {
|
|
const line = code.substring(0, err.pos).split("\n").length;
|
|
console.error(`Location: line ${line}, character ${err.pos}`);
|
|
const start = Math.max(0, err.pos - 50);
|
|
const end = Math.min(code.length, err.pos + 50);
|
|
console.error("Context around error:");
|
|
console.error(code.substring(start, end));
|
|
console.error(" ".repeat(err.pos - start) + "^");
|
|
}
|
|
process.exit(1);
|
|
}
|
|
|
|
// Violations list
|
|
const violations = [];
|
|
|
|
function hasLookbehind(pattern) {
|
|
let index = 0;
|
|
while (true) {
|
|
const match = pattern.indexOf("(?<=", index);
|
|
const match2 = pattern.indexOf("(?<!", index);
|
|
const pos = match !== -1 && match2 !== -1 ? Math.min(match, match2) : match !== -1 ? match : match2;
|
|
if (pos === -1) break;
|
|
|
|
let backslashes = 0;
|
|
for (let i = pos - 1; i >= 0; i--) {
|
|
if (pattern[i] === "\\") backslashes++;
|
|
else break;
|
|
}
|
|
if (backslashes % 2 === 0) {
|
|
return true;
|
|
}
|
|
index = pos + 4;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
function checkNode(node) {
|
|
if (!node || typeof node !== "object") return;
|
|
|
|
if (node.type) {
|
|
// 1. Optional catch binding (ES2019 / iOS 11.3+)
|
|
if (node.type === "CatchClause" && !node.param) {
|
|
if (target < 2019 && ios === null) {
|
|
violations.push({
|
|
feature: "Optional catch binding (ES2019 / iOS 11.3+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
} else if (ios !== null && ios < 11.3) {
|
|
violations.push({
|
|
feature: "Optional catch binding (iOS 11.3+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
}
|
|
|
|
// 2. Dynamic import (ES2020 / iOS 11.3+)
|
|
if (node.type === "ImportExpression") {
|
|
if (!allowDynamicImport) {
|
|
violations.push({
|
|
feature: "Dynamic import (ES2020 / iOS 11.3+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
}
|
|
|
|
// 3. import.meta (ES2020 / iOS 11.3+)
|
|
if (node.type === "MetaProperty" && node.meta && node.meta.name === "import") {
|
|
if (!allowDynamicImport) {
|
|
violations.push({
|
|
feature: "import.meta (ES2020 / iOS 11.3+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
}
|
|
|
|
// 4. Optional chaining (ES2020 / iOS 13.4+)
|
|
if (node.type === "ChainExpression") {
|
|
const isProhibited = ios !== null ? ios < 13.4 : target < 2020;
|
|
if (isProhibited) {
|
|
violations.push({
|
|
feature: "Optional chaining (ES2020 / iOS 13.4+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
}
|
|
|
|
// 5. Nullish coalescing (ES2020 / iOS 13.4+)
|
|
if (node.type === "LogicalExpression" && node.operator === "??") {
|
|
const isProhibited = ios !== null ? ios < 13.4 : target < 2020;
|
|
if (isProhibited) {
|
|
violations.push({
|
|
feature: "Nullish coalescing (ES2020 / iOS 13.4+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
}
|
|
|
|
// 6. BigInt literal (ES2020 / iOS 14.0+)
|
|
if (node.type === "Literal" && node.bigint !== undefined) {
|
|
if (!allowBigInt) {
|
|
violations.push({
|
|
feature: "BigInt literal (ES2020 / iOS 14.0+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
}
|
|
|
|
// 7. Logical assignment (ES2021 / iOS 14.0+)
|
|
if (node.type === "AssignmentExpression" && ["||=", "&&=", "??="].includes(node.operator)) {
|
|
const isProhibited = ios !== null ? ios < 14.0 : target < 2021;
|
|
if (isProhibited) {
|
|
violations.push({
|
|
feature: `Logical assignment operator '${node.operator}' (ES2021 / iOS 14.0+)`,
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
}
|
|
|
|
// 8. Numeric separators (ES2021 / iOS 14.0+)
|
|
if (node.type === "Literal" && typeof node.value === "number" && node.raw && node.raw.includes("_")) {
|
|
if (!allowNumericSeparator) {
|
|
violations.push({
|
|
feature: "Numeric separator (ES2021 / iOS 14.0+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
}
|
|
|
|
// 9. Class Fields (ES2022 / iOS 14.0+ public, iOS 14.5+ private/static)
|
|
if (node.type === "PropertyDefinition") {
|
|
if (!allowClassFields) {
|
|
const requiredVersion =
|
|
node.key.type === "PrivateIdentifier" || node.static ? "iOS 14.5+" : "iOS 14.0+";
|
|
violations.push({
|
|
feature: `Class field definition '${node.key.name || node.key.value || "#private"}' (ES2022 / ${requiredVersion})`,
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
}
|
|
|
|
// 10. Class Static Initialization Blocks (ES2022 / iOS 16.4+)
|
|
if (node.type === "StaticBlock") {
|
|
if (!allowClassStaticBlocks) {
|
|
violations.push({
|
|
feature: "Class static initialization block (ES2022 / iOS 16.4+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
}
|
|
|
|
// 11. RegExp lookbehind assertions (ES2018 / iOS 16.4+)
|
|
if (node.type === "Literal" && node.regex) {
|
|
if (!allowRegexpLookbehind && hasLookbehind(node.regex.pattern)) {
|
|
violations.push({
|
|
feature: "RegExp Lookbehind assertion (iOS 16.4+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
if (!allowRegexpIndices && node.regex.flags.includes("d")) {
|
|
violations.push({
|
|
feature: "RegExp 'd' (indices) flag (ES2022 / iOS 15.0+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
if (!allowRegexpVFlag && node.regex.flags.includes("v")) {
|
|
violations.push({
|
|
feature: "RegExp 'v' (Unicode properties) flag (ES2024 / iOS 17.0+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
}
|
|
|
|
if (
|
|
(node.type === "NewExpression" || node.type === "CallExpression") &&
|
|
node.callee &&
|
|
node.callee.name === "RegExp"
|
|
) {
|
|
if (
|
|
!allowRegexpLookbehind &&
|
|
node.arguments[0] &&
|
|
node.arguments[0].type === "Literal" &&
|
|
typeof node.arguments[0].value === "string"
|
|
) {
|
|
if (hasLookbehind(node.arguments[0].value)) {
|
|
violations.push({
|
|
feature: "RegExp Lookbehind assertion (iOS 16.4+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
}
|
|
if (
|
|
node.arguments[1] &&
|
|
node.arguments[1].type === "Literal" &&
|
|
typeof node.arguments[1].value === "string"
|
|
) {
|
|
const flags = node.arguments[1].value;
|
|
if (!allowRegexpIndices && flags.includes("d")) {
|
|
violations.push({
|
|
feature: "RegExp 'd' (indices) flag (ES2022 / iOS 15.0+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
if (!allowRegexpVFlag && flags.includes("v")) {
|
|
violations.push({
|
|
feature: "RegExp 'v' (Unicode properties) flag (ES2024 / iOS 17.0+)",
|
|
pos: node.start,
|
|
node,
|
|
});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for (const key in node) {
|
|
if (key === "loc" || key === "start" || key === "end") continue;
|
|
const val = node[key];
|
|
if (Array.isArray(val)) {
|
|
for (const child of val) {
|
|
checkNode(child);
|
|
}
|
|
} else if (val && typeof val === "object") {
|
|
checkNode(val);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Run compatibility checks on the AST
|
|
checkNode(ast);
|
|
|
|
if (violations.length > 0) {
|
|
console.error(`\nCompatibility Check Failed: Found ${violations.length} prohibited features.`);
|
|
violations.forEach((v, index) => {
|
|
const line = code.substring(0, v.pos).split("\n").length;
|
|
console.error(`\n[${index + 1}] Prohibited feature: ${v.feature}`);
|
|
console.error(`Location: line ${line}, character ${v.pos}`);
|
|
const start = Math.max(0, v.pos - 50);
|
|
const end = Math.min(code.length, v.pos + 50);
|
|
console.error("Context around feature:");
|
|
console.error(code.substring(start, end));
|
|
console.error(" ".repeat(v.pos - start) + "^");
|
|
});
|
|
process.exit(1);
|
|
}
|
|
|
|
console.log(`\nCompatibility Check Passed: '${file}' matches the compatibility rules.`);
|
|
process.exit(0);
|