securemark 0.297.0 → 0.297.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -1,5 +1,13 @@
1
1
  # Changelog
2
2
 
3
+ ## 0.297.2
4
+
5
+ - Refactoring.
6
+
7
+ ## 0.297.1
8
+
9
+ - Fix note processing.
10
+
3
11
  ## 0.297.0
4
12
 
5
13
  - Enhance annotation parser to allow nested annotations.
package/design.md CHANGED
@@ -293,7 +293,7 @@ Securemarkはブロック単位の差分更新によりリアルタイムレン
293
293
 
294
294
  CommonMarkは初手設計ミスったせいで構文一つで最悪計算量32nを叩き出す最悪計算量O(n\*32^m)のクソ言語になり拡張性がなく終わってる。
295
295
 
296
- SecuremarkのAnnotation構文に典型的であるように文脈を変更する構文の中にその文脈に依存し変更される他の構文が存在する場合文脈の相違から解析結果を再利用不能(`αAβ | αA'B`)なバックトラックが生じる。またこの結果再帰的バックトラックが生じる可能性があり再帰的バックトラックは一般的にメモ化により解決されるがCommonMarkは最小計算量と実行性能を追及するためメモ化を廃止していることからメモ化により性能を低下させてまで文脈依存構文の問題を解決するつもりはないと思われる(すなわちCommonMarkは機械を至上とし人間に制約を課す低水準の言語であり人間の需要と生産性を至上とする高水準の言語であるSecuremarkの対極に位置する)。従ってCommonMarkは再帰的バックトラックなしで解析可能な構文と最小計算量に制約され基本的に再帰的バックトラックが生じるものである文脈依存構文を追加できないという拡張性の欠陥が存在する(CommonMarkは文脈を変えず`*A*`の記号と解釈を変更しただけの`~~A~~`のような文脈自由構文は容易に追加できるが他の構文を包含可能な文脈依存構文はリンク構文で発生している再帰的バックトラックまたはその修正のために激増した計算量を組み合わせ爆発により指数関数的に致命的に悪化させるため追加できない。例えばすでに最悪計算量32nであるリンク構文を包含した追加構文が1回バックトラックするだけで最悪計算量が32n\*2=64nに悪化し追加構文がリンク構文と同じく最悪計算量32nなら全体の最悪計算量は32n\*32=n\*32^2=1024nとなる。すなわちCommonMarkの最悪計算量は再帰的バックトラックが生じる構文の数mに対してO(n\*32^m)と驚異的かつ絶望的に劣悪な計算量であり構文数が一つであろうと自己再帰可能ならば入れ子数mに対して同じくO(n\*32^m)となる。なお一般的に実用可能な計算量の上限はO(n^3)であるためCommonMarkは文脈依存構文数または入れ子数m>3になるだけで実用可能な計算量の上限を超える。すなわちCommonMarkは自己再帰可能な文脈依存構文を定義することが不可能である。この事実に開発開始から数年後に直面し再帰数の上限を32回に制限することで指数計算量を防ぐも依然として32nという非常識な計算量になっているのがリンク構文とイメージ構文である。なおリンク構文とイメージ構文を入れ子にできているのはイメージ構文を入れる角括弧部分が文脈自由部分だからであり文脈依存部分である丸括弧部分は入れ子になっておらず文脈依存部分の入れ子が生じていないからである。しかし文脈依存構文を追加してリンク構文を包含すれば丸括弧部分を包含し文脈依存部分が入れ子になり計算量の爆発を避けられない)。CommonMarkの仕様策定者が構文の拡張に(名称を維持するか否かにかかわらず)不自然なまでに消極的であり進展がないのは正当な理由や単なる怠慢からでなく構文追加可否を議論すればCommonMarkは理論的欠陥により最悪計算量が絶望的に劣悪であるため文脈依存構文を追加できないというCommonMarkの設計の破綻と失敗が明らかになることが避けられないためおよび最小計算量を放棄して現在の高い実行性能を低下させたくないためであり陳腐な自尊心を守るためにこのような計算量と拡張性の根本的欠陥を秘匿しCommonMarkとその仕様策定者である自分の評価が下がらないよう画策しているからである。でなければこの計算量と拡張性の根本的欠陥を何年も隠蔽せず速やかに公表して助力を求めていなければならず不都合な事実を隠し陳腐な自尊心を守るために全Markdown利用者および開発者を不必要に足止めした罪は重い。
296
+ SecuremarkのReference構文に典型的であるように文脈を変更する構文の中にその文脈に依存し変更される他の構文が存在する場合文脈の相違から解析結果を再利用不能(`αAβ | αA'B`)なバックトラックが生じる。またこの結果再帰的バックトラックが生じる可能性があり再帰的バックトラックは一般的にメモ化により解決されるがCommonMarkは最小計算量と実行性能を追及するためメモ化を廃止していることからメモ化により性能を低下させてまで文脈依存構文の問題を解決するつもりはないと思われる(すなわちCommonMarkは機械を至上とし人間に制約を課す低水準の言語であり人間の需要と生産性を至上とする高水準の言語であるSecuremarkの対極に位置する)。従ってCommonMarkは再帰的バックトラックなしで解析可能な構文と最小計算量に制約され基本的に再帰的バックトラックが生じるものである文脈依存構文を追加できないという拡張性の欠陥が存在する(CommonMarkは文脈を変えず`*A*`の記号と解釈を変更しただけの`~~A~~`のような文脈自由構文は容易に追加できるが他の構文を包含可能な文脈依存構文はリンク構文で発生している再帰的バックトラックまたはその修正のために激増した計算量を組み合わせ爆発により指数関数的に致命的に悪化させるため追加できない。例えばすでに最悪計算量32nであるリンク構文を包含した追加構文が1回バックトラックするだけで最悪計算量が32n\*2=64nに悪化し追加構文がリンク構文と同じく最悪計算量32nなら全体の最悪計算量は32n\*32=n\*32^2=1024nとなる。すなわちCommonMarkの最悪計算量は再帰的バックトラックが生じる構文の数mに対してO(n\*32^m)と驚異的かつ絶望的に劣悪な計算量であり構文数が一つであろうと自己再帰可能ならば入れ子数mに対して同じくO(n\*32^m)となる。なお一般的に実用可能な計算量の上限はO(n^3)であるためCommonMarkは文脈依存構文数または入れ子数m>3になるだけで実用可能な計算量の上限を超える。すなわちCommonMarkは自己再帰可能な文脈依存構文を定義することが不可能である。この事実に開発開始から数年後に直面し再帰数の上限を32回に制限することで指数計算量を防ぐも依然として32nという非常識な計算量になっているのがリンク構文とイメージ構文である。なおリンク構文とイメージ構文を入れ子にできているのはイメージ構文を入れる角括弧部分が文脈自由部分だからであり文脈依存部分である丸括弧部分は入れ子になっておらず文脈依存部分の入れ子が生じていないからである。しかし文脈依存構文を追加してリンク構文を包含すれば丸括弧部分を包含し文脈依存部分が入れ子になり計算量の爆発を避けられない)。CommonMarkの仕様策定者が構文の拡張に(名称を維持するか否かにかかわらず)不自然なまでに消極的であり進展がないのは正当な理由や単なる怠慢からでなく構文追加可否を議論すればCommonMarkは理論的欠陥により最悪計算量が絶望的に劣悪であるため文脈依存構文を追加できないというCommonMarkの設計の破綻と失敗が明らかになることが避けられないためおよび最小計算量を放棄して現在の高い実行性能を低下させたくないためであり陳腐な自尊心を守るためにこのような計算量と拡張性の根本的欠陥を秘匿しCommonMarkとその仕様策定者である自分の評価が下がらないよう画策しているからである。でなければこの計算量と拡張性の根本的欠陥を何年も隠蔽せず速やかに公表して助力を求めていなければならず不都合な事実を隠し陳腐な自尊心を守るために全Markdown利用者および開発者を不必要に足止めした罪は重い。
297
297
  CommonMarkは小さく単純であるうちだけ破綻せずに正しく見せかけられる象牙の塔であり仕様策定者はこの見せかけの正しさを失わず正しいままでいたいがために象牙の塔に引きこもり小さな表面的正しさに固執し続けているに過ぎない。しかしCommonMarkは実際にはすでに致命的に破綻しており本来文脈依存言語であるMarkdownを文脈自由言語として解析しようとして失敗したことで指数計算量(正確には指数計算量より悪い階乗計算量)になっており実際のところCommonMarkは最初から最後までずっと壊れている。CommonMarkは可能な限り文脈自由言語として設計することでバックトラックなく最小計算量で解析しようとしているが実際には文脈依存言語であるMarkdownから文脈依存構文を文脈自由構文に変換して再帰的バックトラックを除去することに失敗しているためCommonMarkは最初の数年間は再帰的バックトラックに気づかず最悪計算量が指数計算量になっており修正後も依然として最悪計算量が当初の想定の1nから32nへと劇的に悪化している(より正確にはCommonMarkは少なくともCode構文とHTML構文により+2nされるが説明の簡略化のため省略しこれら以外のInline構文のみを比較する)。CommonMarkが最初の数年間他人に指摘され修正されるまで指数計算量であった事実(https://github.com/commonmark/cmark/commit/45f4fc9b917c11221aa03e70a41e3046335a235d)はCommonMarkが初歩的な再帰的バックトラックの原理すら理解していないド素人により設計された素人仕事である事実を証明しておりたかが強調構文の解析のためにメモ化を行い(https://github.com/commonmark/commonmark.js/commit/6d7d6cf150dedb53b7f0972b79313df3364ebbed https://github.com/commonmark/commonmark.js/blob/ac8529c9f55da7fdc1186e3f34313cf411de6e71/js/stmd.js )他人にメモ化を使わない正しい実装に修正された事実(https://github.com/commonmark/commonmark.js/commit/8837f199608ac2e321f75653736747b1e692072f)もまたCommonMarkの仕様策定者がその任に堪える能力のないド素人である事実を証明している。スタックを使う代わりにメモ化するド素人の能力を誰が擁護できるのか?ド素人が作った結果初歩的な再帰的バックトラックすら他人に指摘されるまで気づかず修正後も最悪計算量が32nにまで著しく悪化した設計が正しいと言えるのか?不可能である。一貫してド素人により設計開発仕様策定されているCommonMarkは仕様策定が停滞し未だに1nの最小計算量に固執しているがそんなものはCommonMarkの破綻と失敗が露呈するを恐れて隠蔽しまたとっくの昔に破綻している最小計算量のための設計に未練がましく執着しているだけである。32nという最悪計算量は適切に設計し計算量が抑制された文脈依存言語の最悪計算量より悪く最初から文脈依存言語として適切に設計するほうが自然で破綻がなく拡張性を確保できていた。たった一つの構文だけで最悪計算量が32nも増加するクソ言語に拡張性がないことは明らかである。さらにSecuremarkは再帰的バックトラックを予め対策して設計されているため文脈依存構文数が増えても最悪計算量はたかだか構文数に比例して線形にしか増加しないのに対してCommonMarkは再帰的バックトラックに根本的対策を取れないため文脈依存構文数が増えると最悪計算量が指数関数的に致命的に爆発する。または計算量を組み合わせ爆発させないために文脈依存構文の入れ子使用を制限する、存在自体が破綻と失敗の宣言に等しい制限が必要になる。文脈依存構文を強引に文脈自由構文として解析しようとして失敗したために最悪計算量が当初の想定の1nから32nに劇的に悪化し結局文脈依存言語の妥当な最悪計算量より悪い水準に落ちていることおよび文脈依存構文を追加すると最悪計算量が指数関数的に悪化することから文脈自由化に失敗したCommonMarkの破綻と失敗は明らかでありCommonMarkは文脈自由構文に固執せず最初から多少の文脈依存構文を許容するよう設計しなければならなかった。実際には文脈依存言語であるにもかかわらず文脈自由言語の範囲でしか構文解析できなければ構文解析が破綻し構文が増えるほど破綻が拡大することは自明でありすでに破綻済みで失敗済みのCommonMarkに未来などない。文脈依存言語であるMarkdownに対して文脈自由構文解析器として作られたCommonMarkは最初から開発戦略と技術選択を間違え失敗しており最初から破綻していた。CommonMarkが文脈依存言語を文脈自由言語として最小計算量で解析するために使用した手法は邪道の小手先の技術に過ぎずCommonMarkは邪道を選んだ挙句失敗に終わったのである。文脈依存言語を文脈依存言語のまま解析する正道を選んだSecuremarkが正着し文脈依存言語を文脈自由言語に歪める邪道を選んだCommonMarkが失着に終わったのは当然の帰結であり最初の解析戦略時選択の時点で決まっていたことである。文脈依存言語であるMarkdownを文脈自由言語として解析しようとした結果行き詰ったCommonMarkとその閉塞に技術的合理性はなくCommonMarkは最初からの失敗していた過去の遺物であり廃棄すべき負債である。CommonMarkに動きがないのはすでに破綻しており死んでいることに気付かれないように死んでいるからに過ぎない。このようにCommonMarkは完全に破綻し失敗に終わっているためCommonMarkの拡張や発展を期待しても無駄であり既存の文脈依存構文の最悪計算量を健全化し新たに文脈依存構文を追加可能な拡張性の高いMarkdown仕様は新しく作り直さなければ作れない。しかしCommonMarkの仕様策定者は独自の新しい仕様においても依然としてMarkdownを文脈自由化して文脈自由言語として設計しているため文脈依存構文を追加すると指数計算量になり救いようがない。しかもその構文と仕様は機械可読性を至上としているため非常に醜く人間が書くことも読むことも困難で人間に対する実用性が欠如している。
298
298
  Securemarkはスーパークラス構文が解析に失敗した入力をサブクラス構文で解析しないことにより再帰的バックトラックを回避する(解析中の構文自身はスーパークラスとサブクラスの両方に含まれるものとする)。スーパークラス構文A(`αAβ`)の解析が失敗すればサブクラス構文B(`α'Aβ'`)の解析も失敗することは自明であり試みるまでもなく解析を省略できる。これは二つの構文の文法が生成する各言語空間がスーパーセットとサブセットの関係にあるならスーパーセットの言語空間の外にある文字列はサブセットの言語空間の内に入る余地がないことからも自明である(この解析法は事前解析によっても可能だが文脈内外のオートリンクURLの括弧解析などを高速に行うこともバックトラックなしで解析することも困難であり本解析に近い解析コストを要するためMarkdownをこの事前解析により高速化することは難しい)。メモ化は解析結果の再利用による高速化の副次的または追加的効果として再帰的バックトラックを回避しているのでありメモ化は再帰的バックトラックを回避するだけなら過剰であり不要である(メモ化はバックトラックがなければ使用されないためバックトラックが少ないほとんどの入力に対してはほとんど使用されず無駄であり空間計算量を不必要に常時数倍以上増加させることから構文解析において必須にも標準にもできない。再帰的バックトラック回避のためにメモ化するとバックトラックなしで解析可能な場合も常に不必要に空間計算量が増加し平均計算量が有意に悪化することがメモ化の最大の欠点である(この問題は解析失敗時のみメモ化すれば解消可能のはずだが基本的にはこうなる)。特に文脈自由構文解析器におけるメモ化の使用は完全に無駄でありバグである。バックトラックが発生しないか他の方法で解決されるならば使用されず破棄されるメモ化は無駄である。複数の文脈で解析結果が同一である文脈独立性のある構文ならメモ化した解析結果を異なる文脈で再利用でき有用だがそのような構文は基本的に少数であるため効果が限定的であり最悪計算量は改善されない。文脈依存構文の解析を困難にするメモ化のコストの重さは解析に本来不要な高速化のために生じておりメモ化からバックトラックがない通常時も常時高コストで低速化し平均実行速度とスループットを低下させるバックトラック時限定の不要な高速化とこのためのコストを除去すれば再帰的バックトラック回避機能だけを低コストで標準的に使用しこれまで高コストであったため敷居が高く使用困難であった文脈依存言語を低コストで手軽に定義および解析できるようになる。もうCommonMarkのように低機能のくせに最悪計算量が劣悪で拡張不能なクソ言語を我慢して使い続けなくてよくなるのである)。この解析法により、CommonMarkならば最悪計算量がn\*32^5=33,554,432nとなる拡張Markdown言語をSecuremarkはメモ化なしで5nの最悪時間計算量で解析している。言語差を考慮せず直接比較してさえ単純なCommonMarkの最悪計算量32nに対して複雑なSecuremarkは5nに過ぎずSecuremarkはCommonMarkより最悪計算量が非常に小さい。またSecuremarkはメモ化を行っていないため空間計算量も小さい。時間計算量と空間計算量を合わせてO(n, n)と表記すると文脈依存言語の通常の最悪計算量はO(n^2, n)、メモ化により効率化できた場合もO(nm, nm)(S(m)>=m byte)(解析結果の位置や構文木等を記録するため空間使用量S(m)>はm byte以上となり数値一つやポインタ一つだけでも8byteを消費する)に過ぎないがSecuremarkのマーキング法はO(nm, nm)(S(m)=m bit)(解析の失敗フラグしか記録しないためS(m)=m bit。また包含文字列を含め全体でn byteの構文1つに対して解析結果の構文木を保持するメモ化は少なくとも構文木のサイズn byteを消費するがマーキング法のメモリ消費量は構文全体のサイズにかかわらず1bit固定である。よって100KBの構文1つに対してメモ化は100KB以上消費するがこの場合もマーキング法は1bitしか消費しない。なお成功フラグによる解析は成功フラグがないのが失敗したからか未到達だからかを判別するための情報と処理が追加で必要になり処理が複雑化しさらに成功フラグは解析のほとんどを占める成功した解析に対してメモリ消費と追加処理が発生し解析効率が全体として悪化するが失敗フラグは少数の失敗した解析でしか解析効率が悪化しないため失敗フラグを記録するほうが全体および平均として解析効率が高く優れている)と極めて効率的であり最も優れている。以上のようにSecuremarkの構文解析アルゴリズムの優位性は理論と実践いずれにおいても圧倒的である。現在のSecuremarkは開発効率と安全性優先の実装により実行性能が大きく低下しているがクライアントで個別のユーザーの操作に応じるには十分高速であるためクライアントで解析するなら解析の効率または速度が実用上問題となることはなく仕様が固まり実行効率優先の高速な実装に移れば他の環境での懸念もないだろう。またSecuremarkの再帰数制限はパーサーコンビネーターの使用による再帰的関数呼び出しに対する実装方式依存の制限であるため計算量と無関係であり再帰的関数呼び出しが生じないよう書き換えれば再帰数制限もない。従ってCommonMarkは再帰上限数と最悪計算量が一致し再帰上限数32回と最悪計算量32nが一致するがSecuremarkの最悪計算量は再帰上限数が100回だろうと100万回だろうと5n固定である。SecuremarkをCommonMarkのような再帰数制限のない実装に変換することは設計上および理論的に何の問題もないがCommonMarkをSecuremarkのような正常な文脈依存言語解析器に変更することは現在の異常な解析規則の破壊的変更による修正なしに不可能である。具体的には二重リンク`[[]()]()`を解析するときCommonMarkはバックトラックと計算量を最小化すべく文脈自由構文解析器として設計されているためリンク構文内をリンク構文が定義されていない異なる文脈として解析せず外側のリンク構文の解析を破棄して内側のみリンク構文として解析するがSecuremarkは文脈依存構文解析器とし設計されているためリンク構文内にリンク構文が定義されておらず外側のみリンクとして解析し他のMarkdown構文解析器もこのように解析しなければならない(ここでCommonMarkはリンク構文`[]()`のバックトラック除去ひいては文脈自由化に角括弧部分`[]`に対しては成功したが丸括弧部分`()`に対しては失敗したことで最悪計算量が指数計算量ないし32nに悪化した。リンク構文を本来通り文脈依存構文として解析すればリンク構文の最悪計算量が2nとなり角括弧部分に限っては1nから2nに悪化するが丸括弧部分は32nから2nに著しく改善する。ここがCommonMarkの根本的な欠陥と失敗が最も明瞭に表出している部分である)。この問題はイメージ構文においてさらに顕著でありリンク構文と同じ問題が正当な表現`![![]()]()`で発生しさらにHTMLのaltはプレーンテキストとして表示されるためMarkdownのaltもプレーンテキストとしてそのまま表示されなければならないにもかかわらず文脈を一致させ再帰的バックトラックを防ぐためにMarkdown文字列として解析しaltに`*a*`と書かれたものを`a`に不正に変換して表示する。無論新しい文脈依存構文を追加する場合も同じ制約が永遠についてまわり構文内文字列をMarkdownとして解析する文脈依存構文においてこの制約を破ると最悪計算量が32\*2^mないし32^mで指数関数的に増加する。すなわちCommonMarkは文脈依存構文を追加すると最悪計算量が32\*2^mないし32^mで指数関数的に悪化するという拡張性の致命的欠陥が存在する。こんな最悪計算量が32^mで組み合わせ爆発する欠陥言語を拡張できるわけがないことはもはや明白である。また大多数のプログラミング言語を見ても明らかなように文脈依存言語は構文内で使用可能な構文を定義しその他の構文は使用できず例外処理するのが通常でありCommonMarkのように本来使用不能な構文を外側の構文を無効化して使用可能に変える異常な言語はほとんどの人間はCommonMark以外に見たことがないだろう。ほぼすべての人間において他のすべての言語が同じ一貫した規則を持ち同じ規則で統一的に使用できるのに対してCommonMarkだけが他と異なる異常な挙動をして使用者に認知的負荷をかけるのである。破壊的変更を避けるため旧構文だけ従来通り文脈自由構文として解析し新構文を文脈依存構文として解析すればキメラ的な非常に不自然かつ歪で一貫性のない解析規則によりCommonMarkという一つの言語の中だけでもユーザーを混乱させるものとなり旧構文で使用した苦肉の策を不必要に新構文でも使用して一貫させれば文脈依存言語なのに文脈自由言語の苦肉の策の解析規則で解析されるこれもまたキメラ的な非常に不自然かつ理論的に設計ミスが明白で実用的にも認知的負荷の高い言語となる。正しく設計された言語と解析器にはありえないこのような解析規則の根本的不整合はその不整合の存在そのものが言語と解析器が誤った設計による失敗作であることを証明しており不整合が存在する限りその不整合により自らが失敗作であることを宣言し続けることになる。CommonMarkとはそういう失敗した言語と解析器なのである。このようにCommonMarkは新構文だけまたは解析器だけ文脈依存に変更しても理論的齟齬が解析結果と使用感の不自然さと違和感に明白に表れるためCommonMarkが失敗した言語である事実は到底隠し切れるものではない。Markdownはもはや負債以外の何物でもないCommonMarkの異常な解析規則を捨てて素直な文脈依存構文言語として新しい仕様を作り直すのが賢明である。
299
299
  文脈依存言語を再帰的バックトラックなしで解析可能な方法は2つある。一つは解釈を予言的に決定する方法、もう一つは予備的または中間的その他事前の解析結果に対する解釈を事後に決定する方法である。前者は構文を峻別するための制約が強く後者は共通構造の適切な設計が必要でありいずれも再帰的バックトラックを防ぐために未終端の構文も受理する必要がある。しかし予言的解析は類似の異文脈構文へのバックトラックが不可能になるため共通構造についての事前解析結果のような追加情報を利用できるのでなければ類似の異文脈構文の定義が不可能になる。また共通構造をバックトラックなしで解析するにはそのための言語設計が必要であることに変わりなく任意の文字列を使用可能である必要があるURL、コード、数式などに含まれる任意の文字列が他の構文と文脈で解釈される際に共通構造を破壊するためこのような文字列を構造的に決定的に解釈する基本構造(バックトラック可能だが入力に対する解析結果がすべての文脈で構造的に共通すなわち決定的でなければならない。構文が解析されない場合も構文全体が包含されれば問題ない)を共通構造に定義しなければ構造的共通化は不可能であり分離構造は終端不要な開放可能構文としての予言的構文でなければバックトラックが不可避となる。共通構造の事前解析は一見すると最悪計算量を2nに削減できるように見えるが実際には共通構造をバックトラックなしで解析することが元の言語より容易にはなっても依然として難しくバックトラックなしで解析できたとしてもすべての解析ひいては平均計算量が+1nされるため平均的にはむしろ計算量を悪化させる方法であり仮に最悪計算量32nを事前解析により2nに抑えられたとしても平均計算量が1nならば平均計算量が2nに倍加し実行性能が半減および実行コストが倍加するため明らかに事前解析を行うべきでない。実用的なアルゴリズムの計算量において重要なのは最悪計算量がO(n^3)相当以下の実用的な計算量であるかと平均計算量でありその他は重要ではない。メモ化が空間計算量を代償に悪化させる最悪時間計算量線形化手法であるのに対して事前解析は平均時間計算量を代償に悪化させる最悪時間計算量線形化手法でありマーキング法は何も代償に悪化させない最悪時間計算量線形化手法である(無論ゼロコストではないが解析の他の部分やASTに要するコストより小さいため無視できる。空間計算量は実際にはASTにより容易に数倍になりメモリ使用量は入力サイズと並列数に制約されるため空間計算量自体はさほど重要ではなく空間計算量を増加させる処理により生じる時間オーバーヘッドのほうが本質的に重要となる。メモ化が常時記録かつ実データ記録ゆえデータ数もデータサイズも大きいのに対してマーキング法は失敗時記録かつフラグのみ記録ゆえデータ数もデータサイズも小さいことでオーバーヘッドが最小化されバックトラック率の低い入力が大多数を占める場合の平均実行速度が高速化される。平均性能すなわち重要指標が向上するならばその他の指標は許容範囲内で悪化させることができ不要な指標値を必要な指標値に変換できるならばそうしなければならない)。なおマーキング法を使用するSecuremarkは本来の構文解析と共通構造解析を統合し同時解析することで計算量の増加なく共通構造解析を行い共通構造を利用して最悪時間計算量を指数計算量から5nへ削減している。
package/dist/index.js CHANGED
@@ -1,4 +1,4 @@
1
- /*! securemark v0.297.0 https://github.com/falsandtru/securemark | (c) 2017, falsandtru | UNLICENSED License */
1
+ /*! securemark v0.297.2 https://github.com/falsandtru/securemark | (c) 2017, falsandtru | UNLICENSED License */
2
2
  (function webpackUniversalModuleDefinition(root, factory) {
3
3
  if(typeof exports === 'object' && typeof module === 'object')
4
4
  module.exports = factory(require("Prism"), require("DOMPurify"));
@@ -4302,6 +4302,12 @@ function bind(target, settings) {
4302
4302
  type: 'cancel'
4303
4303
  };
4304
4304
  }
4305
+ yield {
4306
+ type: 'break'
4307
+ };
4308
+ if (rev !== revision) return yield {
4309
+ type: 'cancel'
4310
+ };
4305
4311
  for (const el of (0, figure_1.figure)(next(0)?.parentNode ?? target, settings.notes, context)) {
4306
4312
  el ? yield {
4307
4313
  type: 'figure',
@@ -6323,7 +6329,7 @@ const bracket_1 = __webpack_require__(4526);
6323
6329
  const visibility_1 = __webpack_require__(6364);
6324
6330
  const util_1 = __webpack_require__(4992);
6325
6331
  const dom_1 = __webpack_require__(394);
6326
- exports.annotation = (0, combinator_1.lazy)(() => (0, combinator_1.constraint)(128 /* State.annotation */, (0, combinator_1.surround)((0, combinator_1.close)('((', visibility_1.beforeNonblank), (0, combinator_1.precedence)(1, (0, combinator_1.recursion)(4 /* Recursion.inline */, (0, combinator_1.recursion)(5 /* Recursion.bracket */, (0, combinator_1.recursion)(5 /* Recursion.bracket */, (0, combinator_1.some)((0, combinator_1.union)([inline_1.inline]), ')', [[')', 1]]))))), '))', false, [], ([, ns], context) => {
6332
+ exports.annotation = (0, combinator_1.lazy)(() => (0, combinator_1.constraint)(128 /* State.annotation */, (0, combinator_1.surround)((0, combinator_1.open)('((', visibility_1.beforeNonblank), (0, combinator_1.precedence)(1, (0, combinator_1.recursion)(4 /* Recursion.inline */, (0, combinator_1.recursion)(5 /* Recursion.bracket */, (0, combinator_1.recursion)(5 /* Recursion.bracket */, (0, combinator_1.some)((0, combinator_1.union)([inline_1.inline]), ')', [[')', 1]]))))), '))', false, [], ([, ns], context) => {
6327
6333
  const {
6328
6334
  linebreak
6329
6335
  } = context;
@@ -7123,22 +7129,8 @@ function baseR(n, r) {
7123
7129
  return acc;
7124
7130
  }
7125
7131
  function signature(target) {
7126
- for (let es = target.querySelectorAll('code[data-src], .math[data-src], .remark, rt, rp, br, .annotation, .reference, .checkbox, ul, ol, .label[data-label]'), len = es.length, i = 0; i < len; ++i) {
7132
+ for (let es = target.querySelectorAll('code[data-src], .math[data-src], .remark, rt, rp, br, .annotation, .reference, :is(.annotation, .reference) > a, .checkbox, ul, ol, .label[data-label]'), len = es.length, i = 0; i < len; ++i) {
7127
7133
  const el = es[i];
7128
- switch (el.tagName) {
7129
- case 'CODE':
7130
- el.replaceWith(el.getAttribute('data-src'));
7131
- continue;
7132
- case 'RT':
7133
- case 'RP':
7134
- case 'UL':
7135
- case 'OL':
7136
- el.remove();
7137
- continue;
7138
- case 'BR':
7139
- el.replaceWith('\n');
7140
- continue;
7141
- }
7142
7134
  switch (el.className) {
7143
7135
  case 'math':
7144
7136
  el.replaceWith(el.getAttribute('data-src'));
@@ -7153,25 +7145,28 @@ function signature(target) {
7153
7145
  el.remove();
7154
7146
  continue;
7155
7147
  }
7156
- }
7157
- return target.textContent.trim();
7158
- }
7159
- exports.signature = signature;
7160
- function text(target) {
7161
- for (let es = target.querySelectorAll('code[data-src], .math[data-src], .remark, rt, rp, br, .annotation, .reference, .checkbox, ul, ol'), len = es.length, i = 0; i < len; ++i) {
7162
- const el = es[i];
7163
7148
  switch (el.tagName) {
7164
7149
  case 'CODE':
7165
7150
  el.replaceWith(el.getAttribute('data-src'));
7166
7151
  continue;
7167
- case 'RT':
7168
- case 'RP':
7169
- case 'BR':
7170
7152
  case 'UL':
7171
7153
  case 'OL':
7154
+ case 'RT':
7155
+ case 'RP':
7156
+ case 'A':
7172
7157
  el.remove();
7173
7158
  continue;
7159
+ case 'BR':
7160
+ el.replaceWith('\n');
7161
+ continue;
7174
7162
  }
7163
+ }
7164
+ return target.textContent.trim();
7165
+ }
7166
+ exports.signature = signature;
7167
+ function text(target) {
7168
+ for (let es = target.querySelectorAll('code[data-src], .math[data-src], .remark, rt, rp, br, .annotation, .reference, :is(.annotation, .reference) > a, .checkbox, ul, ol'), len = es.length, i = 0; i < len; ++i) {
7169
+ const el = es[i];
7175
7170
  switch (el.className) {
7176
7171
  case 'math':
7177
7172
  el.replaceWith(el.getAttribute('data-src'));
@@ -7183,6 +7178,19 @@ function text(target) {
7183
7178
  el.remove();
7184
7179
  continue;
7185
7180
  }
7181
+ switch (el.tagName) {
7182
+ case 'CODE':
7183
+ el.replaceWith(el.getAttribute('data-src'));
7184
+ continue;
7185
+ case 'UL':
7186
+ case 'OL':
7187
+ case 'RT':
7188
+ case 'RP':
7189
+ case 'A':
7190
+ case 'BR':
7191
+ el.remove();
7192
+ continue;
7193
+ }
7186
7194
  }
7187
7195
  return target.textContent;
7188
7196
  }
@@ -7484,7 +7492,7 @@ const optspec = {
7484
7492
  rel: ['nofollow']
7485
7493
  };
7486
7494
  Object.setPrototypeOf(optspec, null);
7487
- exports.textlink = (0, combinator_1.lazy)(() => (0, combinator_1.bind)((0, combinator_1.subsequence)([(0, combinator_1.constraint)(8 /* State.link */, (0, combinator_1.state)(251 /* State.linkers */, (0, combinator_1.dup)((0, combinator_1.surround)((0, combinator_1.close)('[', visibility_1.beforeNonblank), (0, combinator_1.precedence)(1, (0, combinator_1.some)((0, combinator_1.union)([inline_1.inline]), ']', [[']', 1]])), ']', true, [3 | 4 /* Backtrack.common */ | 64 /* Backtrack.link */, 2 | 32 /* Backtrack.ruby */], ([, ns = new parser_1.List()], context) => {
7495
+ exports.textlink = (0, combinator_1.lazy)(() => (0, combinator_1.bind)((0, combinator_1.subsequence)([(0, combinator_1.constraint)(8 /* State.link */, (0, combinator_1.state)(251 /* State.linkers */, (0, combinator_1.dup)((0, combinator_1.surround)((0, combinator_1.open)('[', visibility_1.beforeNonblank), (0, combinator_1.precedence)(1, (0, combinator_1.some)((0, combinator_1.union)([inline_1.inline]), ']', [[']', 1]])), ']', true, [3 | 4 /* Backtrack.common */ | 64 /* Backtrack.link */, 2 | 32 /* Backtrack.ruby */], ([, ns = new parser_1.List()], context) => {
7488
7496
  if (context.linebreak !== 0) {
7489
7497
  const head = context.position - context.range;
7490
7498
  return void (0, combinator_1.setBacktrack)(context, 2 | 64 /* Backtrack.link */ | 32 /* Backtrack.ruby */, head);
@@ -8273,19 +8281,19 @@ function* note(target, notes, opts = {}, bottom = null) {
8273
8281
  yield* (0, exports.reference)(target, notes?.references, opts, bottom);
8274
8282
  }
8275
8283
  exports.note = note;
8276
- exports.annotation = build('annotation', 'sup.annotation:not(.annotations .annotation)', n => `*${n}`, 'h1, h2, h3, h4, h5, h6, aside.aside, hr');
8277
- exports.reference = build('reference', 'sup.reference:not(.references .reference)', (n, abbr) => `[${abbr || n}]`);
8284
+ exports.annotation = build('annotation', 'annotations', n => `*${n}`, 'h1, h2, h3, h4, h5, h6, aside.aside, hr');
8285
+ exports.reference = build('reference', 'references', (n, abbr) => `[${abbr || n}]`);
8278
8286
  // Referenceを含むAnnotationの重複排除は両構文が互いに処理済みであることを必要とするため
8279
8287
  // 構文ごとに各1回の処理では不可能
8280
- function build(syntax, query, marker, splitter = '') {
8281
- splitter &&= `${splitter}, .${syntax}s`;
8288
+ function build(syntax, plural, marker, splitter = '') {
8289
+ splitter &&= `${splitter}, .${plural}`;
8282
8290
  const refMemoryCaller = (0, memoize_1.memoize)(target => new Map() ?? target, new WeakMap());
8283
8291
  return function* (target, note, opts = {}, bottom = null) {
8284
8292
  const refMemory = refMemoryCaller(target);
8285
8293
  const refInfoCaller = (0, memoize_1.memoize)(ref => {
8286
8294
  const content = ref.firstElementChild;
8287
8295
  const abbr = ref.getAttribute('data-abbr') ?? '';
8288
- const clone = content.cloneNode(true);
8296
+ const clone = ref.cloneNode(true);
8289
8297
  const txt = (0, indexee_1.text)(clone).trim();
8290
8298
  const identifier = abbr ? (0, indexee_1.identity)('', undefined, abbr.match(/^(?:\S+ )+?(?:(?:January|February|March|April|May|June|August|September|October|November|December) \d{1,2}(?:-\d{0,2})?, \d{1,4}(?:-\d{0,4})?[a-z]?|n\.d\.)(?=,|$)/)?.[0] ?? abbr.match(/^[^,\s]+(?:,? [^,\s]+)*?(?: \d{1,4}(?:-\d{0,4})?[a-z]?(?=,|$)|(?=,(?: [a-z]+\.?)? [0-9]))/)?.[0] ?? abbr)?.slice(2) || '' : (0, indexee_1.identity)('mark', undefined, (0, indexee_1.signature)(clone))?.slice(6) || '';
8291
8299
  return {
@@ -8298,12 +8306,11 @@ function build(syntax, query, marker, splitter = '') {
8298
8306
  for (const [ref, {
8299
8307
  content
8300
8308
  }] of refMemory) {
8301
- content.replaceWith(content.cloneNode(true));
8302
- ref.prepend(content);
8303
- refMemory.delete(ref);
8309
+ ref.replaceChildren(content);
8304
8310
  }
8311
+ refMemory.clear();
8305
8312
  const defs = new Map();
8306
- const refs = target.querySelectorAll(`${query}:not(.disabled)`);
8313
+ const refs = target.querySelectorAll(`.${syntax}:not(.disabled)`);
8307
8314
  const identifierInfoCaller = (0, memoize_1.memoize)(identifier => ({
8308
8315
  defIndex: 0,
8309
8316
  defSubindex: 0,
@@ -8329,7 +8336,7 @@ function build(syntax, query, marker, splitter = '') {
8329
8336
  if (defs.size > 0) {
8330
8337
  total += defs.size;
8331
8338
  const note = el.tagName === 'OL' ? el : target.insertBefore((0, dom_1.html)('ol', {
8332
- class: `${syntax}s`
8339
+ class: plural
8333
8340
  }), el);
8334
8341
  yield* proc(defs, note);
8335
8342
  }
@@ -8394,7 +8401,7 @@ function build(syntax, query, marker, splitter = '') {
8394
8401
  if (note || defs.size > 0) {
8395
8402
  const el = splitters[iSplitters];
8396
8403
  note ??= el?.tagName === 'OL' && el.nextElementSibling == splitters[iSplitters + 1] ? (++iSplitters, el) : target.insertBefore((0, dom_1.html)('ol', {
8397
- class: `${syntax}s`
8404
+ class: plural
8398
8405
  }), splitters[iSplitters] ?? bottom);
8399
8406
  yield* proc(defs, note);
8400
8407
  }
@@ -8411,30 +8418,13 @@ function* proc(defs, note) {
8411
8418
  const {
8412
8419
  children
8413
8420
  } = note;
8414
- const size = defs.size;
8415
- let count = 0;
8416
- let length = children.length;
8417
- I: for (const [key, def] of defs) {
8418
- defs.delete(key);
8419
- ++count;
8420
- for (; length > size;) {
8421
- const node = children[count - 1];
8422
- if (equal(node, def)) continue I;
8423
- yield note.removeChild(node);
8424
- --length;
8425
- }
8426
- const node = count <= length ? children[count - 1] : null;
8427
- if (node && equal(node, def)) continue;
8428
- yield note.insertBefore(def, node);
8429
- ++length;
8430
- }
8431
- for (; length > size;) {
8432
- yield note.removeChild(children[size]);
8433
- --length;
8421
+ for (let defs = note.children, i = defs.length; i--;) {
8422
+ yield note.removeChild(children[i]);
8434
8423
  }
8435
- }
8436
- function equal(a, b) {
8437
- return a.id === b.id && a.innerHTML === b.innerHTML;
8424
+ for (const [, def] of defs) {
8425
+ yield note.appendChild(def);
8426
+ }
8427
+ defs.clear();
8438
8428
  }
8439
8429
 
8440
8430
  /***/ },
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "securemark",
3
- "version": "0.297.0",
3
+ "version": "0.297.2",
4
4
  "description": "Secure markdown renderer working on browsers for user input data.",
5
5
  "private": false,
6
6
  "homepage": "https://github.com/falsandtru/securemark",
@@ -124,6 +124,8 @@ export function bind(target: DocumentFragment | HTMLElement | ShadowRoot, settin
124
124
  yield { type: 'block', value: el };
125
125
  if (rev !== revision) return yield { type: 'cancel' };
126
126
  }
127
+ yield { type: 'break' };
128
+ if (rev !== revision) return yield { type: 'cancel' };
127
129
  for (const el of figure(next(0)?.parentNode ?? target, settings.notes, context)) {
128
130
  assert(rev === revision);
129
131
  el
@@ -12,7 +12,7 @@ describe('Unit: parser/inline/annotation', () => {
12
12
  assert.deepStrictEqual(inspect(parser, input('', new Context())), undefined);
13
13
  assert.deepStrictEqual(inspect(parser, input('(', new Context())), undefined);
14
14
  assert.deepStrictEqual(inspect(parser, input('()', new Context())), undefined);
15
- assert.deepStrictEqual(inspect(parser, input('((', new Context())), [['<span class="paren">(<span class="paren">(</span></span>'], '']);
15
+ assert.deepStrictEqual(inspect(parser, input('((', new Context())), undefined);
16
16
  assert.deepStrictEqual(inspect(parser, input('(())', new Context())), [['<span class="paren">(<span class="paren">(</span></span>'], '))']);
17
17
  assert.deepStrictEqual(inspect(parser, input('(()))', new Context())), [['<span class="paren">(<span class="paren">(</span></span>'], ')))']);
18
18
  assert.deepStrictEqual(inspect(parser, input('(("))', new Context())), [['<span class="paren">(<span class="paren">("))</span></span>'], '']);
@@ -1,7 +1,7 @@
1
1
  import { AnnotationParser } from '../inline';
2
2
  import { State, Recursion } from '../context';
3
3
  import { List, Node } from '../../combinator/data/parser';
4
- import { union, some, recursion, precedence, constraint, surround, close, lazy } from '../../combinator';
4
+ import { union, some, recursion, precedence, constraint, surround, open, lazy } from '../../combinator';
5
5
  import { inline } from '../inline';
6
6
  import { indexA } from './bracket';
7
7
  import { beforeNonblank, trimBlankNodeEnd } from '../visibility';
@@ -9,7 +9,7 @@ import { unwrap } from '../util';
9
9
  import { html, defrag } from 'typed-dom/dom';
10
10
 
11
11
  export const annotation: AnnotationParser = lazy(() => constraint(State.annotation, surround(
12
- close('((', beforeNonblank),
12
+ open('((', beforeNonblank),
13
13
  precedence(1, recursion(Recursion.inline, recursion(Recursion.bracket, recursion(Recursion.bracket,
14
14
  some(union([inline]), ')', [[')', 1]]))))),
15
15
  '))',
@@ -109,23 +109,9 @@ assert(baseR(62, 62) === '10');
109
109
  export function signature(target: Element | DocumentFragment): string {
110
110
  assert(!target.parentNode);
111
111
  assert(!target.querySelector('br:not(:has(+ :is(ul, ol)))') || target.nodeName === 'MARK');
112
- for (let es = target.querySelectorAll('code[data-src], .math[data-src], .remark, rt, rp, br, .annotation, .reference, .checkbox, ul, ol, .label[data-label]'),
112
+ for (let es = target.querySelectorAll('code[data-src], .math[data-src], .remark, rt, rp, br, .annotation, .reference, :is(.annotation, .reference) > a, .checkbox, ul, ol, .label[data-label]'),
113
113
  len = es.length, i = 0; i < len; ++i) {
114
114
  const el = es[i];
115
- switch (el.tagName) {
116
- case 'CODE':
117
- el.replaceWith(el.getAttribute('data-src')!);
118
- continue;
119
- case 'RT':
120
- case 'RP':
121
- case 'UL':
122
- case 'OL':
123
- el.remove();
124
- continue;
125
- case 'BR':
126
- el.replaceWith('\n');
127
- continue;
128
- }
129
115
  switch (el.className) {
130
116
  case 'math':
131
117
  el.replaceWith(el.getAttribute('data-src')!);
@@ -140,28 +126,31 @@ export function signature(target: Element | DocumentFragment): string {
140
126
  el.remove();
141
127
  continue;
142
128
  }
143
- }
144
- return target.textContent!.trim();
145
- }
146
-
147
- export function text(target: Element | DocumentFragment): string {
148
- assert(!target.parentNode);
149
- assert(!target.querySelector('br:not(:has(+ :is(ul, ol)))'));
150
- for (let es = target.querySelectorAll('code[data-src], .math[data-src], .remark, rt, rp, br, .annotation, .reference, .checkbox, ul, ol'),
151
- len = es.length, i = 0; i < len; ++i) {
152
- const el = es[i];
153
129
  switch (el.tagName) {
154
130
  case 'CODE':
155
131
  el.replaceWith(el.getAttribute('data-src')!);
156
132
  continue;
157
- case 'RT':
158
- case 'RP':
159
- case 'BR':
160
133
  case 'UL':
161
134
  case 'OL':
135
+ case 'RT':
136
+ case 'RP':
137
+ case 'A':
162
138
  el.remove();
163
139
  continue;
140
+ case 'BR':
141
+ el.replaceWith('\n');
142
+ continue;
164
143
  }
144
+ }
145
+ return target.textContent!.trim();
146
+ }
147
+
148
+ export function text(target: Element | DocumentFragment): string {
149
+ assert(!target.parentNode);
150
+ assert(!target.querySelector('br:not(:has(+ :is(ul, ol)))'));
151
+ for (let es = target.querySelectorAll('code[data-src], .math[data-src], .remark, rt, rp, br, .annotation, .reference, :is(.annotation, .reference) > a, .checkbox, ul, ol'),
152
+ len = es.length, i = 0; i < len; ++i) {
153
+ const el = es[i];
165
154
  switch (el.className) {
166
155
  case 'math':
167
156
  el.replaceWith(el.getAttribute('data-src')!);
@@ -173,6 +162,19 @@ export function text(target: Element | DocumentFragment): string {
173
162
  el.remove();
174
163
  continue;
175
164
  }
165
+ switch (el.tagName) {
166
+ case 'CODE':
167
+ el.replaceWith(el.getAttribute('data-src')!);
168
+ continue;
169
+ case 'UL':
170
+ case 'OL':
171
+ case 'RT':
172
+ case 'RP':
173
+ case 'A':
174
+ case 'BR':
175
+ el.remove();
176
+ continue;
177
+ }
176
178
  }
177
179
  return target.textContent!;
178
180
  }
@@ -1,7 +1,7 @@
1
1
  import { LinkParser } from '../inline';
2
2
  import { Context, State, Backtrack, Command } from '../context';
3
3
  import { List, Node } from '../../combinator/data/parser';
4
- import { union, inits, sequence, subsequence, some, consume, precedence, state, constraint, surround, open, close, setBacktrack, dup, lazy, fmap, bind } from '../../combinator';
4
+ import { union, inits, sequence, subsequence, some, consume, precedence, state, constraint, surround, open, setBacktrack, dup, lazy, fmap, bind } from '../../combinator';
5
5
  import { inline, media, shortmedia } from '../inline';
6
6
  import { attributes } from './html';
7
7
  import { str } from '../source';
@@ -18,7 +18,7 @@ Object.setPrototypeOf(optspec, null);
18
18
  export const textlink: LinkParser.TextLinkParser = lazy(() => bind(
19
19
  subsequence([
20
20
  constraint(State.link, state(State.linkers, dup(surround(
21
- close('[', beforeNonblank),
21
+ open('[', beforeNonblank),
22
22
  precedence(1,
23
23
  some(union([inline]), ']', [[']', 1]])),
24
24
  ']',
@@ -17,7 +17,7 @@ describe('Unit: parser/processor/note', () => {
17
17
  it('1', () => {
18
18
  const target = parse('((a b))');
19
19
  for (let i = 0; i < 3; ++i) {
20
- assert.deepStrictEqual([...annotation(target)].length, i === 0 ? 2 : 1);
20
+ assert.deepStrictEqual([...annotation(target)].length, i === 0 ? 2 : 3);
21
21
  assert.deepStrictEqual(
22
22
  [...target.children].map(el => el.outerHTML),
23
23
  [
@@ -39,7 +39,7 @@ describe('Unit: parser/processor/note', () => {
39
39
  it('2', () => {
40
40
  const target = parse('((1))((12345678901234567890))');
41
41
  for (let i = 0; i < 3; ++i) {
42
- assert.deepStrictEqual([...annotation(target)].length, i === 0 ? 4 : 2);
42
+ assert.deepStrictEqual([...annotation(target)].length, i === 0 ? 4 : 6);
43
43
  assert.deepStrictEqual(
44
44
  [...target.children].map(el => el.outerHTML),
45
45
  [
@@ -202,7 +202,7 @@ describe('Unit: parser/processor/note', () => {
202
202
  it('id', () => {
203
203
  const target = parse('((a b))');
204
204
  for (let i = 0; i < 3; ++i) {
205
- assert.deepStrictEqual([...annotation(target, undefined, { id: '0' })].length, i === 0 ? 2 : 1);
205
+ assert.deepStrictEqual([...annotation(target, undefined, { id: '0' })].length, i === 0 ? 2 : 3);
206
206
  assert.deepStrictEqual(
207
207
  [...target.children].map(el => el.outerHTML),
208
208
  [
@@ -222,7 +222,7 @@ describe('Unit: parser/processor/note', () => {
222
222
  });
223
223
 
224
224
  it('nest', () => {
225
- const target = parse('((a((b))))((a))((b))');
225
+ const target = parse('((a((b((c))))))((a))((b))');
226
226
  for (let i = 0; i < 3; ++i) {
227
227
  [...annotation(target)];
228
228
  assert.deepStrictEqual(
@@ -249,21 +249,31 @@ describe('Unit: parser/processor/note', () => {
249
249
  ]),
250
250
  html('sup', [
251
251
  html('a', { href: '#annotation::ref:a:1' }, '^1'),
252
- html('a', { href: '#annotation::ref:a:2' }, '^3'),
252
+ html('a', { href: '#annotation::ref:a:2' }, '^4'),
253
253
  ])
254
254
  ]),
255
255
  html('li', { id: 'annotation::def:b:1', 'data-marker': '*2' }, [
256
256
  html('span', [
257
257
  'b',
258
+ html('sup', { class: 'annotation', id: 'annotation::ref:c:1', title: 'c' }, [
259
+ html('a', { href: '#annotation::def:c:1' }, '*3')
260
+ ]),
258
261
  ]),
259
262
  html('sup', [
260
263
  html('a', { href: '#annotation::ref:b:1' }, '^2'),
261
- html('a', { href: '#annotation::ref:b:2' }, '^4'),
264
+ html('a', { href: '#annotation::ref:b:2' }, '^5'),
265
+ ])
266
+ ]),
267
+ html('li', { id: 'annotation::def:c:1', 'data-marker': '*3' }, [
268
+ html('span', [
269
+ 'c',
270
+ ]),
271
+ html('sup', [
272
+ html('a', { href: '#annotation::ref:c:1' }, '^3'),
262
273
  ])
263
274
  ]),
264
275
  ]).outerHTML,
265
276
  ]);
266
- target.lastChild?.remove();
267
277
  }
268
278
  });
269
279
 
@@ -376,7 +386,6 @@ describe('Unit: parser/processor/note', () => {
376
386
  ]),
377
387
  ]).outerHTML,
378
388
  ]);
379
- target.lastChild?.remove();
380
389
  }
381
390
  });
382
391
 
@@ -18,24 +18,24 @@ export function* note(
18
18
 
19
19
  export const annotation = build(
20
20
  'annotation',
21
- 'sup.annotation:not(.annotations .annotation)',
21
+ 'annotations',
22
22
  n => `*${n}`,
23
23
  'h1, h2, h3, h4, h5, h6, aside.aside, hr');
24
24
  export const reference = build(
25
25
  'reference',
26
- 'sup.reference:not(.references .reference)',
26
+ 'references',
27
27
  (n, abbr) => `[${abbr || n}]`);
28
28
 
29
29
  // Referenceを含むAnnotationの重複排除は両構文が互いに処理済みであることを必要とするため
30
30
  // 構文ごとに各1回の処理では不可能
31
31
  function build(
32
- syntax: 'annotation' | 'reference',
33
- query: `${keyof HTMLElementTagNameMap}.${string}`,
32
+ syntax: string,
33
+ plural: string,
34
34
  marker: (index: number, abbr: string) => string,
35
35
  splitter: string = '',
36
36
  ) {
37
37
  assert(syntax.match(/^[a-z]+$/));
38
- splitter &&= `${splitter}, .${syntax}s`;
38
+ splitter &&= `${splitter}, .${plural}`;
39
39
  const refMemoryCaller = memoize((target: Node) =>
40
40
  new Map<HTMLElement, {
41
41
  readonly content: Element;
@@ -54,7 +54,7 @@ function build(
54
54
  const refInfoCaller = memoize((ref: HTMLElement) => {
55
55
  const content = ref.firstElementChild!;
56
56
  const abbr = ref.getAttribute('data-abbr') ?? '';
57
- const clone = content.cloneNode(true);
57
+ const clone = ref.cloneNode(true);
58
58
  const txt = text(clone).trim();
59
59
  const identifier = abbr
60
60
  ? identity(
@@ -74,13 +74,11 @@ function build(
74
74
  };
75
75
  }, refMemory);
76
76
  for (const [ref, { content }] of refMemory) {
77
- content.replaceWith(content.cloneNode(true));
78
- ref.prepend(content);
79
- refMemory.delete(ref);
77
+ ref.replaceChildren(content);
80
78
  }
81
- assert(refMemory.size === 0);
79
+ refMemory.clear();
82
80
  const defs = new Map<string, HTMLLIElement>();
83
- const refs = target.querySelectorAll(`${query}:not(.disabled)`);
81
+ const refs = target.querySelectorAll<HTMLElement>(`.${syntax}:not(.disabled)`);
84
82
  const identifierInfoCaller = memoize((identifier: string) => ({
85
83
  defIndex: 0,
86
84
  defSubindex: 0,
@@ -104,7 +102,7 @@ function build(
104
102
  if (~iSplitters << 32 - 8 === 0) yield;
105
103
  if (!scope && el.parentNode !== target) continue;
106
104
  if (el.tagName === 'OL' && (el.nextElementSibling !== splitters[iSplitters + 1] || defs.size === 0)) {
107
- assert(el.matches(`.${syntax}s`));
105
+ assert(el.matches(`.${plural}`));
108
106
  el.remove();
109
107
  continue;
110
108
  }
@@ -112,7 +110,7 @@ function build(
112
110
  total += defs.size;
113
111
  const note = el.tagName === 'OL'
114
112
  ? el as HTMLOListElement
115
- : target.insertBefore(html('ol', { class: `${syntax}s` }), el);
113
+ : target.insertBefore(html('ol', { class: plural }), el);
116
114
  assert(note.parentNode);
117
115
  yield* proc(defs, note);
118
116
  assert(defs.size === 0);
@@ -188,7 +186,7 @@ function build(
188
186
  const el = splitters[iSplitters];
189
187
  note ??= el?.tagName === 'OL' && el.nextElementSibling == splitters[iSplitters + 1]
190
188
  ? (++iSplitters, el as HTMLOListElement)
191
- : target.insertBefore(html('ol', { class: `${syntax}s` }), splitters[iSplitters] ?? bottom);
189
+ : target.insertBefore(html('ol', { class: plural }), splitters[iSplitters] ?? bottom);
192
190
  yield* proc(defs, note);
193
191
  assert(defs.size === 0);
194
192
  }
@@ -196,7 +194,7 @@ function build(
196
194
  if (~iSplitters << 32 - 8 === 0) yield;
197
195
  if (!scope && el.parentNode !== target) continue;
198
196
  if (el.tagName === 'OL') {
199
- assert(el.matches(`.${syntax}s`));
197
+ assert(el.matches(`.${plural}`));
200
198
  el.remove();
201
199
  }
202
200
  }
@@ -205,37 +203,11 @@ function build(
205
203
 
206
204
  function* proc(defs: Map<string, HTMLLIElement>, note: HTMLOListElement): Generator<HTMLLIElement | undefined, undefined, undefined> {
207
205
  const { children } = note;
208
- const size = defs.size;
209
- let count = 0;
210
- let length = children.length;
211
- I:
212
- for (const [key, def] of defs) {
213
- defs.delete(key);
214
- ++count;
215
- for (; length > size;) {
216
- const node = children[count - 1] as HTMLLIElement;
217
- if (equal(node, def)) continue I;
218
- yield note.removeChild(node);
219
- --length;
220
- assert(children.length === length);
221
- }
222
- const node = count <= length
223
- ? children[count - 1]
224
- : null;
225
- if (node && equal(node, def)) continue;
226
- assert(def.parentNode !== note);
227
- yield note.insertBefore(def, node);
228
- ++length;
229
- assert(children.length === length);
206
+ for (let defs = note.children, i = defs.length; i--;) {
207
+ yield note.removeChild(children[i] as HTMLLIElement);
230
208
  }
231
- for (; length > size;) {
232
- yield note.removeChild(children[size] as HTMLLIElement);
233
- --length;
234
- assert(children.length === length);
209
+ for (const [, def] of defs) {
210
+ yield note.appendChild(def);
235
211
  }
236
- }
237
-
238
- function equal(a: Element, b: HTMLElement): boolean {
239
- return a.id === b.id
240
- && a.innerHTML === b.innerHTML;
212
+ defs.clear();
241
213
  }