aboutsummaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
-rw-r--r--.github/workflows/gen.yml4
-rw-r--r--docs/code.js59
-rwxr-xr-xstd/generate_map.py48
-rwxr-xr-xstd/generate_map.sh20
4 files changed, 87 insertions, 44 deletions
diff --git a/.github/workflows/gen.yml b/.github/workflows/gen.yml
index d93dea9..5c20fff 100644
--- a/.github/workflows/gen.yml
+++ b/.github/workflows/gen.yml
@@ -31,8 +31,8 @@ jobs:
run: cd docs/ && ./gen.sh
- name: Generate std map
run: |
- sudo apt install jq
- cd std/ && ./generate_map.sh >../docs/std_map.json
+ sudo apt install python
+ cd std/ && ./generate_map.py >../docs/std_map.json
- name: Upload artifact
uses: actions/upload-pages-artifact@v1
with:
diff --git a/docs/code.js b/docs/code.js
index 6093a9e..7572c0b 100644
--- a/docs/code.js
+++ b/docs/code.js
@@ -188,7 +188,9 @@ const bruijnParse = (stdMap) => {
} else if (token.classList[0] === "pp") {
// preprocessor
const instr = inner.trim();
- if (instr.startsWith(":import") || instr.startsWith(":input")) {
+ const isImport = instr.startsWith(":import");
+ const isInput = instr.startsWith(":input");
+ if (isImport || isInput) {
const parts = instr.split(" ");
const path = parts[1].replace("std/", "") + ".bruijn";
if (!(path in stdMap)) {
@@ -196,27 +198,28 @@ const bruijnParse = (stdMap) => {
continue;
}
- // TODO: inputs currently don't respect its imports
stdMap[path].forEach((d) => {
- const name = parts[0].startsWith(":input")
+ const name = isInput
? d.name
: parts[2] == "."
? d.name
: parts[2] + "." + d.name;
const link = d.source.replace("/", "_");
- root.children.push({
- kind: "import",
- elem: token,
- indent: 0,
- action: (f) =>
- window.open(
- `https://bruijn.marvinborner.de/std/${link}.html#${f}`,
- "_blank",
- ),
- children: [],
- nodes: [],
- name,
- });
+ const original = path.replace("/", "_");
+ if (isInput || (isImport && d.kind !== "import"))
+ root.children.push({
+ kind: "import",
+ elem: token,
+ indent: 0,
+ action: (f) =>
+ window.open(
+ `https://bruijn.marvinborner.de/std/${f == "" ? original : link}.html#${f}`,
+ "_blank",
+ ),
+ children: [],
+ nodes: [],
+ name,
+ });
});
} else if (instr.startsWith(":test") || instr.startsWith(":time")) {
root.children.push(fresh);
@@ -241,12 +244,14 @@ const bruijnParse = (stdMap) => {
) {
// namespaced prefix
const decoded = decodeHTML(tokens[j].innerHTML.trim());
+ const prefixed =
+ decoded[decoded.length - 1] === "‣" ? decoded : decoded + "‣";
fresh.nodes.push({
kind: "prefix",
elem: tokens[j],
- name: tokens[j - 1].innerHTML.trim() + decoded + "‣",
+ name: tokens[j - 1].innerHTML.trim() + prefixed,
bruijnStack: [...bruijnStack], // for subs
- canonical: decoded.slice(1) + "‣",
+ canonical: prefixed.slice(1),
});
} else if (
tokens[j].classList[0] === "special" &&
@@ -254,21 +259,26 @@ const bruijnParse = (stdMap) => {
) {
// normal prefix
const decoded = decodeHTML(tokens[j].innerHTML.trim());
+ const prefixed =
+ decoded[decoded.length - 1] === "‣" ? decoded : decoded + "‣";
fresh.nodes.push({
kind: "prefix",
elem: tokens[j],
- name: decoded + "‣",
+ name: prefixed,
bruijnStack: [...bruijnStack], // for subs
- canonical: decoded + "‣",
+ canonical: prefixed,
});
// normal mixfix TODO: this is a ugly hack and only works if lucky and binary
+ const mixfixed = decoded[decoded.length - 1].includes("…")
+ ? decoded
+ : "…" + decoded + "…";
fresh.nodes.push({
kind: "mixfix",
elem: tokens[j],
- name: "…" + decoded + "…",
+ name: mixfixed,
bruijnStack: [...bruijnStack], // for subs
- canonical: "…" + decoded + "…",
+ canonical: mixfixed,
});
} else if (
j > 1 &&
@@ -362,6 +372,11 @@ const bruijnParse = (stdMap) => {
metaStacks[sup.name] = metaStack;
});
}
+
+ // TODO: case is missing:
+ // foo [bar]
+ // baz [0 1 2] # won't get highlighted
+ // bar [baz]
}
}
}
diff --git a/std/generate_map.py b/std/generate_map.py
new file mode 100755
index 0000000..62c9920
--- /dev/null
+++ b/std/generate_map.py
@@ -0,0 +1,48 @@
+#!/bin/env python
+
+from pathlib import Path
+import json
+
+
+def list_defs(path, kind, prefix):
+ res = []
+ for line in open(path).readlines():
+ if line.startswith(":input"):
+ parts = line.split(" ")
+ input_path = parts[1].strip().split("std/")[1] + ".bruijn"
+ # not using kind="input" is important, "import"s should be inherited
+ res = res + list_defs(input_path, kind, prefix)
+ elif line.startswith(":import") and kind == "input":
+ parts = line.split(" ")
+ import_path = parts[1].strip().split("std/")[1] + ".bruijn"
+ new_prefix = (
+ "" if parts[2].strip() == "." else parts[2].strip() + "."
+ )
+ new_prefix = prefix + new_prefix if prefix else new_prefix
+ res = res + list_defs(import_path, "import", new_prefix)
+ elif (
+ line.startswith(":")
+ or line.startswith("#")
+ or line.strip() == ""
+ or line[0].isspace()
+ ):
+ continue
+ else:
+ parts = line.strip().split(" ")
+ res.append(
+ {
+ "name": prefix + parts[0],
+ "source": path,
+ "kind": kind,
+ }
+ )
+ return res
+
+
+res = {}
+files = Path(".").glob("**/*.bruijn")
+for file in files:
+ path = str(file)
+ if path != "All.bruijn" and "Generic" not in path:
+ res[path] = list_defs(path, "input", "")
+print(json.dumps(res))
diff --git a/std/generate_map.sh b/std/generate_map.sh
deleted file mode 100755
index 193a62b..0000000
--- a/std/generate_map.sh
+++ /dev/null
@@ -1,20 +0,0 @@
-#!/bin/sh
-
-FILES="$(find * -type f -name "*.bruijn" ! -name "All.bruijn" ! -path "*Generic*")"
-
-# TODO: also handle imports for :input intelligence
-list_defs() {
- grep -Po "^[^:# \t][^ \t]*" "$1" | sed -e "s#\\(.*\\)#{\"name\": \"\\1\", \"source\": \"$1\"}#g"
- inputs="$(awk '/^:input/ {print $2}' "$1")"
- for i in $inputs; do
- list_defs "${i#std/}.bruijn"
- done
-}
-
-{
- for f in $FILES; do
- echo "{\"$f\":"
- list_defs "$f" | sed 's/\\/\\\\/g' | jq -s .
- echo "}"
- done
-} | jq -s add