mirror of
https://github.com/nunocoracao/blowfish.git
synced 2025-04-21 09:11:53 +02:00
183 lines
No EOL
25 KiB
JavaScript
183 lines
No EOL
25 KiB
JavaScript
"use strict";
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
exports.isUnformattable = exports.isRoot = exports.isMultiBlock = exports.isBlock = exports.isInline = exports.parseGoTemplate = void 0;
|
|
const create_id_generator_1 = require("./create-id-generator");
|
|
const parseGoTemplate = (text, parsers, options) => {
|
|
var _a, _b, _c, _d, _e, _f, _g, _h, _j;
|
|
const regex = /{{(?<startdelimiter>-|<|%|\/\*)?\s*(?<statement>(?<keyword>if|range|block|with|define|end|else|prettier-ignore-start|prettier-ignore-end)?[\s\S]*?)\s*(?<endDelimiter>-|>|%|\*\/)?}}|(?<unformattableScript><(script)((?!<)[\s\S])*>((?!<\/script)[\s\S])*?{{[\s\S]*?<\/(script)>)|(?<unformattableStyle><(style)((?!<)[\s\S])*>((?!<\/style)[\s\S])*?{{[\s\S]*?<\/(style)>)/g;
|
|
const root = {
|
|
type: "root",
|
|
content: text,
|
|
aliasedContent: "",
|
|
children: {},
|
|
index: 0,
|
|
contentStart: 0,
|
|
length: text.length,
|
|
};
|
|
const nodeStack = [root];
|
|
const getId = (0, create_id_generator_1.createIdGenerator)();
|
|
for (let match of text.matchAll(regex)) {
|
|
const current = last(nodeStack);
|
|
const keyword = (_a = match.groups) === null || _a === void 0 ? void 0 : _a.keyword;
|
|
const statement = (_b = match.groups) === null || _b === void 0 ? void 0 : _b.statement;
|
|
const unformattable = (_d = (_c = match.groups) === null || _c === void 0 ? void 0 : _c.unformattableScript) !== null && _d !== void 0 ? _d : (_e = match.groups) === null || _e === void 0 ? void 0 : _e.unformattableStyle;
|
|
const startDelimiter = ((_g = (_f = match.groups) === null || _f === void 0 ? void 0 : _f.startdelimiter) !== null && _g !== void 0 ? _g : "");
|
|
const endDelimiter = ((_j = (_h = match.groups) === null || _h === void 0 ? void 0 : _h.endDelimiter) !== null && _j !== void 0 ? _j : "");
|
|
if (current === undefined) {
|
|
throw Error("Node stack empty.");
|
|
}
|
|
if (match.index === undefined) {
|
|
throw Error("Regex match index undefined.");
|
|
}
|
|
const id = getId();
|
|
if (unformattable) {
|
|
current.children[id] = {
|
|
id,
|
|
type: "unformattable",
|
|
index: match.index,
|
|
length: match[0].length,
|
|
content: unformattable,
|
|
parent: current,
|
|
};
|
|
continue;
|
|
}
|
|
if (statement === undefined) {
|
|
throw Error("Formattable match without statement.");
|
|
}
|
|
const inline = {
|
|
index: match.index,
|
|
length: match[0].length,
|
|
startDelimiter,
|
|
endDelimiter,
|
|
parent: current,
|
|
type: "inline",
|
|
statement,
|
|
id,
|
|
};
|
|
if (keyword === "end" || keyword === "prettier-ignore-end") {
|
|
if (current.type !== "block") {
|
|
throw Error("Encountered unexpected end keyword.");
|
|
}
|
|
current.length = match[0].length + match.index - current.index;
|
|
current.content = text.substring(current.contentStart, match.index);
|
|
current.aliasedContent = aliasNodeContent(current);
|
|
current.end = inline;
|
|
if (current.parent.type === "double-block") {
|
|
const firstChild = current.parent.blocks[0];
|
|
const lastChild = current.parent.blocks[current.parent.blocks.length - 1];
|
|
current.parent.length =
|
|
lastChild.index + lastChild.length - firstChild.index;
|
|
}
|
|
nodeStack.pop();
|
|
}
|
|
else if (isBlock(current) && keyword === "else") {
|
|
const nextChild = {
|
|
type: "block",
|
|
start: inline,
|
|
end: null,
|
|
children: {},
|
|
keyword: keyword,
|
|
index: match.index,
|
|
parent: current.parent,
|
|
contentStart: match.index + match[0].length,
|
|
content: "",
|
|
aliasedContent: "",
|
|
length: -1,
|
|
id: getId(),
|
|
startDelimiter,
|
|
endDelimiter,
|
|
};
|
|
if (isMultiBlock(current.parent)) {
|
|
current.parent.blocks.push(nextChild);
|
|
}
|
|
else {
|
|
const multiBlock = {
|
|
type: "double-block",
|
|
parent: current.parent,
|
|
index: current.index,
|
|
length: -1,
|
|
keyword,
|
|
id: current.id,
|
|
blocks: [current, nextChild],
|
|
};
|
|
nextChild.parent = multiBlock;
|
|
current.parent = multiBlock;
|
|
if ("children" in multiBlock.parent) {
|
|
multiBlock.parent.children[multiBlock.id] = multiBlock;
|
|
}
|
|
else {
|
|
throw Error("Could not find child in parent.");
|
|
}
|
|
}
|
|
current.id = getId();
|
|
current.length = match[0].length + match.index - current.index;
|
|
current.content = text.substring(current.contentStart, match.index);
|
|
current.aliasedContent = aliasNodeContent(current);
|
|
nodeStack.pop();
|
|
nodeStack.push(nextChild);
|
|
}
|
|
else if (keyword) {
|
|
const block = {
|
|
type: "block",
|
|
start: inline,
|
|
end: null,
|
|
children: {},
|
|
keyword: keyword,
|
|
index: match.index,
|
|
parent: current,
|
|
contentStart: match.index + match[0].length,
|
|
content: "",
|
|
aliasedContent: "",
|
|
length: -1,
|
|
id: getId(),
|
|
startDelimiter,
|
|
endDelimiter,
|
|
};
|
|
current.children[block.id] = block;
|
|
nodeStack.push(block);
|
|
}
|
|
else {
|
|
current.children[inline.id] = inline;
|
|
}
|
|
}
|
|
if (!isRoot(nodeStack.pop())) {
|
|
throw Error("Missing end block.");
|
|
}
|
|
root.aliasedContent = aliasNodeContent(root);
|
|
return root;
|
|
};
|
|
exports.parseGoTemplate = parseGoTemplate;
|
|
function aliasNodeContent(current) {
|
|
let result = current.content;
|
|
Object.entries(current.children)
|
|
.sort(([_, node1], [__, node2]) => node2.index - node1.index)
|
|
.forEach(([id, node]) => (result =
|
|
result.substring(0, node.index - current.contentStart) +
|
|
id +
|
|
result.substring(node.index + node.length - current.contentStart)));
|
|
return result;
|
|
}
|
|
function last(array) {
|
|
return array[array.length - 1];
|
|
}
|
|
function isInline(node) {
|
|
return node.type === "inline";
|
|
}
|
|
exports.isInline = isInline;
|
|
function isBlock(node) {
|
|
return node.type === "block";
|
|
}
|
|
exports.isBlock = isBlock;
|
|
function isMultiBlock(node) {
|
|
return node.type === "double-block";
|
|
}
|
|
exports.isMultiBlock = isMultiBlock;
|
|
function isRoot(node) {
|
|
return node.type === "root";
|
|
}
|
|
exports.isRoot = isRoot;
|
|
function isUnformattable(node) {
|
|
return node.type === "unformattable";
|
|
}
|
|
exports.isUnformattable = isUnformattable;
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parse.js","sourceRoot":"","sources":["../src/parse.ts"],"names":[],"mappings":"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","sourcesContent":["import { Parser } from \"prettier\";\nimport { createIdGenerator } from \"./create-id-generator\";\n\nexport const parseGoTemplate: Parser<GoNode>[\"parse\"] = (\n  text,\n  parsers,\n  options\n) => {\n  const regex =\n    /{{(?<startdelimiter>-|<|%|\\/\\*)?\\s*(?<statement>(?<keyword>if|range|block|with|define|end|else|prettier-ignore-start|prettier-ignore-end)?[\\s\\S]*?)\\s*(?<endDelimiter>-|>|%|\\*\\/)?}}|(?<unformattableScript><(script)((?!<)[\\s\\S])*>((?!<\\/script)[\\s\\S])*?{{[\\s\\S]*?<\\/(script)>)|(?<unformattableStyle><(style)((?!<)[\\s\\S])*>((?!<\\/style)[\\s\\S])*?{{[\\s\\S]*?<\\/(style)>)/g;\n  const root: GoRoot = {\n    type: \"root\",\n    content: text,\n    aliasedContent: \"\",\n    children: {},\n    index: 0,\n    contentStart: 0,\n    length: text.length,\n  };\n  const nodeStack: (GoBlock | GoRoot)[] = [root];\n  const getId = createIdGenerator();\n\n  for (let match of text.matchAll(regex)) {\n    const current = last(nodeStack);\n    const keyword = match.groups?.keyword as GoBlockKeyword | undefined;\n    const statement = match.groups?.statement;\n    const unformattable =\n      match.groups?.unformattableScript ?? match.groups?.unformattableStyle;\n\n    const startDelimiter = (match.groups?.startdelimiter ??\n      \"\") as GoInlineStartDelimiter;\n    const endDelimiter = (match.groups?.endDelimiter ??\n      \"\") as GoInlineEndDelimiter;\n\n    if (current === undefined) {\n      throw Error(\"Node stack empty.\");\n    }\n\n    if (match.index === undefined) {\n      throw Error(\"Regex match index undefined.\");\n    }\n    const id = getId();\n    if (unformattable) {\n      current.children[id] = {\n        id,\n        type: \"unformattable\",\n        index: match.index,\n        length: match[0].length,\n        content: unformattable,\n        parent: current,\n      };\n      continue;\n    }\n\n    if (statement === undefined) {\n      throw Error(\"Formattable match without statement.\");\n    }\n\n    const inline: GoInline = {\n      index: match.index,\n      length: match[0].length,\n      startDelimiter,\n      endDelimiter,\n      parent: current!,\n      type: \"inline\",\n      statement,\n      id,\n    };\n\n    if (keyword === \"end\" || keyword === \"prettier-ignore-end\") {\n      if (current.type !== \"block\") {\n        throw Error(\"Encountered unexpected end keyword.\");\n      }\n\n      current.length = match[0].length + match.index - current.index;\n      current.content = text.substring(current.contentStart, match.index);\n      current.aliasedContent = aliasNodeContent(current);\n      current.end = inline;\n\n      if (current.parent.type === \"double-block\") {\n        const firstChild = current.parent.blocks[0];\n        const lastChild =\n          current.parent.blocks[current.parent.blocks.length - 1];\n\n        current.parent.length =\n          lastChild.index + lastChild.length - firstChild.index;\n      }\n\n      nodeStack.pop();\n    } else if (isBlock(current) && keyword === \"else\") {\n      const nextChild: GoBlock = {\n        type: \"block\",\n        start: inline,\n        end: null,\n        children: {},\n        keyword: keyword,\n        index: match.index,\n        parent: current.parent,\n        contentStart: match.index + match[0].length,\n        content: \"\",\n        aliasedContent: \"\",\n        length: -1,\n        id: getId(),\n        startDelimiter,\n        endDelimiter,\n      };\n\n      if (isMultiBlock(current.parent)) {\n        current.parent.blocks.push(nextChild);\n      } else {\n        const multiBlock: GoMultiBlock = {\n          type: \"double-block\",\n          parent: current.parent,\n          index: current.index,\n          length: -1,\n          keyword,\n          id: current.id,\n          blocks: [current, nextChild],\n        };\n        nextChild.parent = multiBlock;\n        current.parent = multiBlock;\n\n        if (\"children\" in multiBlock.parent) {\n          multiBlock.parent.children[multiBlock.id] = multiBlock;\n        } else {\n          throw Error(\"Could not find child in parent.\");\n        }\n      }\n\n      current.id = getId();\n      current.length = match[0].length + match.index - current.index;\n      current.content = text.substring(current.contentStart, match.index);\n      current.aliasedContent = aliasNodeContent(current);\n\n      nodeStack.pop();\n      nodeStack.push(nextChild);\n    } else if (keyword) {\n      const block: GoBlock = {\n        type: \"block\",\n        start: inline,\n        end: null,\n        children: {},\n        keyword: keyword as GoBlockKeyword,\n        index: match.index,\n        parent: current,\n        contentStart: match.index + match[0].length,\n        content: \"\",\n        aliasedContent: \"\",\n        length: -1,\n        id: getId(),\n        startDelimiter,\n        endDelimiter,\n      };\n\n      current.children[block.id] = block;\n      nodeStack.push(block);\n    } else {\n      current.children[inline.id] = inline;\n    }\n  }\n\n  if (!isRoot(nodeStack.pop()!)) {\n    throw Error(\"Missing end block.\");\n  }\n\n  root.aliasedContent = aliasNodeContent(root);\n\n  return root;\n};\n\nfunction aliasNodeContent(current: GoBlock | GoRoot): string {\n  let result = current.content;\n\n  Object.entries(current.children)\n    .sort(([_, node1], [__, node2]) => node2.index - node1.index)\n    .forEach(\n      ([id, node]) =>\n        (result =\n          result.substring(0, node.index - current.contentStart) +\n          id +\n          result.substring(node.index + node.length - current.contentStart))\n    );\n\n  return result;\n}\n\nfunction last<T>(array: T[]): T | undefined {\n  return array[array.length - 1];\n}\n\nexport type GoNode =\n  | GoRoot\n  | GoBlock\n  | GoInline\n  | GoMultiBlock\n  | GoUnformattable;\n\nexport type GoBlockKeyword =\n  | \"if\"\n  | \"range\"\n  | \"block\"\n  | \"with\"\n  | \"define\"\n  | \"else\"\n  | \"prettier-ignore-start\"\n  | \"prettier-ignore-end\"\n  | \"end\";\n\nexport type GoRoot = { type: \"root\" } & Omit<\n  GoBlock,\n  | \"type\"\n  | \"keyword\"\n  | \"parent\"\n  | \"statement\"\n  | \"id\"\n  | \"startDelimiter\"\n  | \"endDelimiter\"\n  | \"start\"\n  | \"end\"\n>;\n\nexport interface GoBaseNode<Type extends string> {\n  id: string;\n  type: Type;\n  index: number;\n  length: number;\n  parent: GoBlock | GoRoot | GoMultiBlock;\n}\n\nexport interface GoBlock extends GoBaseNode<\"block\">, WithDelimiter {\n  keyword: GoBlockKeyword;\n  children: {\n    [id: string]: GoNode;\n  };\n  start: GoInline;\n  end: GoInline | null;\n  content: string;\n  aliasedContent: string;\n  contentStart: number;\n}\n\nexport interface GoMultiBlock extends GoBaseNode<\"double-block\"> {\n  blocks: (GoBlock | GoMultiBlock)[];\n  keyword: GoBlockKeyword;\n}\n\nexport type GoSharedDelimiter = \"%\" | \"-\" | \"\";\nexport type GoInlineStartDelimiter = \"<\" | \"/*\" | GoSharedDelimiter;\nexport type GoInlineEndDelimiter = \">\" | \"*/\" | GoSharedDelimiter;\n\nexport interface GoUnformattable extends GoBaseNode<\"unformattable\"> {\n  content: string;\n}\n\nexport interface WithDelimiter {\n  startDelimiter: GoInlineStartDelimiter;\n  endDelimiter: GoInlineEndDelimiter;\n}\n\nexport interface GoInline extends GoBaseNode<\"inline\">, WithDelimiter {\n  statement: string;\n}\n\nexport function isInline(node: GoNode): node is GoInline {\n  return node.type === \"inline\";\n}\n\nexport function isBlock(node: GoNode): node is GoBlock {\n  return node.type === \"block\";\n}\n\nexport function isMultiBlock(node: GoNode): node is GoMultiBlock {\n  return node.type === \"double-block\";\n}\n\nexport function isRoot(node: GoNode): node is GoRoot {\n  return node.type === \"root\";\n}\n\nexport function isUnformattable(node: GoNode): node is GoRoot {\n  return node.type === \"unformattable\";\n}\n"]}
|