From 76562f425002286c94b2dc2c7bba9e413e451471 Mon Sep 17 00:00:00 2001 From: sabaimran <65192171+sabaimran@users.noreply.github.com> Date: Wed, 6 Sep 2023 12:04:18 -0700 Subject: [PATCH] Add front-end Electron application for Khoj local file syncing (#473) * Initial version - setup a file-push architecture for generating embeddings with Khoj * Use state.host and state.port for configuring the URL for the indexer * Fix parsing of PDF files * Read markdown files from streamed data and update unit tests * On application startup, load in embeddings from configurations files, rather than regenerating the corpus based on file system * Init: refactor indexer/batch endpoint to support a generic file ingestion format * Add features to better support indexing from files sent by the desktop client * Initial commit with Electron application - Adds electron app * Add import for pymupdf, remove import for pypdf * Allow user to configure khoj host URL * Remove search type configuration from index.html * Use v1 path for current indexer routes --- .gitignore | 3 +- pyproject.toml | 2 +- src/interface/desktop/README.md | 29 + src/interface/desktop/assets/icons/chat.svg | 693 ++ .../desktop/assets/icons/circular-add.svg | 5 + .../desktop/assets/icons/confirm-icon.svg | 1 + .../desktop/assets/icons/favicon-128x128.ico | Bin 0 -> 205167 bytes .../desktop/assets/icons/favicon-128x128.png | Bin 0 -> 12518 bytes .../desktop/assets/icons/favicon.icns | Bin 0 -> 31531 bytes src/interface/desktop/assets/icons/folder.svg | 28 + src/interface/desktop/assets/icons/github.svg | 1 + .../assets/icons/khoj-logo-sideways-200.png | Bin 0 -> 13011 bytes .../assets/icons/khoj-logo-sideways-500.png | Bin 0 -> 36306 bytes .../assets/icons/khoj-logo-sideways.svg | 5385 +++++++++++ src/interface/desktop/assets/icons/link.svg | 5 + .../desktop/assets/icons/logotype.svg | 1 + .../desktop/assets/icons/markdown.svg | 1 + src/interface/desktop/assets/icons/notion.svg | 4 + .../desktop/assets/icons/openai-logomark.svg | 1 + src/interface/desktop/assets/icons/org.svg | 1 + src/interface/desktop/assets/icons/pdf.svg | 23 + .../desktop/assets/icons/plaintext.svg | 1 + .../assets/icons/question-mark-icon.svg | 1 + src/interface/desktop/assets/khoj.css | 116 + .../desktop/assets/markdown-it.min.js | 8476 +++++++++++++++++ src/interface/desktop/assets/org.min.js | 1823 ++++ src/interface/desktop/assets/pico.min.css | 5 + src/interface/desktop/chat.html | 559 ++ src/interface/desktop/config.html | 342 + src/interface/desktop/index.html | 499 + src/interface/desktop/main.js | 359 + src/interface/desktop/package.json | 25 + src/interface/desktop/preload.js | 49 + src/interface/desktop/renderer.js | 177 + src/interface/desktop/todesktop.json | 6 + src/interface/desktop/yarn.lock | 1205 +++ src/khoj/configure.py | 2 +- src/khoj/processor/github/github_to_jsonl.py | 2 +- src/khoj/processor/jsonl/jsonl_to_jsonl.py | 2 +- .../processor/markdown/markdown_to_jsonl.py | 11 +- src/khoj/processor/notion/notion_to_jsonl.py | 2 +- src/khoj/processor/org_mode/org_to_jsonl.py | 13 +- src/khoj/processor/pdf/pdf_to_jsonl.py | 26 +- .../processor/plaintext/plaintext_to_jsonl.py | 13 +- src/khoj/processor/text_to_jsonl.py | 34 +- src/khoj/routers/indexer.py | 155 +- src/khoj/search_type/text_search.py | 9 +- src/khoj/utils/fs_syncer.py | 3 +- src/khoj/utils/helpers.py | 16 + src/khoj/utils/rawconfig.py | 18 - tests/test_client.py | 6 +- tests/test_pdf_to_jsonl.py | 5 +- tests/test_rawconfig.py | 12 - tests/test_text_search.py | 59 + 54 files changed, 20132 insertions(+), 82 deletions(-) create mode 100644 src/interface/desktop/README.md create mode 100644 src/interface/desktop/assets/icons/chat.svg create mode 100644 src/interface/desktop/assets/icons/circular-add.svg create mode 100644 src/interface/desktop/assets/icons/confirm-icon.svg create mode 100644 src/interface/desktop/assets/icons/favicon-128x128.ico create mode 100644 src/interface/desktop/assets/icons/favicon-128x128.png create mode 100644 src/interface/desktop/assets/icons/favicon.icns create mode 100644 src/interface/desktop/assets/icons/folder.svg create mode 100644 src/interface/desktop/assets/icons/github.svg create mode 100644 src/interface/desktop/assets/icons/khoj-logo-sideways-200.png create mode 100644 src/interface/desktop/assets/icons/khoj-logo-sideways-500.png create mode 100644 src/interface/desktop/assets/icons/khoj-logo-sideways.svg create mode 100644 src/interface/desktop/assets/icons/link.svg create mode 100644 src/interface/desktop/assets/icons/logotype.svg create mode 100644 src/interface/desktop/assets/icons/markdown.svg create mode 100644 src/interface/desktop/assets/icons/notion.svg create mode 100644 src/interface/desktop/assets/icons/openai-logomark.svg create mode 100644 src/interface/desktop/assets/icons/org.svg create mode 100644 src/interface/desktop/assets/icons/pdf.svg create mode 100644 src/interface/desktop/assets/icons/plaintext.svg create mode 100644 src/interface/desktop/assets/icons/question-mark-icon.svg create mode 100644 src/interface/desktop/assets/khoj.css create mode 100644 src/interface/desktop/assets/markdown-it.min.js create mode 100644 src/interface/desktop/assets/org.min.js create mode 100644 src/interface/desktop/assets/pico.min.css create mode 100644 src/interface/desktop/chat.html create mode 100644 src/interface/desktop/config.html create mode 100644 src/interface/desktop/index.html create mode 100644 src/interface/desktop/main.js create mode 100644 src/interface/desktop/package.json create mode 100644 src/interface/desktop/preload.js create mode 100644 src/interface/desktop/renderer.js create mode 100644 src/interface/desktop/todesktop.json create mode 100644 src/interface/desktop/yarn.lock diff --git a/.gitignore b/.gitignore index cbf0772d..8e99392c 100644 --- a/.gitignore +++ b/.gitignore @@ -12,6 +12,7 @@ __pycache__ .vscode .env .venv/* +todesktop.json # Build artifacts /src/khoj/interface/web/images @@ -29,7 +30,7 @@ node_modules # Don't include the compiled obsidian main.js file in the repo. # They should be uploaded to GitHub releases instead. -main.js +src/interface/obsidian/main.js # Exclude sourcemaps *.map diff --git a/pyproject.toml b/pyproject.toml index 49072af8..688992e1 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -56,11 +56,11 @@ dependencies = [ "uvicorn == 0.17.6", "aiohttp == 3.8.5", "langchain >= 0.0.187", - "pypdf >= 3.9.0", "requests >= 2.26.0", "bs4 >= 0.0.1", "gpt4all >= 1.0.7", "anyio == 3.7.1", + "pymupdf >= 1.23.3", ] dynamic = ["version"] diff --git a/src/interface/desktop/README.md b/src/interface/desktop/README.md new file mode 100644 index 00000000..f7a8226d --- /dev/null +++ b/src/interface/desktop/README.md @@ -0,0 +1,29 @@ +# Run it locally +## Prerequisites +Install the runtime dependencies. This command should install all dev dependencies. +```bash +yarn add +``` + +Run the application +```bash +yarn start +``` +# Deploying the Electron App +## Prerequisites +Install the ToDesktop CLI. Full documentation can be found here: https://www.npmjs.com/package/@todesktop/cli +```bash +yarn global add @todesktop/cli +``` + +Configure the `todesktop.json` file. Fill in the `id` based on the application ID. + +## Build + +This will prompt you to login. It triggers builds for all platforms. + +```bash +todesktop build +``` + +If you get an error saying the command is not found, make sure that your `yarn` global bin directory is in your `PATH` environment variable. You can find the location of the global bin directory by running `yarn global bin`. Add this line to your `.bashrc` or `.zshrc` file: `export PATH="$PATH:$(yarn global bin)"`. diff --git a/src/interface/desktop/assets/icons/chat.svg b/src/interface/desktop/assets/icons/chat.svg new file mode 100644 index 00000000..ca42280c --- /dev/null +++ b/src/interface/desktop/assets/icons/chat.svg @@ -0,0 +1,693 @@ + + + + + + + + diff --git a/src/interface/desktop/assets/icons/circular-add.svg b/src/interface/desktop/assets/icons/circular-add.svg new file mode 100644 index 00000000..9e6ac67b --- /dev/null +++ b/src/interface/desktop/assets/icons/circular-add.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/src/interface/desktop/assets/icons/confirm-icon.svg b/src/interface/desktop/assets/icons/confirm-icon.svg new file mode 100644 index 00000000..98d30500 --- /dev/null +++ b/src/interface/desktop/assets/icons/confirm-icon.svg @@ -0,0 +1 @@ +confirm diff --git a/src/interface/desktop/assets/icons/favicon-128x128.ico b/src/interface/desktop/assets/icons/favicon-128x128.ico new file mode 100644 index 0000000000000000000000000000000000000000..ed751b2d14935cd7772f94fcde5b47e7a012a08d GIT binary patch literal 205167 zcmeF41(=l8_s3^ziiu*Nh+=m)7Pg`& ziUG^Q`G3Cmok2PU(dYm2e7Wzu`@Z*{b8g>z&bdBchA+=oTIwUL@f{rS`L6W&e36Lv z`|LcQZywKTYP{dK^WIs-K41Ixsoza}zP8mqU$0)?@8+BMeDC%4`TF)veZK?WpXKxI zvrp>%%lZBzpKti^)cd&^zP3X$e7cl=bga*})6h&`bAFQ)PrT;@J|A=}?MfsPzN$o( zFPTVg0v-TgfN#Mk;016o?=&Iw#bU9=qS5z}cqD_rUk_|4I7;6O=_@NM;LFPL`}BP* z9?Rr!$AWoa(KvqQ*%sBwYF|7a_xb&P7e#h3HZz`O6oJ8fvOtp-|rn?aWBu(7^>lDM}b z?N(V?nXUPzO-@dhFDFNF1j2m3SvVZ-mJElxhIc#khTtjJTjd3lEquYV7C(HnCHfCO zFBz%asG_30EolzWHz57JM}Hg&VzHeoPQ2`u^7HPo@^kOCiV^o(WcT{!X8>DiypwVV&jBzy?DR<@7C(h@Q_VYjfHtDn`(Cv8Gq_Js2J)+` zYXX#$orvT0cr<1L-#(K_B>U{S+d<+p1ZwrY$A=dxF4Y6dnJd?XXM!KVG;mT?vdULe zU8C>kLwqcFRaI5_!jW*g9+yDJiQpJeM7j4Jbn`(AzV7ZPlb^z>TvKy=uAQN&^U<8S z_Q%CTo4346br`yYJoxJe)NZ^8J_YZBSHWXI&ky0fMrH9bA5~;|QMxzmK+~4=4nS#lLW!%yNGx?)|_n#69J&_!E|l z3$)1!|H_wf#JfA?bYq^mx|OaIsuy0l4+MGE8w3jr3&g|Tpgr$DNW1OD*QkjlT6MBS zyY7~(Yn6)MrE@m#ZpOQtRFsz~ezjp8sQZDx;`itAY%g$SAdsoDunpgAn2bi-BpNk- zCA8ns7QAG#g|7d^LbrTru@Ns>V*3*;S=;oUL^xE*ySwpCYx;){#Bn*;jlL<9xYG3d zQf*D^Kp>+j@2?NfJJ8@FC&F&5|MZ4nEMSAm*>JJ;!pA3v_yyebimp#xB9C4S02M)K;5f@tM@bfLY z-~TLj(s+xWaL>$myYBrG@%Z7iQ$+;@d8v3jxx2_0`43to|1!QgCf>Bg(G|y>H?Qc9 zH>}{vpXqbvTK@BMt?d4vEO_l(7CGRIKNHophr#QSe53Y}wl32l9j%wYUV17^)m7E% z&$7Ysw4bBO&KNZ(`#rOqHySMO!Oq+0!os&f)%8Ksoi=a%kF>K6Tg7j{A(ueRdjg(x(!4&YN>A@3pyB za^F`L?X&X@^qcCNHv#E>p%d|+LY$Wn*H(UifIh~R)%mTp&w2(z}-Z z{#*;ZJIAs>?i=&0oCp+UN?} zffEwJV29`x*MA+JR&T}M&bHjC^%nYMwrzOY1yf=rCEbbl1p2^$%BHT=@Xy_+|I=8~ z8xOerW?bh_Fy6{1jbn>*a++^<{0T4K_}z5d{<~kT^EWfC-^btDW(OVmNM%V$58@w4 zJd#6!;*cEDvPj48jrUZp0_E^91RTZRviSC3`unrnwr%^v&O7g7TkpPyZMnx@)@P?( ztacG&amq{PO{Qt{f4M@_OhJ;y+M-BYyYZU*CJO z5Y5ZF>gwuzg@uK#uP2@s;5aZT11K=1!RuAT|% zQR@=dN1_OWM7VNI3uh@(cl~An+G)Cs{y@$=7H|g1aJkf zva+%Tc~p7m3Djra34gBxjmMq@6TuzeN^m6T%{#$Vd+qXY1(RIpwYYiy@Fpm9b3EEMmG{#anVxHoak1q-ILBKIFrE~-`DSLM>2TC~H* zeuNs2u1J~-zVjhzGYnd_^pnZ?b&yDs&#q+pf0FKycru=;^p~_guq>IR(uyO+|i7e78Gr>E`C+Hz!el^9eQA?Nu2HDSgIf0mfp1Ye=8< z8E>wfo2&jWK>lQ_ZoVPmT28|Qlg@`B3V`C=eskIV^hYqHSl*U@!kT~6W{4Ey6(k4A$Z317CLE^g%3H$A_tvomBa70(EnHq8g!|}4+MwY zV2S-NuTQk;v1>9|Qb_ve(w~P&zmhmNCM*!&ntv>qzGw(A5EC65gD7NeMDlO0IBI>q z;>tY^9asF&7gm1VixwVyzQs1*)xx_UW8qznuvp(cEOzLX7CZMfi(dMH#W&mS;$&G- zW4=}0$tx-j>pFPPUM_$X^f;mjrO|K%qdhzY)(RegNajooy+|_q!zHwnP673&r zv)(I3H$Q8+uQ3;39G%NpUE_AmI|`nkXQhvW$L3n-jB%D|)bt_dvx7)i{h|0+YVsOe zQYKRJQKg+@DXrYp%;$SXPE;s;YKtzoO$HWN=m@8hxPr z*mFsrHMlqCSpEy*d#*S4LJs;}GI@@L_c?J+B3yY9e5=jwF8&o?O^x4;XLZj-Yb09S zn%h9qSKYXs@FGx^ddJCt^tS|;CCkhD2lqH+dd_RVJGm>p{1@T>bJPQaADmMV?RXCQ$o(ZB{YdX1 zZ~)Jn67u)-y07YA=6rfdS7lKB-$?k6$GWb6ZQ+F1Ec@dIH=hfhdkDHORs*y z;w{^sMEkrU-(14lnA#vGfAJ%^YEI$hyI1D)d})_x2YJMCE$aqt;&n|AEE#$CyzDQS zQ!?iZeBjM1RUUF)C;j2q%qHyte6thzew@BnU$*F_zKCYu^+}8VZNcB7UF-XV8e4Cf zEHCd?@xP0H$^Ew8a=)By)@BU#e4?gajNI_l;-aQ`d0^Huff^jbI-tYgb zFIK)JV(<`8YEH5rD~++Lfroyy;Ws~9qaS8l(X?5X&)hWYt9tA5!8g`t`yF3q$`c~p zi{O3zXf)#TUTLztyxdFeBHfY8C(@=q`kUw@(%u4`!foFATlSwYVag}=;BT{R#~*&T zjlTcgw)%0F9q{x^wqExhqss~k%1Qq!kRZ)m>UE*kHq@@G9%!9=NmqV;7HO-Dx^kg9 zaWNRiyIG{II&c~D)*(%sH2rGl-FCNw4;^H?95Tpu8#LH<9ejjsxx-FYTV3-yeDA^g zS2k{3M<1_s@xTRyTakVN;e2^4N`q)r+LY1tz&PH$5F~;X+WN5jnSv@&JEzaHf`Eu(8dLbb8|4BxXuNoV9})A@4P&ZcVr~qd(uy4i=J34?u*1C z8pmoZp)trkU`g|VKluB#{7rOjP6$69Po$}}rh|!(wK{Lyu_zJg`M2^&?|8^@V`s$$ z3Xnlh!b`!D(*Kj^8n1O?z@7JI@={NX%!xqFWpRXsV z9uAho$7Gh;c)zExq`#lel7r%#!T5P9O&RYm1oLI&*kr%iFYxAWh$r+Km-`_9L<%NnF*+TeVAT>XM8=OVaz8 z^j4I<6?%_2ht^iru5>+J8D3RJ!Fpio-%dYsfhN^W)&~93-T2^drhhJR9zmRIoPO%A z_5p^Gmg?dPm5t@RC)ytbuC4FVrOV0`i4}fVbt_rrwVm`OO6$4h(6yqxO`}frATHO2 zuFm@V(wKo(pT<44b+s9!RSL8Qv9fEIOXFu6I)55BOQyO||OoNPZ6@^j&uODCiMQLKf?! zfg{28nzm@2bkR%iA$#VLf#bm&;5jgub?kzoqC%A)t-Fo|yQKC~gQRmjc#HM%xry33 zi)_BVg$Do6qNm(rk@KHJe`JbP-2RnCuKdvAr$1|nLs&=NWKT;rYN|E$U-(vgy(o5*RtmNqZPg|&w>w`g~rdd;DfAvvGy4p z-(b;OSqmTCVDYP`!^icOY|-((L^#-i{b|?klgBET6F~dQO0CQLqu^}P{}Q|ccA)G~ z{QueqUzCZwJa5lHWuCgMIsGAE1oTJC%iMZ_%I8SxSPt^@B}1WTqIJ9HV|})^;K1W7 zH1rCmw-X%0I{tlsTE)ZYrc9h?MZOH^K+dcLE#s2zRbip_9<=VFbOgs-rjd`e*Ik20o>pHn{!EM+|9>X+7p@1h=;XnhL#p!*1lWwy3hW#wjw3Wp+ z>SwVYeYkIF@!mUG^yGUidd*iByLlF6X|BZ%z7qZ2#6HQ|CMy5OtL>y72Q-FI+LE`{ zwR3V7FYk+xy4;_{uXZR5{VJnd)X|%>;bEw6)I)pqJVU_1UWtAt~I0t>Lu&l2u10;xV;t4+HIst&uUb`twAD^{cqszqklp zuZm+Czc%G$q>+JkZWU+`1(jOucOl~0&!<&V-vjrq#r8*R;= zWkh=tjle6=uerVJyH-}dEAl?^s}5+aax3@!64A)pl{+8QkoV#rZofjhUV_`Z5D(H{ zOYxySox*3?%XwO0P5s_)E#9p+I>S-v+^g=sLcFb&#+tZtvvR#UjLtIYJxN*vX@8!G zb=m08f(f5G{cY)!d3#Xs0PT6~!#Ewh6ffGF5f8=CUq13hOEhoG*gyUN@f`)-kI}}3 z)y}QVMZci7b}^_A(4GeU_52H(#X}d;dYh28H2zScu9+3x{G{7I%VN()dlr(l^b!l6 zqdcKII6n`#FMNjmtS9GLXgK8$HQdzj6Z!+tUqqW)1*D&r zpOYi~Pq%ON4&f)z`x*F{`-z~7)|rGPieE`hc& z(EYLp|Mk#Vtv-(5eYhXMJNt3JFVEJeA6`uP3Ylx)$4_`I^pC1zy#t-i9mD;0n4Rfr^__u zz<&T>Us8J}ZvW*{=skk5foueL&)XMJdH445RK^w`qt+yAT-!|fL4P-K5A)UBWH8t% z*0smmMv>e5&^vS2y!7W)e`rT;Dn08ebXh4cpF;nK z9^I0?_)r;geS-ReSL-df-!VqLIjyR?R(p_|2Tfqykf*W3n!Y3>P?Ms6+azPHWMz18 z=#V4l=6&_2<$PXmSsx+uPwFj?FdO<^`FDH4(Er@e7T$W--zaz6HL79^n@C&@E{rwx z^xD=m{gfHifgt?;Y7;mtTuM{=o{qWAB)~+oEq~TlIIdtnjnhw5{*~ zDybj+C%$csJ9hsv77S|NAeT5M5-JbY$hb%H(B7}c-L%8nv(VmO2KhdcG|v$~?0?tA zyKnGW`1v=j_NUoa4Bh$P&azVWq1`?E;W=N-w&b)qHuT={)~b2)=Ykc%a_H8a`emT; z!y2Jq^B8X)pf*Z#%p6~J!s+av2>rLRC&OlLuq0mBVAL?zZvoExoWO7U)6B6ie2VMixt8neT+7ew$YeU>0SKz;5Y%eo1ysWI zvu!XuUH0Ql+v$J;eXs;HKM#(E54RS|y=>^rg>L;yYgiuUKl^Ll=LVkL z#J3k}JxOH~+O-dlu2JAE(z_FEO08qw?&-#?C=bv~Sqvn=J;e11@tj4zYD?3V%lU1P zx1J??S?Mo)307oJK>K&E^3BQMMQ{jwuzz1bn(9*n=qWkdq&G?TZLlqQ;p~4_lzx>R z#v2-2uznKg1OMoPY5h^-*-Y{#>Z_}+Q$KbD>Az2S1~eU(;$i;U8o!rx|0LhYK1O@0 zFY&F~QAb))QJyQa8UjHBwo(3TNp~o4wwNnQzvQoR5He5YtN1BE_FfFiZ>0&!py5=) z#|WPWzkwyuyx_C%q5m210O%8p1Q%*IRaT&z^s~hadNu)CZ|PC4DD`^R8w+{m$Q$1+ z`ZX=LFCKMkLVoD4gqBUA{j(LIUwn3=43^i`)j2&9zFjE#^q5_|2)b6cJzXdPerUb8 ztfb841N3Md6Cr$m1?Yc>wE*e7xHU+X#eez|{m>I5T#1(%3^jqmk$c<_n#HL2&YI)c$UT{4=? zf^MzpXkPd#`O%-*mD|Tzow%3utxZLti2g!h&dP+7>Dl%Ea0V1h2q)|L8Q8r0xEO*h zFPy6PSlUNN%4ta1Gh`dlk2ppG3ox;8>>ZsXdz;7`y1qzkEgL3VII0JZNQ1MSCL4^9A*VGQ38nc72<97=ImqxApDt1!RN z7p@FT?@jBIy`ggqcpdx-795+c&b?$N*=SB7J&0EFGb4StVELNW>sy{|Ro+>TY5tc{mn)o~Uq4AZ*OtL3k8qI6s`RkN}TBZL_Uhu8D zSVpLNs&cv}cwJJQpQ#?K3;e@-6g<8bd6zssh4+^KH2>s*^1mGZZ-geb;Y-S^zx^5M zX`Z{-y7=0z3((+;g`J*=Y)3y@OZu=5o^O$ZyuBnUlks(9K<%P$hH8A7sPpR?u-f zq4cZN2WV`&Ch|QUr*vYU1=9Dt3G7VWD-cc1>)bN#(#ZbXo@>A|n|SKk2Q{V4dp z5NHiJP3xKn=fL|I($joOYbS5;d!I((Mj5g_UK^KqN0%!TK2Ly-7oeeCRherC`tf_( z8s=X*xkEJWrTv$__!TM()QMvGf?D0n)8+NnR+pqps`%kiX=f2?om}%-t?{ZIkbRft zdut-!U7Vz2q;tIH9jwZ#4LSz?p9b3RUtRXyz8Wja+N%hJLZv>f`)j?toOxCeijulq z{jW}DkoHjUB&bc*2k9|?4%Qg|iW?aZOV(gdR2kBo^CF(9{z=DnA^x>akdx!-Yij>R z`wqp$OZMrJ{!n`<>^EtD0ET7iqwH86TyF$ympd0tHdkzs! z1k2z*z3+uCw9|=3=Q$ev>wek#+=}-%(vyCX%3(Rb?*~^TlK4GH@h|@vMMV|xp6&Kk z;3v*rL1*~U{!^OOC59gVfe3ux$NgvEGO#229kr#Ux%dK+9MLx;EwB*~|Jy15SqIQO zcyH+W1}rQC#}_=;G`2*G4wmTB+v1z*TmCEPY@lN$KP;j9=2+b6(L{X3C==ywO|fppZ5r#Y*1 z8Jm&XXk3O6|382%{VGRo3JUTy=9hlkZO}1}zPDWc5B(x~Xlff0 z+f>El-^Y6Nw35s3u%i1vz=!WN^jzz$;vr84&L0i@mrs~uC6n+&az8S-V~#~{ga45~ zS>l}cEphZXOLpz2GVm0rAuYE*7Y;gKN7_S?Zfu>W7oM33eFwqsbi$tid$j&O=!O@m zUz1*x!WF&v*6k~k|Ad)&{MNp27wSlENlA(9wp8DT1E=SaY7>1`;c!W!w(&#Jy$?nA z`WTCCytT!<_jdkgA_tslm1m5{e&S;b-u07}KU{BRkD&Yez)TC>_?bn|eB9y(Uyg5} zLoL~T8%qe=(l%_g(>(Yey3H*nAJe{~k|NFh{s*MnT&ePp{tETh`81~7XYNUxU8r_n z?Sao%$Ugjva=!3;`T5zYI^ne|(&6aZwVUe$DEE6(20nv^^;Q2>H_+T<|>QqG6dcFtt{T7uf?}U z4}BoMfsVc3;^*O;=en8hKGCH0v)BYwq{=`M;aJigLwzXp^y;%{_p|83vof1EU*GYq z{o<9#@`V{+#xK|3dvm@`py5Mk*+u1F&*A@&&|ZhmDSUpWl{_`q=@getrjFk~%|aJV zveTKHj<`{Xu(A7o=%xi|uxtMJ{~NBBOt?=$JXQ1@#uY30y*3(7E^bRp>-& zoI{!k!WY0fjp`ctwf?TMt@c~(vvhmc^rd`&U#&r^ZI;e#c3qA1PD5t}hn#5z=vEg# z_MMg9@S=rIxEdRTy~+Dth97NzF%@z+|#yZ%xY3T>lwNmFy}4+!^jV`2KfwSA@d*V^v`&~XdxVR2Q0 zIbkGnL&czzt>mJyR=LC87Nzft(gsxSer&7ttAdMh3OlZ8gU zhmWvZDF-K7e3P9l-gAIO_8e-#^Bf(yt)Rt%2&@wxqjSz9CmtZlo#jN23`6lnN}uuD9efdANa*qvGN^^?#S$M=)`ZejnC`zre5==myidiyE+iClF0 zrI#(8{vz6ovPZpfLB+$e2|*`#-0v0{bcrRZYi7|045aVZ-17>c{T7V@SDA;^`7o4^ zpB1eG#1rZRK0zK^5_V$T^Jct7o4G}!pHKN6!}pPN;pcB|AB<~9*aMV z-t@Hc;n#ZkE?s-{?6cm%?^8o6 z`cntQx69)N?RXZnr&@sj(#dnJ;<{Jy2j2mkn3&Qy73@Iy*Bo;X^?hppSLe#i%<_#G z(H}mu8A5OfMs}8;wI9aP*$gN0Go@p!wIIz47QLWF`7hhOJIMbjpbnWF#vI`Hcgqcb$%-`NU~@6Hb&`CIsG%8n^r2hweV!?3ow?d!fem%4hwJ5Zzk&*z=zWPEX-f?{{D@ zWh3eN+*~x3`DHv(baFSth@f; zb;>?vL-sA|BMSL@$&>IuY0kXxkW2r-mT@+`4-1wCrCWYBkbM_z?ULKR6#v-CW`?n~ zuV4-!AMf268}7pV4P9v2bp`Mj;OvsI2M)XsX7cid#}3lo(3r8y7i3?Fw9jw74^&^r$N2)>6W z)^=of<;TBhW-)Tvfae2w#@Pw}V|jiSzYpYZz0fa}lEo4y&I|Cr8T5S5@8LS*Bo>Vn z(gutV?Yy65zcz!i@8z#+^HMSp@6Hb)Jm<)cCS_kKTP%%I#T zh`J@4theEt`EasqY5cpCOTe&YSYYO&H@;m9q zKJZIjS9>5I|IQwY(AnZpNAh2Ue^7{z((+L+GDlkP5A0EQCw}d@NT2ElWH1usf>e$# z9C$thv62X66u<|zftjW71TxB~8|D0FZj3=Y^J1;*YTm?eWF|dw&qhIV(fo0Fmn#EF zbU@-x?|K}0RHF51_#YH+w_Zc>#Fs4ROZdg^DEnP+44}S1_FsZuVak@um4DfS!G8hv z%@ya~`zLMR*NI4^V`_da{q!G5bAPwq!1!xPmw1r9K72ZU#EZw*!e<(AT+wP@$^(!G zMnl(|@L%QfDjMyQXxjYy;!)!)=QGMWXCY*Lj1AfQ^f-4ya$c673#OF{KuA^U31tjc?8x`nscN#njdFefa)5?uUWK-}v4 zTPlrp;Zi<8mv9L8KSI~GDw`Z?l+YEPsw}nE*SuoebogwHlY1? za|GIk9Og`UPkh!8?cS>aU&g0)Yu`iToIK*blQiTXDX4VUg-iMGWFUX9`1<7j812d9 zhFm^KhD3XAI;Y^x&n@>`^7sp6@R^4K>=A?9=T5;ck^Fc4pQ{7Zef%|-Ue8)!tMurq7^V}X~G}2;~yOR&f})dw8M`)N_%}zrt)3!_lN)KgolcE&AtE0D`kK0d{IG{>Oelcj|G!~ zTc>3Ys}g#jNen*Pwwd;;4f$oZb;f?O6rKyn|6=mLgx?DDB^PWd^C%zNP5Ih3xY z@LZ2fJwG<_#KFY%9qDMzb+zX`t0Fti1!Z^57>cn%8G9}z{LWO__mxM=wH9BISQ`ew^7|KJTb-lTr=hacOI4Rh_+x%2GnISqCTcBuROJPXXW z1AnTwQ?Nrl=hu3>2Rqx#Z@b-^Hf=V8eY7ozPyYSX_TNLC6G4&Wt&4onc+QQ_HJ8&k z?(e>I42I^usdIv~H(JX$T6>GwPX)=Ob11F%%>}>1`whsj2l0K$?~_!{%F3jRp)*heeZeb0^Vh4PZ#QU) zXwHH?^lHmMeCZe;C~3;NtSy9UrmkYM=g_CIdj+bpy?K%c`bY0{!5taB#UV`)K;bGb40G> ze`@FOhk>7x!1;jnfxx@q1mw~dwn_W1GNwAo-eey1j)m5@z;4_>485NL*?uf0i)B1h z-hTp~CXT^Cwq_dlyLLrvs7z|lOKUpF0G*A%(cmNaSN_}tZB69cM(?cZrF9hie-sek z-o&kQRyHMGMC$Fa$&X?B(jF{jTn@K?BXDZOzZx79Z7hUi4)UrbG@A3kh zcz+z;TTurvODm?`TA0pN`33LlYv5V_7Q8lLN%7QnaDGoTHq)B$_TVgNzcafF+97LROb_iwQ)bs zq?O$SCyre+OJ$${#E}6Rac->LYexvUW{LToV}O{59hCB{@3%( zz(}R1YyGd~A6=yyw-(}bMAZh|3T^??o0I+iEHDSi=l&#c4;TmUmr(bcsLZcR_V6p4 z4_&)<^)+eI#8-mfRd~u}ZCvXD(qTN8-&!v}99#~jg6-%(i)imk=;Ly>8?c?zCzQV5 zzxit3yt!)&;Saq)r?={@6=T1E&aC!xw4eJW80h?l@SgfO<-NvK|D{Vd3XNiof5Em{1N=vq7FVz}`r#Cc>Q1;=6Y0XT1oyIt8^OCOs z94?s6Q>uSM*r0fNnt2WlOnwk>ys%g}b%>rSn54O7r`*|ZKUW=&G1zz912MaNoCQnI`upiI=M++ zl3zN+GlTo>!r?3p>31r04D>Xc%IkQD&|@cf1-v-Vg}!t+EcKhP7n2R&uz4Ai;hr#O zD&gc|E+V3Ff~{7_7sHkl>d$=0Nj~#KSxxl?7lRC64gYfu@Pak@zjQW>apqRnxX#zI zsoaP8GHCD0Y5QAHC#2gxkaqA4>cpkgkGr`aM7@#Eh@n7sXL|vyx5*bv7yd3ioGSVk z%@J80^ks1Jj607`ZK>)NOs`3n{ksiQj?rL$v1bEBF?C20jMwfya^QwS+?ncLHsYP2uLHo4dAGa%$0{#oxxu z-x-l;>({Se+NV$v{P%}`jjbO6Yr^(Wacb@1Zm=I^tdVrCq<7S@`#R%YI{I`7UR&F8`X$ft<8uWy@ewpkLH|JXCPJ`-_gTpx~HH0E9% zepmI|>7;uUZIXXo8{bjBHTKl_^aAox^OIHOakai9{fD5VMD(mnS4uwl$YwaQnf(vS zX9Bd8u8ZZVzn2f8%P2d6Mc0!5 zX<=zhs5TM$>cHCeDWklYN4hV8&Khg1x1Kxt{ht=g{PdyAtqrhlpuP5i(EG(ImVs3& zccl3;G;E9?vJBQ~=F_#V+*G2_C#2-0G0-hw-rAB8X0DQvsLf`~?(#_ar2P7)$3eA8&i8EObc*E5X>;h?9og*9{cCGg zKAvxr-AR8N((VCDn>KBV{(j+V?c=Ua{giJg|%$nU9Kbl8M7vn{Kf47$9BmXLI zkTP@X{~U>jW<;LJe>T~o>~`5v#(k6X4-ci23C9? z@AR)0@J%BhHOu!&GhhCs|1#;-K!e7qG455r$0D23MdjnkB?lg5n~|MEU8w@tY%RYX zm%e36K0EULm&gg59-U2^)fPWL=w_kE%$6_a?g~f@>7Hsi}D`0a}w&f$1YFjg!r|fmYI>EHr0_aa>;oGT5au7BHAA>_MH_23f(H4j<*;_sY{#q}ST#!e+ zbqCHy-Oi%B54G5loIN<~e$GdI)`B;EY!!EWW1*X-TJ*xVEIw?K#RqXd=hjDBvfT!J zEB`+8+cnKcwHLS>$U`5}`FY{KH_yKYhf9w@^hn>#lZo_^ve7%u+75m=(=6~ic#3+m zAyUbbf0h-wii>k8dl@>zqBnR1{6;*#p*!>t*ogT2Iu8q4v<}}E8OZMLB04Uu0l9H2 zWl8J7WMN$8OMk~+wY`N7JjqILe9rQp{?7R~3Ex|9mG|QJ=Dr4ez;IUDMC{wKagW@N ze~VkdjrhE|_GgQac+(QQkFaE`E|i;u>?!^STGTYD$xywP%};h_w#x6`JfDhO`pX~7 z;fEgyFFFs@`7de(ADl(z&WV*DoXPOBwG3KUL@o=yLz!{yGa)*lfs=_#I-qlb&aK^% z{>?96%*cmUF!2&}-9uPT==QMXo1~WWuz1#yJbx$F9iKQ?jl<6Kcguf$p5?zRoBsyR zqV;?NIln*;%yE8UN+#heXk0yK)XuW#XwDYJKT!OJdW(MW! zI^D`HxC8sY7p(lA@2uj1Svz#*~iL9!d#2oh2NmD$mfRX=FX2DaZcr1Q& z`rsR6y$vkhzPrUX-NB-Jo@n9I##!i^w=FpKd(H*Mw&am|E5Cn+1#g&Q(bFErr_P0( ztGgdE!LL?_jV-a^4i?|@982tYvL$Njo=LGpc z05_BFExe$Et1-;7$9YSV$`eX zKLGzCGp+c^*;YR8TMJ(EvQ?gXEB@6^u-JzEEz!D@B|28a}$3@35+MweNRJ{ ziR;k+$#J$k@x)Hh@eSBXd#U8n+7iQ3(~*40C+%H42wfAQsakC`&pRa=)qPoX)8ke& z_C+hlra#hWd-&amGq8JGc&CFcc+z#8oBIOiN`Gy|PjJp3zI=))Uxkx?vr^8^4qfz= zMGn8x;#=-#i5{Gx-Ocm!6+Zq3E5Gd%t2}hXY~*u0etq*|)iJFl91Elqwz+f{)UW;1 z7qn;|rg6{p$mf0J(~(g6xf>*^8+}u`3+Lx;u&G6RY-*J|A8HlDuD8-#-?E~|fA!A2 zeNN|9Ui36)SaX(p@P5WexBh71E8n)raW`9R`@ytLTUfO3?iTC0 z^&g~l3u)$3hDC$cb>;hTJ>YEE*9G}No5st-DKi@D@6@zWQy+QSfxP`J*1ofqpL8i@ z=yl6~Y6gBFIEPdIK;#cW{$Nu6aOV4k!B>XvB`=jF`9%scE|TBC-~-ss-t!A*e!pr_ zd}VQ7ID204QRI*OfNJdZG**!ah`P>&C6b+tk!Alg-oVE$K#}}J%WE)H1pF}UO`tMAiW;hPH5~g1{>gMv924< zE4=?hPd=}CKBeR@#mPywHI=E7CsOBS)5np{ zZ(vW-+>r2la4qw$?0RdvHI4 zwrEIMuq;D!i*@CKE{&yQ;2G$A85-q}Np@2YGPbD?9&sk+1V0Se*~=%@{Bz+|j#6dE z`K01Jb=M9_K6);{R4KnT^2sINJSB{yN*|tKvA(;?HD_3FoWh4&?JKwDv|?vMv0QHJ-WQ?e?Y}p6UG2%=cL(zck3ey~;Q@Rau zWeNG@xwa_fyXXnpANuxq=bp$X`Udj27i>tnSA#D|)A^15yULH&&Ad5^&S#aKmOC4l zP-AgD%S^2;Ej>G+PkR|^lP`is=8efc$iG_f0=_IPy8b4X_t;eD)64mEliv>Nnq;A~ z?OnY|`?aDBE)L;2EIGJ?R-bt75BcSz`rIw%c z*~=P7Ye}`JNb?UjM{{!z$%6F??zLu`16X9t@wa7OwI%2W`hz}T3-5QJJMR}36-jk; z$(Q6)oG4ZsbOex%&goS(Rq{bM5FWmXw`^+#W1h6E_dK6huH8|;j;|-lM|H=Q9q0Rr za)rMi*M7UU$&+Kcew9AX+Q(^k{E?`wLq2YOtU2jKq2Yagzd)${xFr+HQ^^O}Xekhb zO&|7)&ZSu?a?!lI1#2Nu_+r6Bd(H6?%roW|pL;!iU+{4vUx4uKd}2|4=KI5xY}9{ACU0<0 z`H~z}&NNn%UnsRbWsm$ZFS6gTKM})kq&JwJU1g;?W%et6-_OXW5GGGp^hqkKDJo zmu%9$v*b64w#ki!?we-#Ox3vXMdo-U*)FoHJ5E$m=#jg5SumftnVQ1a@=Z~U(Lo8VVV{>PWOxG(u=9}dK! z@fm)<1!^Q8c!<+Rzg)TPuK359?tI2%i*NPaYI`LY>RJxIoOjDrYG<=D$E0 zq8=7MP1`*BUDl9#sP7ud*@b@A<(>Zu(o=mqJ(h?q)Ama8A-_ZL$QrbN4}R|lulsU; zAUFUV0@TJ-hwx0G%iELi)^>Q;t!E?C)OtXstE}FeOaEWG3LuC491X2n|6NaWMEa3D zczH0|eM5X_eQa5uHdq$^g`A8ucBVdMQ)aT{cMAAZq33%D`5{No_ZaOI^R)6SpJ+(5 zY+p~DM<`8f8Rg$ZW3HdU4wT`g6Ukp9ksQb0zWcWI_3c}mepfpe4zy1h1qIp{^VUMr zFRX63c_;MP!_+6{(3GL?$;$!Fnl|%wXxEXsSnRBL+xAv6{wd4-8lPnNACm7K`6hKf z$s`l`BTEDSRFaRAk;;&K(@}<;AN%1q&cw&z&&X#}?MI?hBOetS18GmEDOhDz)P5|j zja}tP7x)mFsEs}hdNr;+_R%XIb!T_NOTR>uX1|qOe!J(h3|~aq`0WaOfZrPnRv zcP|<7+xf40%bSm=F6QCmxZtVZ=2h-}Y&~uB6zZ(joRnTI>3v3ek5hI+t0SKUSXm>t zq78a?fnJsM|FORzJ3aaKXTFjAy!?n$E%yU_8{^Y08^1+?k5YcUB$Jd+Gx>Q_U2{Hf z)mEzxs;$DeS^lG6TBMir>;D+-bI8djmROJUq~CHIYnNK*SR=gs9g(K;Q|mw9AeVcz zZU)`dfmnz>Y*KjZohVGVwfX}2q2A z7Q=^D<+MMo@GIJ;&t_YWWJFt(!+oP~>+Pm*rrWN2a}Gr;{yK42OFqnj<+J2*())_M zaQZFlvV1gWrf=6g>~QcidkA~#Ea22QTK&xX;Xy}Q=<6BQ*eAJMX=Zt?%Zn&&T3-R-w0oj8s-@2zA!XQ^4&9u1oS^j>lLX z#Sq2~nh)zz83$ViPwGr9_Dx=*PpX{@x?eN|wcKsi7?9>^vtm_Z6El3$E zL_S5xMn35jI)8mCTll7{_+gIi`{Go~;X40m@_= zbLZu04Zw{%-56VI0?;bD+f!E5hus_u1)V)aRdrQ>c6vn1j-71Oi?7)i4H(SUH`w#D zJ->CWex&RmBPSo)rDDoXnBUdYX4}^IaXa9e>#eRy(?5{UK*@mkC3oq9{t1pyx{{B^ zcN+U^9>Lsbt%^%!MHh1u{*NFdf0X;32sschZ17RX zXg&2yKTQ{8a4jqu%nK)tMGL<`p&y- z?6~`E4B^H1+-GOrbH9zc?|vKez=L+@Ll4_!;~%sm$30|M+&0!W?Y)`m)@Q`CEAfOY zla(37e?5?2!A^v(%&=Y|+Kz&zTgkV+&>BK6S!q9jE22lYvNDt&x>siMHE$zt`~yk< z1@Z}>V}W~k)}1)r>JalG%^CgV@d$YNg>VC)_WEleoy2+AemmcHEn2j+R;^l_!j^C*0NRFH=e$qCOM=e-Xih#A z$Uh!yJN})Zu|{VkdHe9oT#i^N@e~&q(#}Y?Gk`xxe>-Gy1^A9MXMu^N%lUDE`O2Rf zzp4yD`@ul_n+L$pIQUTgaNrs1H!B9|F6iC#x2bR{X~++k^wK5)=@~h`bT%R~C}f>x z0J50`Sj!1af!5ohwHs9q|KZa8N_8kJC5^>jR(%4&w-*B;wm5ml^P&vCxUouQ}4(6c*)oMSja3NVc256W^W0yFbuc#{zR|=_vi9VWnoeNm-qV%zTJhmH{sh6K<&^Z;;aS>$wyq~(-6r9 zK=});(huxOcp>GVb<;o-b}#rF1Gm+@)bwc0pSq-UfbzxDIbT}T^1tEMu43^he5lM! zg$MbOklo0agsUPSjdP|T6Me6;^f2hwyjgQ!Sw-1m@=4Pc4wtDt($5TN&*xscs&w?) zyIREAV)~;s9EP-T3rUHv(|MErruwi0;Y+?vwa26V+dbn`K--FGeQ>Wjvs0pxTPd@ippE$Qz%<9##E z(#&WcZ|>G58I!wncUOd-b@3hTvoytyPxDr6cYV##s}Ve|gWoK|a0Iwh_*^ zcU>3xUKN@ugO$Ff&C#1mBoAFh`Anni^+`l%i}9KHU%TYrr9+1fYDZVK{HRUdT;+%G zza}16A8Kmcdhy1TpSOYhET~TtW`YIlR|CJbzNU54G$t|*Y_7UmTU+~|luv5zA^Wv9 z@ZA%B*QYI!-q}MulZ>7N9{|nCFW`AU!Yx5x%1sm2C!Nhl`}XbsbMjH2ukwRTT>h)Q zc78}XYZ5uU0<`bi1Zb=+9o_SgNsjgy@#F5k<(cct)$adSuim|Tvrbax_Q*BYr>wL< zHu8IZI_ER%oVHCRAD)jymW7gu?1bdYcU|=_wx;r`DvC}+j6Je=aujm;7~BBl!%cdy zd)Bn6apS)=mHxUOZ>%)tYtgNR)p_Z&ImKy$6#r_!BAf=XWhHMsj>D|$b51NZ*PB?MEwA%J)9O(}+G0dJ~%#n)?r zd$+KcdcN4bM*@yH`OYH^ca+@QaQFHQdn-{Vj-{gIq&E_OehC+X*Z z-TO-1z4O57-TPPKell@-?nh@-d-oYUS-1>L)jKJAmUWNAPWP?eS;D<1MDHDs!fN-v zUe^cw=Zc;Z)4iU`i75Z2?p?U>z2=mqspl-~vEJ{!Ka2Y+@7{FZ^ECI|6{emkI+bCb z*G&GO?EJ)fPgeGRhS>GuCxKz}@8_lOn-lLoU^2+?)$l*p#a^&3{I4@mWozHAZ9Crv zT{ckvo=<;PLOWFh+JRFU2OLBjKZthgMlhT)cY*pyyD4-L+egj)n0XURC33->$u{CRXFdv`TxxfC0Xa9oId-lWQV>qQT+aYkBg+ec_8sp7FCzbskgLY^pYr=Lp#27|@4iAPzw4S4 z=$$vg!$A8tW56}wVsHZ3g>-uIO=J4*Qs~H)op)h=p_?=DoHePOfAyNgvhqx86|ysz z>{;WLUSv6P?SfnmgLib6+BfSR*&mh$N zqZXP9x_0jB#^zdk>efwd+JEO#{g%%qg_QRSc-|IyUx^GRgO|V@u&B(|^zVA;m;qh_ zs&^aHW|c9&$&ihQuK!BA)VXtKU(cRBWrG<6!;rOXvDUPFS0?^>$o~^)lMlyk)CbpR z=&Xf*`=vQ$%H~%3p_2XD^)I$^`8)?2wFl5YRv8P($KJpF(%OgQPx*-;FJ(&l>0SfuC2w-_f70(R`t-dBlDcM+ z(*A#HZ$@)Z*-mOswMN_T)r?aFY1T$z>Z*DZ-ZnX*&ka)%~$^I z;&=3m57lMzAwhT#slRXfKIJV>;qI*obADXA_HK;>8V7*8i#HUZxcXzt!JIc z2k4m&ZTmr|*5!O{+qV6$h~Dv`wp4il^3i`i6MomVI-oVx?rJOBwQKjEkUcVXby%`z z?brDoWZ$uvHs~t;CVk^|qU@*W_>O(i&DBn_F6-J$M>qNLZ`4_7-MqE4`ya5!0JZz_ z%N>NzQu3je`#ZQ_*T$jKu{qlZ+Va`g7ri>$)Ss%e|Ej(~Iy|y1td3X9=C~VWeFXf- zul87l{MKH{x>WY@<%RyfcUHnJgf~Oy#n8DgWv5Y5QBg*DdHKJhO-lPm(!L)u-Wnd{ zlT5k~pYvO`!!L8MGhWuM@*l_kBJS*Xr0???_z}pbk$m)Z`d0`aB2`YbhXl=4$XR7w z`-J29y)XN<(jSzc%7@oM`Pb9_-v(WqGd|cE8js`tVZv$T%azpSFzwPhpAC|I=Wd;~ zjz55Y_Durccp*)2M@P~Rq7WbOa0Q{^9;G#(Y|U?dNu*9;@jI+4G9Nw|&lH`exr{N>lZ4wVrZY{@VZ8`3Q2pO><- z7&)R*_8ydZ`9gdbx?cqD&^@(HHMKR_JU<1je;h6G!QB4g?8x_ zy$S6bi2h|=vYpJ$%|oxrDP2mXaFvY!rHho8SAacuKxZ4=hmGN(xT4vpxFn_(r15~GW8W%{cc8Pj2R6^^Bl~19C|gkZ6!^=Q5&an_e`r;odmH!}z7Aj= zI$LYewKbee$2mGYm!8{-t-(o_PPWK7!I?FUnp(1XdviYid*Ih(!>ug7{+5>LvY92@ zZ)nLD>shk4$zS=gob#rj<9FmgmN8njY&xZvjc#HV<$W;nmkrHLyi*DmlbCcU^Ye?G zJ*Ve)GE3)8TnIh{Zy>vk`Do!nx>zhreSnR0pfPbMzAu36%L)DR<9IpuyCeUQ+9TDM zbpETJf@0`B7s%daC@27Fzu6i5y*0mYgg5D=FGc@~KC!CCorx3giHrvvX0d_z&p#4> z_$S|M;q#xe;LV(iGxi$`j{e-jSAAfyvtF?H(D4>O_-adRcbp}=Z_OD)oI9|TwhkJ; zVNK*T%rbpq^`fEE|*Fy?bY116oB})DQly1ez0jYjPS3)-}aX{Eikp@d_&(^MsY&_O69T zbN;|xe_HtN*_^wBzjV%*DSycG3tonA@X)=SnKF7dXUgCo=%zm{el=&Oob!&wkG#_o zefGCxT}$#o{uB~(Nl$hfvf161`Bq`$+Qz;T&Ucd?UZ_0eW2~>S=PmsGXUfXqojP^O z)ZcW$KV?m4cF6~=Ka1b;CvqcD8PypylFcoY-A%FWRVTU1{eaSwp9RdmGf1Z?I179N zj5Pivjj3RS>Itu6XYF@mXIdYvckY2qrFZ!ld`KPJ+3Tp^Rvpnre~^jJv&I>x@!rqL zs%ndO-^9vKz0!&w{2afL_$__c^RFpC^*Xypvaciz-B<6lP#(geV941Iscn?5lJ-hfPN1(AderWJ z2KsNi?Y0@4Y_f@?*|V2+zD#5Ai@y~A>&T)W%;SD0;jPF6zqWpvLvXEX(f7{tkPjx= z^A{H8%CAp8vfl>K2FbsMUupj8rRi@;=u(7c@gW~UD%ToAbj&ZwcfQ5bbC|j&bxt>n zx>3d7@8JxshRBu!tcdcSgTL6k*Xu3+Re-P95`0ewIU{KPnM#y>-Am^3k6(gs;rVAP z-NxBU$_ID85+P@r#IOCsov{=@^+`*vKY(_v`pZN-zGro!8vk{F+2YhB)>P(@{p%|0{EcFbob`irTWe}LL$|t0 z>vZ^hNX*gsXVHx|v(huKw4z&{u#$T}w6X`k!!Pk~oI61I)>%zDt3k3Sl=Pe;NMU;B_uCrycOCPdpzP^db!5@5rz7i^#^m{ZfA~{~&pJ zt}ZY)2+-gAcOdPaJkKLvT$(B`Kx=S!!{aonRi6Ejg|2dD@(tEV0u`mh3TraZyK0Hf?K(j+OyS}#I z)h}E4q+2bv`^lE*y*p=2^+evB)1hueI2KgDl>0BgT@9{Tcr!+H`jL6WN<{()K&WqW$)=co)t8-ix`{ge6chx{g^o%^R-yw3eIxRbvDK5YD04zS|6f+ z_?d8!&Kpy|@Nc_lb0lNwy{Vu35NyT?pH8ny?GtQn!aKi|pMA3xaQ;onll7bhpz{DY zOJd?oD;@uhRb2C&RUUP@MRz=mb2oY!XD2W>Xl`+pf6g(B>~e%fwjF4(uA6veUu{*3 z_RNX+)j{?QbgGry@Vo_w-)xcHXs^2c4 zaA*VN{~7YY)lF!ZZJPF7G-h}X9@f*GK>hN+?Sg)d=d|uH8h+%XG?NWXqN*lwqAF`7qrD&wx^BX3Yi{l!Ba+A*-fugC%&_i$9{L` zK@>k@-nn$NQ-#lP4%^ezp$RiBc-z+&zW4=;4(5Eitq-9t{2zPQ0T@-$gfF@DLVCNq zkPre1y-7!U?*Sqfnn)FtE}{(C|#uqq4zGmh2BX*OV9q_x9{C~giu8A zCy)n|xBK3^_jYGyXJ=<;cN(L=C*7mkKH(oTCd@A2KLlrv zzu5OSN7>?WUW4-Iqi)20E2|xpnE>ux`z_!uXfPT4FZX6gTem@dheAb7@%SW4_%v<_ zI}hYWfnvh1Qa$0_vZwGEI#amL-Y;^jjm0<^F#-_lg|SKqj8EpI#=YZp2Q{4|T8(`R z{nUF=B6$6Z@cI6L@a_LIbf>=3R%a*-I+U&^d_NkJs;f{h9_8AGMk?)_;0p(o^-BQl zJ_}J^J?oJb50szs-x|;x{1gX#b1pbEG!OR5^7FR>&-KT(M8??Bl_)E`TXhp|U(6ED z3;q&LtFI#N3HB;R92w%CaV!|;%R$DKVQd*QK>QEJA!584=go+L7Ojj0vc^VF&@Bc~Ev62<`Sj_96*Alq@7}p=@LkkGkF$)mG@FrqCBq;I0_9D&`;$z5o zK8#hvxM7TK#Qpq12j#59&oIVm;hC|ZNCU>CljE`9fOp0=O?f$M;$ zL-$V66)Gm2mL5{@3ww_pfw7W^e*$nX@h|To&-iU>Y&S^<(nQilX#+|;P9>(2dxrRl zw39MMImgL}{=)Z8%HvKMRezP|x9hs#^i{ zrRf-`59D~}9pL{ouHDh+2@$T}{i@>C4F5(N7|&{ZFyWu=0ByrW3*?npdpH~8#v!g1 z<&iVuG&*lYyFmZovMx=yFTE-J>a>t!2$UlouoQgmg!0w}Ttc3{8%@~na!=2!dJzAB z0Mvys3-2R63_1q)jv50vj%yoXixgg6BEiE+z@@_TjDN*AD>jJZWAm39D}ymt5DUtP zZ^|1fdnEtk9%YoI3F1ZJ8v6t%#Li;;EF}&kQAD%ufI?R{S)MWqs>}3x{=ABz`6C z4+-PMJQ)5?NwR%VBZCu&IWbtk9tw3AYbJN3`QOI z;QCV7AX9ZE%84AaHY5I<5@*l~d{6u0kgZFI?f$t%A7&BDH>7m?_cva*I z@k+TOV?RPCHOd{Meq{ zZ;v|m1Ki`BEam`&d&DU4zZ(As@?XYlJwjT5|H0cF12}F2NW7D;Dc=>H#}0Bx#wKNa zQne3|F$B>*NGr-61^*bI!Cv9lpiMgP54*UY_UDx-S0R+KAkMo1_b@NrhrGk`zwCpt zOz0xv;NS&Z+aZpD4dQj!Att9C;(S_Kz^-p;XJ?C(1I(_@7S7I?47O10{LfN0)Q^2Y z8Nfc&jr!h~sGk<~Edii^RvP;NulG9%yRFxO?=;Et#5>ynV^SOC4RNpZp&XYfxRre{ zaW3VLLMxeOd*Qf@_*Y|mx~_-rGwXPou6%9cp7Ze6!2`WCm?cNLBFc9Oz*y_;QO;+n zp6MBG?aw$PM%Hw5b3|Ik1O;yGk*2-c5@#ibtnr6dRyiyXdm{jIISl~t@6onG8+Z@r z*5LgxKy#*X?Hxch`2KKS`I!lTda-Vt&t6UZqi%CBuJJ^_K-*9!z)AX2`BbYf>{g!? z)<@GN&r|L&MyixM>bTda+dFN+m{^W`fp_EBj55loqf%$Zb@socmm+^4|1nSR`Tdsr z`hq1GV^!eW(ZF+_TsaXB5@QpTF9~P*T-Bug@L3Lof`v~9=f$LdASY-oFi*odV#E%$ z^akz%F*jQhcnrZjnI8Ce0dLzlIoV)No$(^AFpqCF4R8U^UItviGiqAK`Zy0d%Imm? z7_HiOF%8M7$Y(lG7vi6|o+a_G=sktx{G&hOvFi;3013Ln#f9VijfjnyCS{L|V}~)X zQI@lxHO91M>|f&c@3^+Ww=u38W4kd88c=4D_c^D)xdqoXN$EZj)9x99@+L#>-30Di zLdL_xK=MD=ACW(fp{xaQex~}VJPurIYmirgyV~~l+6Fkc1WrH4^*O-tcHH|NfVel> z5xD*#;up9nF$@^rRXY#Y;Ljj^dy<)ky5)|AU<9pIxM49D`_^>YMf88*bN5c1&1e@OPhS(x?{~#>wcvxog&^#JpGgK0|ELq&cRK+0&Zcd*A^SJp zgH0OaPQwhuau)7Ah9Xw+ePzt0@-ySeN*YL=UZn$QA!R&c^Wr(%LpTGBZGdu!F^(Bq zm2HLnuf+f2%R=`yd=;gRuW!RT5YK1>mNpUPbDaUss3+I zC!lR|E|fMx%Ks?<2Y`7Mo|d)*fBFJnh&is602Yji3uDvI_0M zD1$Js>G9p~X^6oL9n7#9-wxsUwnL{5Y@f9I(BI)EXwd^_S+=Jk-Tc0bi|i21WdhR# z@~Z{VYrz;=2yq77YT?7LkzoFgmz9>X#vd``4Td?uGi@$kp`K0v>GOnqWn90dFIrr< z{IMDF8!--u!5EBmK)g%#2S&Su#Qibg`#1o2C+@Y(1OJr!1U~zB{A*>*zoOLv-VlgBV7(!KLE$$2S7Cnvwm8H=W{gqB9JM}1djtTyZUXW=@vZ`6+sZKm z@z4HS?Ym_k%swFcUaC*)KCu|<(Y`aJf(Co;dib&q+*|%D;6D}Ri3S9bUtXsJ^`k#7 zeWF=U+7}Ok_k+|q5Bdt(;odmvKyHJ+6E-Jdhdu%P6Nvwlh`EfIuae)5GyuL$al2Li zmwkpDH_(m^8gL9I`;Bzry!ceAZ>{hYe49S;AA0BZmOfW`0`q9`ZO(adJTM>S@?yJu zoett3buz{}Tr%=-Y7xP(LIj#De%o+(_PQ0Nh;htyWh!uG}Z=&H<0-fPcs$ z#?hAYp7F*BDjkgdF8Q8zJM-A%?1N?earE1^htq}2h@aB*^q}A648f?pm(QS*k(~M^>)LpzaDf*c?qtbsIc75o3%r zgHAsaV?Ss5nXBv7X!nUhc`l-C^>LPK0G_w5uj22>PplV!cA&KY&iVa_z4>ea^h0O= z>;n8xG~^CWbsINL*k4!hZ%6(=3ti}x3TO*9$eZHIlO~Y=a?HCQ_@@2A*cO0)+kJS? z=4%_y^M8vpdQo=hdnIpd76M-1B=Mbc3P*S=Lt(kN>7jltCEpdw|D&LVelS zt$?0qXVmr34&W1n`KO$L$zFrL5Dw?>BF1>Kj3v%k^)g;P&WtVYjB`H3h%Rvj@!3fm zvW$iYceTx+>CfhsW18RZ)53mO5o^R)F!sWzO;d}!)M67Zb#xF-a zbH*}f%=a9)X1Jb!7~P0zjy}TX4Db(`M82mSvH|X`PbG;Ew4Lv7r--O|OGRGjw)oyj z;5skkFmR2OzpuXo$}vU7|7x9|POejX9r4dNO{gpT0nUYSA4>YT7Ah1{NaNz7ES1V> z@GlM>=!U*d1L3jbFX4fB-ySy;gxd|ok!Q?!;GV!3@}!B!jRfI}G{+0nlWZZI>z0nYQ{o4Wz@H|#~5zatiJy{?DK zb0vTW&V}8^`TYQc8n+wxfbNaB7={HLhxtc-De7OpBSzm(65qrpin=$kW&-@|0z5l{ z7S6yqX+UrR9hhgkaNzx`;D0>hf^*js7euYFrj&Q=^Sc4ZRw}KD|57N!Wxx&GYk>1> zfqKrxWi~&zXmv<9iY!=JwB6b4$(Jjz?&QgN1_i%FaKdXG51=mIGvm-_B==y z8xvB*h&#!m-Yu+egzP8Y*&jF&?*Iqzy-_wfflfKDCy6|O4jZOXHt&^f`0d!Cs30m|jC=wB~cgzyW29J#aq{K%V>_dmY)3F0fC_ zy5oK`T)$nTQ4{g!#cSeRN~*Y-nkM4X(#18LS0^Nk?r}+?DB6Poc=wfnb$1vi_(L}; zhHRAVCtS57Fp4@&13G|xpM6z@H-rk3it)}_8>pH z7r-8ObrHeKkh3gsvcGcGYGU=lzr@+3R1pij#{>UyIFlBC;o4Vsl0*dLQor~lF(5u! zd=B^#`2OudidY2Ll$a_mJV+1&qoRzme<9?XAHPNW8pa&7Gtdu=a_I*EedT32X=?{) zVUNdrtwl~0pOv#LE_ZT~kGxWU$S0UH&|;o|Yp{m^ur65c+0hTUcSB*NUA%JT%9^0S zAhv;)uxb-$6m2$&05hk{2gVl zAw6PJ!M_&ocT{QMf^Tm?8HoQu82@qYg~Th@n*%tf^(A07?DdU713hTM80zp5`4^KR zFMx8&I!t)2SWC2G-vJHQ#9G!tS!ZEBjr*m<0sl*puVG<9?+yk3{50!`c3v50#uB{& zT$I)6buP&FL4F6|I^aI`N)e^YlojR6R}dB6t|;o%t1oKSt|O{duOX_}sx7M3Z6K=G ztuJcTt0!vXUj2rRBvh{7P=wZPD2ml@B&yV`CGzFtz7KL<@f^-;`5k^^_Xlt)=gMF@0MBc4 zqinrVUd(ICvU0w8Ir@TTkU#M3l71PcLQ=+p4|4(j0;D3}7`UhoTwDb%h$|D|o`Amc z7|=a@-4A&c3_6`KFpb~x9p>*unm(Ul=yS^A9=}Pyytk3o&`!mJ^CSTHkXTnoT{!>E z{dZWe4XE$;fKHen%!{U=Wn19b=D$9$-rW&%oYjE$p8yvDnB&*p1HeZ@+YaU8|MU^UQ<=&hB1r74$a|3T^-O)FFh_b*RNS2BF ze?aDIu}71(1LP8SKfs*l!?*Xy;1Toz_uy@8=%CelqJC`vs{ru;^4bN!6x0_1BrR;{lj%UE=_MZa_VOFh z3#@7ac7YD*ID_Y{Rs%YtECqmjM{8?mFn;C*LESk9;2M}n)E|C;hC@h~2ebz+)`13G zD|`y@qV|X*jZXkL$FdC28EM!#4D5T^{-z-949L8|xLUPpDf%GTp|0?4u=){rzXV7{ z{bNz*MSu#H5JiB;zT-vQ;NBA04%rI*P)U4yJj$>Q<(UjQSr%o$sLKuIr0)d%c{Bea z?GU*X9lRpy%r!8i!Oy^dE#%(@PHF;L060(ZyfVi8cbseZ0#FcoRshaWb!1FV%KbQe zyA{V>xSV-`ah3g@R?B_9G~5$R5BL~$-wAr$0UjWatolM8xYB>*mw@e@byfR-uYm*32NOp-fQv7Hi?@+(0HBS+9FNbN=DWWJeJcaP z@y@Y;vjDE~qU_TFo``??id3n>F}>mwj{3q!%Ca8nKL{`w^e79R(3X1Pe|6xy;0I+d zE|iD7@yrM9oy7|mXIucLai8iF1FjJ7@y(1<(!?2z<8z zentZyD*=B2KNkV?6*b4>^QPIS)93LxfHsIf@GbgS1;f@%zf`#|+CN1YWj*;1yf6Ln zX>iF(2kI~F0P22lv;lElSTN*MNt{~(s1H3a&dq<90{Sk%yXxVc9RKEqyQf3Ba^++^ zXyX4(J%X`cXv>x@B^`i&dCr$FA7wFPC2=flj>qRs)2EZVmJ{Til=Hy9=FFKhZz|fK zSnqPh%W1G540V*9FAD$Q|Dn13ayjPG<DJy#lHQCW2`5a>dm=Zs9+)LrTQ0$|5V^U8~@M9 zLb-x+X<(1$zAs!epZo%3&<)6)4j6OWa@?JLJQ)MDPe4DRe^fz0PvHLxKolSna(ozM z3D-*}Ay2)YV}uj{<^364pNIS3BRv{09{Ev#cF-~Xu#cey{SC8^Cu5*ht5%u@4I0Qj z9yvE0kTbvm`;gc}SG5KHa{-^+&y4;e(SQg5=Mb*~Vu0uO0dCM`bAW~}I6K3?&N?hC zOzt5@eK-4fHVjZ#@y{v!wPgH4uF(blX{X-@{2v8`;XY%Ph5&vC+y(Rn(6%k-vgAAP z9=54$z4O`dp4n6M6~_uR`fj}iymNn~jp(BbsXion;G1!6rJW5pcO%}jj~9ml?!AxQ zNH{JF0PeY$&{aSbpgbS~K!3VhphIi;HCVl_K7=n0qyOwp+_OH=H;_R8tQNpC``QPP zN0*SN?|1^zb8rr(E&oln>HIUyKKs)pLP}@~qI9V0V zL+Josk5Qg6Ps>WG*E6Kk1OC3AXlqQzMJ8Gp)6ta@HElHV2#5+WrQybuD7R+79fT1} zRG!e_(%)%~CP2ZUMia=AMkx=$EG0y=vZ+R3AZiw;ZmYV~pzj!p>%DQmI-nTtx5xczSRa;)<35ZX=o@N}v795G z(-Cm^L(igbB`3uB4SIYKajzBfP@k-J5dWly6~=vDpj&OwivBK5LAQ@V*Koj6zl<6*4@{D$|;0N-%z-vrPUP#4dY!FNJ( z<;oQResRS$X)|Cs0FUto!kz5nMPa}SYe}8aCb_?4edvK7fR4)n>7c~}Tw9GZeR?>* zz7c8egJM2U@}kn%*U&&8u0`Olk@#*$l&J#74uOy#-Yg^HFZ~yeIxh;Zuljqe*xt}4 zOJd%vHO>n_TkavrHGVIOrvLUm^ueQV%?8MgDDXrzfB}6A_f(>+%0B)%1}xeC0cfezg4C$f)c!vNP#6$Y?Rrk^b5LNdkm zD}J4QMH1?;7`j6P=$j1mlC3{Jfd|-k0$+Imv^l@5KKwuBCGY@s6wY5vh0dBkTR(U_ z0{};C_bWjA7v&%SpVInC@(A@x1#Cf#hBClF_L$*OHsD@$Ujj!j$~XCyp}iSjB$Efw zA8_qj_F;wr;Kd%g64&a+0kUeJ|5Tnsf53hEiDg@hXU^`bJ+BGy!?n!W?LTugy|!0XyDEJ-UnSel3zt3F?LTuChE9|dHXiOB{a>vW zf8n%!mVSux4RtJ(ITqgBtCauHi&g;WZ)x;<{lC!YS?KsY&w%!{6(0rY-&{&RzUIJ7 zH`T`fT8;rxzx$})3}7I8{GSOM06*?%|DAB=yzT3z1E!+=Z$;foWm5I*ODYC9HijK& z4{-I`^Z}rGD&P>>{(ISkemVxA3q}H8TqmF9bwThk_aONb^=XQlXU_*eo&5B%&WpCv z8#p@+$kMhTls_K)oZar9f$rvy0B51V-NDy@{zZP0~`xxZ9P!$ zAB(j;HJ)DM>?h1I08iY;&;uFMFRR-C;;|3+2j+FlPCtqP?3?6-wYXeY%Z>iDX!jkU zH*p`etWAHt|6TY6c6+qO*;j}G`1vs=S{KOrY4G*?5xAO+^!K3sZ{Pv$CzjRepA0#+ z33VHTx{O0#Fc$Uu9CdC3;2vpjs!y0DmC`v9U2 z9KW7{T$`=dZwF+-74QN5LsM|=FyK?r|4p(Duzw!{xD7}Jtl$_KZK0^zAC!dt$32|6 zkA0Sx|5&@Q4*R2VFN?ycV_85w@c01K?EvaX*|1sl3w$HQ0M}6g7fS$a_q##shN|8t z>yK-C;A9}qTvw5$=@0s{{qM{`SPRZ~f(P2-{$14dDqv8CXI|SZ=vo;#k$d2w{-Z(r zhc*@Ax=gf@Ubx5DL0NkIk2+A!$H2cuU3C68SyAs0+}i=#CqYL1f;(@-yde0#5$G!S zfW|qJZQ*aMC@Ji*^fq{aYc;bv{nJp7)8_KcSXW%jiFU9A^iKz@sa(188?jCZ_^1ya z*dt{-)@NhfpED!x zEo+CkOC=GTq8ZB$U_^v}5Sifb>QKK!yPpNxvTr{I%G zlOf>YRMc-Tc;3fUIV30#aQze1kNw*@(7&_!{pU=xkKz6U56RN!eAh!iLmY=7(0&Er zG~%(011}WC136q><^DSVc+UgWeI#&m4?r8h`{W7p4^m%7yBP@FWa%*=>URe`-UfAh zNVcPXzNk0jl~4vA#4}aR>;F&Fh|y3G@iO|djpDa2JYWN2Fol5jy8&s44RHr?X?EaS z-SAz+CA2AoSPTCsP#Nmn4L}=jD(c=N6B{VkfmZ_T1*8y9Sq)Hb_Wk>@H%L%MdV>Ce zX!Fa|I&DKbcgAO)C6BmRI?$m5XoA>mR<{7h5I3j+e*4IH^I6{aglAk3lcg)@AB%6L z<2$Kf0Fjp)Yhg=sy%^1POiQu)=fi;U4wCe>Dy; z1mzWmVikm;QWK$X{Hf5j9x8kyW(v;B2R^b=*htPMMDGV*Y5{7!6 zgrP)L;UAd$AM+mQo&ufzM|Iu*BRIkJe5lhQJVzVYI_LubcugqUT>;QzA@I-mfuu(| zWb;+X0n))0@W`0s5ez^31@z0{01)(zEjT# zpE=ir@9gWsZ}u(Wy&ztAF1aV%e@C40l}QpjRwN4Vg%5<^Pj`fFYMjtbxhZs$t_c0t zhlHUEc%oWM;h#I-)4y*p+yXs6%tWS{aDw~ALHj$XR|5F|XPloo!W8=ATxg5kfq&xs zKJd@>mOPaIbDgSSwINBG`|0N8vA zp#EDWJ@x*pfe&t1m+8Sh6WCs0BQFX(aeue78TkN_*MvS#sPK8GgYf$5S7E#LlCax# zOSr6#7d{K_3Ey85gzv%x;qz;f@LrrGyp|;kj}5Jf zp_`?`PYFUl<-X8Qju-j~*M2z@GdfY%b0 z{xcGUZU%T@8qVMW9rH-*rvfJ575WhegrR0z;h!sy3fmMtxd8jsA!M7Yy8vMEgF2GvYHE@vXg&JYwQnjRwxk;WH2E(Pp208A{t845FStaxgyGeAJ)gyExFlCZ2ef{>l>1Hpg$`Wt-1ePfKJZ!wegJqb2mOCz`^R&jKcA&6 z@ct!9_{{_s&(nB-wg5a!|0c?R zj&Dk5sHQQ?*tkQ0KgJzrKg;$7Jx}`pXY{jHqp0gUIcdC&KlneNU97DwJEMFtC?8_y z${4ac5#u*JvvQ%{?Lq(3YzLtKm$|$jwvlA~4e^S0qs$xl=M)Bi=y}CU3(v0ogv;2U zh08AogzN8Dpv%RgUEdWReEfgjA|3^q~ci&J8{QFBo!4{2TJ*7rI*S3!k17 zgx91!!gIkD;j#FR@K}~8JeMU&`g^eLlkNc0+6a^ZZqNm&6H-s`{TX9{xp#%%%vg*A z&Ix_QVqs`EM)+6$K=|h`DG=Thav$Fg2oi=o1%;sq#+N7ycwi~&@mSp#a?c-iJqY^W z$9XFE4b2Qkz+*wsdm8A?c-S|A&jV;*9r5gc*~e#Av=?%XzCOS~G=Oq226$=sIK5fw z%rjuD31(DU+_ zP}(RE;_yu4Sf6J>loC?k&`vdp<3t90c4$zBWpmss0S!P{zyX^PQG%mHM5- zPW0JZl7;)yYp~z!5IzypgkOukLSMF~Fcbi7$Pbhclns;{c?$}Cl_o;hqMz^?x>$J5 zI40Z{Ul(pmZwR-A=Y_}EXyMIwgnuIRg-fGd=K{^qzH~=!S;sZH0k6;_v@mW|$9#og4QWsWR^}>fGUJx_@Txd0no7u48}$ z+W090NdHXbChjp7PJ(ZZ`oi`sbUV=9bsgYOWsK*x2I;lQXvayyeR&+(`T^nFZ#vq3 zd!a8;QSyQ!uNC_dc|u>LjPMKVAp9ERnX)yMXQ+#iS4i^^(7Z$?p>Ohu(DnaS_@NK* zn*kkwV?vGve*{bcFg;zp)_W0t3mJqxU{RMxc$sS)K>tDf26+D}(p|#C!ykF>b)oOm z>H!?|1FpEwZewWh+a+jwag_WI?nv}$T zoseTg(wsp1fbl%(GWe!2b%3M*zkqxhD$N3cI7AG3g8-y`8pToCjWDI>!cFUwv1&|9Vk)Og{iy;4stVv z-`N5D&}R5hA0X$-6<_dl*X1$7{-8?h&9L$8ND+?P(C*jXML&K)xX#)yyuSEZ_%-XS z&|K>1pf~&R+#!(prG>6sE#V89@7;5p@Eq}*@QR!(d|G}c{Hiv_c(T0E=PM%h2Fa7; zmCzzWU!#T4b@^WChHn(QaYu#Uv}>R{`U242Puhq;`=1^NFOD0+yQYG+sR8zDf&UxF`TL-hG@0OQA{!WJ|p11-7kp)~43T_+Z@?}yBMUo-sX zwa)?Kqn{hmhYww{g-uDj(6xOioI&6&>4yz7}91|7_ulL#uT~X-eYQIk1 z+fcB$(3P(td>ghF-rYtD_o+LD>+-A67ZcG(q@a&T6**w*lV_w|qLBwUZb`$taUC{5 z_utT#XP*~7BS7P}U!X4tm%5;#KuJY5@Oy>J2z{M4!soL|!hOOP;qv<}yc4$Nc7tL` zf2{SJY)(AduR!y6C|6Kl%7Eb^At9NL{mp;(x~DrLAWuR&9U>)eB||0X~;z!(wwVUCR|Anm+4 z71v?w-3)+@nBW8;?Rh;1=K)qLHX=E1ggGIf8JC3L$j!pjW(X7iQ&^Q(Jon%fs(BG9bH_{(<9ftc4^>yaYzbR?pMEU{&FZ2bO z2AzPh5=;m5s1GImL2Iv${V~3Ve$F}mA|-|2+qH%FyIq85pE1I9@&@6&_BLpU{sFoH zfx3aiHqu)GwsA+$*>RhCW*XptJVDYQvcY8?d4M)z%oln|pnSU8Ex`*FE`yip*Yu645=em0b*SqG+@t(I2gu$R0C)i&7-#DY z+Xd)+0As+Ahw!HJe-GgIe>X0Nq0GSXq-bJZ(ts2 zDSQgX?h<~jhd^IwDCGg_LccHSGkwkLfqLb{HTtk{U#mk%)9>|-KhQ21d-{;1zrmk2 zvC2S-$)^I-yl~ujUD)r24oKLAx&GZLpfg}!im(OP?*ibOJ+9gB zQ+U7*JRo5=&v=&4z=lX(u-mEJW18|n=^GRsLCOKlBXPc10%U@5{y?6|2bfRx{^1~G zK|R&}Ed73?%!!M77DOGnwJ_utfXpci~$mH99^287%UHy;-PZ}b7AuT(mH zeCXFhI!k)*1?@4{D}l7%uYlcN(3SKjoyiyE57Hg7zy!=oUQlQ+)8GSRKVZ}mI8Jb1 zMPCsksB=b`JK+3*`|r>dIKK@2U)Qv&Vq3u4R>-nFW_=lT|GMOH$;WyF{oXV6d3yu- zfH6Ed55T=2&Hyj%LOd0{80!E7klz5HJeBf4$LuYCw|67$c_vSQ?h@GUCGBzD#3LDK z&-+Fmkmoe;R+{W56#am*K+z9f|K!}D0`wE&n9=AnME?=aAA2r=k67vI($+vagZ}gZ z?MixbUJH86$143f-?0Vtd1##b2GZSBW#DBTa1sZ&2fPgeZlpgj_@W=u92;E1IgNUs z$KVNnw_zjwNp~at&Cierb_99tVfst{kUXMZS7ibFfWJY}6*;amjvYBxaN2}Ef@4UG z6{#zd7u;tZgC3Y)nLi+}8e|`^ij|~Y8~JgP{>nbU6LI;Cwfmb0uO>iq`hZ^tozo%j zC(uWRw}7wpxSkA{3P_+G&o{gs=x>z!pe<>y%6*#yDewnMmh*!)htL-sd<^}iK1iJp zv?tx!FR+gwKd5?wEzY*PArGJzN;!aWBhTaw_7T(*!0Okj*jwQoZgln16AM_^ONk==-R_cM&6-a+G{lHfC2?{^(c}I*B zNPmtQoHnN@0Q-Uy_#(%KG~qSyCpmAZw(;#K_bsHa)5ZgOdmyja1@wkvs1Mg@WFIO9 zfS)RWBa;4l{}aGZG2G(3599eJ<6eCLeFdd{!dwofgDGu{ejVrK>Lvk=9!fFCfXA*{|Gy> zu2DNV_YXd$t}>kEMtRS3&OfJ*^mSoA5!boiqk1;+|Hc6HW z11SGZv?u-9o@HBBWxi_Tp&Xa8oc%fExv|eT*#}I&0Z*{~gHPnR5N)6KWa@&h7#EWM z92_K?k^Uz1KcN4SN`J56%cKth%ZGYwA|QPLXM?PRYWK&rdzjDo zF`MR(VgU4IpQ)_(*7@%;`hx?9MF1zk1GJk-`Wu2ngu@@lg{@J?XFs6Q-|8s%;0W@N z{fewtd<0~h$2m2QK%0=X@5f{lc;9NmuGw(i<_oc707rFz zqw^}Bl7Y7x96J+NXiIMcZ_DsZioB2cclT~X6nW2ao+<~7{Q$>tvft)dopM{w0jlG5 zjM*HwD)WMJ3@B|ts;;mfykN8+8EuIi7s&TO_N%makvczU55H6Ro2Kb1H@X77gmyjh zKqBU=Olw}~~=yKJkjf0_hNx zS>X-D{{=5Uaea{Tv#0*G#(IFoJ+^`1F7yEQzov1d<*47exb69K!(8e&=UaX|B+i+H~_fY0@EMTM;$L`Si zDBC4q?pKlDN?*V+os|0=8>%)W$vYe~5bz%ME0inD(+)uWf&B;lMO=*ZhwS$mJntUX z8pv@A%61W89^=ard|XuJY0e9uMBeZuJUq)i;Jhwq^pmx<)fmup2;g(Tm$*I&G+72Z zVLuSkt=HGuk%{K9eJHfifIjrmA_%5N#N zNq>cQ3J;i#`ytO|pHJHJdH#;FLxOV6M1S0u^jGbF)E(sbkGenT@7tnp8rB`ju?fom zIp}Yi*HLX4?~?wU4+3oGHFMlQPa1gVn(K?8!7b3@Dgm^*qU514#RWI2%EIhtYmSyP)D@F-xcZ@=ls(kpJ=x< z$yWL4xt-95=G_2K)h{ynR;HJrH?;(g z6g!{6a0d94^8jWKTY~=FHx0J-+`_X{A7On0wjbDg=^HBBEa``KEpvm_ zBke(V(wH*dnl=OQk&Uq*;X7ozhs;oYi|AjZ*nnyO0sX-P?hDRftr&bl6@HCD*-ZTv z%S`?S-4(kP_p{f__@8FgdG4e;Xj&UQ&=-AuSLos&;oJx9|8t}}fzDVrZW##}jJ;yI zp}mg7wSJ&sL(rzWeoigL`M7=WO!%xd|A9Ud8l-m9eVhm6MzcK$V^{MLe zKE{B)wHpfC7})*51DyAfb3PmkfVTAcl72m^4q){0u|=Dgb|Q7$z64_Vl;5;eX3-?6eIGX#S)-sL(jU~l( zuAxwMz_*0s$y>taENFiQG&}(~rnG1Jfl1pxbvzY#jkM=8lICdlTF_pLF$96UAn9)$ zN2udS+4ezy@(_JP*ar|iKcANd-A}bk8isJ3Vk{@l7LaE~e{YUwj-c#eX7{s`k7B?A zG&kvhz*#H$&6_{QHU>Oiqs-NZ<`*t2w+j2qNy6>|pAf}#g__x$!2=KQJq3zVx7%4xobjV0!?d#kdH zJ`Bsv{K3qtl$|!i0PxfSHh=DmOdW7UCVMAqac{1?pEpE!{IozgTuBo4my%>(V1xd^ zCMF4b0RTDyLHY%#GG5vXj)T?&(%cB(1?mCjydq@*bxG)vMjkNgfX?ghrTf)={~Xqy zkhb(u$f&DOSN)OY1SB9m8r6KE{iPY&UYjKF)D7}o?){AO+Dz6+QAWry754!0`0{&U zdm4NHoCp2S0Fbst+U6{5L+}HkFMzRsH_ChyKTzhI(;xCd$_Xh0lA#~ce-Lt?<4Nif zsvXb?HYAtdFW=RbsY<&K`>K<{?!oMf%?~T+GMopZJ&SX~L(Fexr!&U@+I}D4OU?rT zXIv-oh@UL{Li6EzJlFAhwdgGAY;zLyf;~_I+P@8Wfigh)0~zJKQ3jx$OI{%TjdFl# zURV17(%u$p4Nd+7oCi>OfM>T+OH=fP-vaIB-UUlC>VDK)(0^T(^ha5dHra$TuX*-W zVgPh)0bu-Z%FSKCE9V#VZ19Jz&0qMIsU&Q6UlTT`u!b`xLD-_r+oJ8uJ^RBU^cNIg8|T+Yr;>e?dAm9O;tQlK#3PC4}SrXlVzq zKb0tKPf{0z9Doe4Iz`$8sPA!XVAk&&b$%_zjT{5WzMo@9jQyqE-$;MRe)eKgI))50Wo< zjr6Ck$MGRR8T*^j-oyjA#91~94K4WgOO^ig z`ODV*%#ic6B=9y8a8JeG#?1WXfKwP(u`dvO0sN}hR%lQ9OIrZwe-89NZKONS>;upr zSY!VndEICOF!Bfb0P6m7{#d2EoC9E*IskM&X)gvJxs6+v3cD}&dEuPiXxgnZ)QSDm z?}E;8Re6z`yWy7)90sk#`0(9ACCdxRK|b8nfSd&AF7N`bG(K2_YLW< zFIqyluUHTICn`Ju+B-lFIA2Q?&eyRgFu?IDuAy&m!a48FMA0-hK{ULLy_~=s+%Mc( z(wg)K!1gcOI>!#YhcyL;V@aYO(uKBR57>&e(XV4qA-#dV{)JGshxAkF^J9?x7oWkp z6+`8W<=?naE?DPtQJKb&!P0pz%rARLjmg$$srK-vvyLx2pBw8z<+x*qfe`}2w7y<17* zP*R#$w|lQBUACN@^TD^zpr3za|6Z(FSLUe5zgWAD{wvqxYny#1GYkN4+#8l_W(4}r zW_ZTjw$7heVob#`7Up0=gs#s3;eG@A_T0igt~UWU5`+u*!3i{WxUA5g^e2!{Jn)PU zo^eOo9{mCB1+*DTI}-Z>@PZw9gSoh z#gFYc`>Kr3WapoZ0gS6=L%vI2Af!8FG6vMBa`vVH`-14|H5M)>uSwebfd~8nUZ8Of z$N~q@Px8PeMGmlEAP;zg{vMDGe8!o)!Tabp6r?^0{s8}k;2j%5|8Wy1i9!YC zKEIsv9R)d*(eBH6|51|u@CAkJn)T$GX5Y;W1Hju?99zrwZ^$_!!`!O;iCkA}@Q>y* z`eJWk->zs$(?YisL@DG!pw$_L3}!JS0${;fn2bS+7EknWTN0B6vddZFY4nMNMG zK)r$dVFV}qj@S6S3;1H>h3lefgGQ3}_%3Zz6;Y;#^b+J5+eFd*xu-$)oPS2LJyRYy zEMqngygQS5V7adb@Y@c2lAaT7vV50rV%nV~aWy4X97{|QtKt*I9-PPAO%~0-2Su+Z z2v_KRq`Mp&fVNIZn*d`2(w_aj)FDZKjvsIi0Yq)tCCXQ=W~4v&@T2bk(4q3AVjIFd zHtO-O{e2($!0VHFXig_%wa~W*g05T-!ZAP^^BJE(o^!z^L5Dj5&;e>Ue_t#Cok{yk z;DIZExU_T;o0cYyC#Q%JkO!5a6BvwrKIyLl`v~fL_81RRXLJV-xZ$1-{80pBi2iZO zqQ|UxqDZMSM*9B_T4$W|=UyN5_vg5ey8pgR%KWBZvO^3x;*DeajIz^&8}5q_JRShZ zIW)ARvZkA63DBN)z-`2N^*VLMfm^ZSXi|!}oth@%L3@S%>Eb%_$8i1k`^ln1ToP!S zC<=nM!8iwlzIo8j^I#lM2z`D9To1>1uQ7P(lh`D&_i6(TZHGMhAmg~sR9oP3 zA;tg+#3yi>sSPc+Pi}9>fC+r&?TVGen!Wp_%)g$Z(4RbDgjkgyeosgd!ypSL-b)nU z+)War&_9d?Oud&ZR^Z+t(E0*?zZ?8=3Tg7v<@h_INwaW~lY2>X%!z(IGo2K+s;^c0 zr=VQDOqF>wVXee+)NsE;`f?HvuPZND!*9GGe3tvb51H#sxGv!?e3j{kPC1ZiUjQ8U z0WRg*IpFd;JZC!SZv&pli|dVek2o*!R!K2=<}9)Q-8ZSKK1|3I@{Y@l|wye0vIcjYY`Z?a68u&pm6j>&r}A0zTv>(4Hcgo zjTJ#3>j^x!fqpm}_m2UN@|irrc|?PTjYP*TT}7RG^+oHpABl)TgGKL%exhsl9-<@0 zZlCn%DY`@q6dfW4iLMd-MUVaf-0w4RkQg{*s2CVEMD!RODc%_{ScFDIigNuTMc2q7 zqGg*8(;@pM{UMWXg8o0^yPu&P6>&Dr?IL{!W#Mz&|KYJ~{n#FW2gXXndK{}0kc)_y zr75Sz`5{gy6SDGQ#WllNm+%Q>{W{Q{Ysa`AH+4xo&~|Bq+T`!Jv`%0_=rca)Ry2}%F>%xpt=P9A9g2CSu;2VAJ@ z?W}Gi{TtVG05|V~A2_zml6FJP*0G;|o9o0t@9O}pJ(KItuHcMVwOpfOlOOftnuUkt zr$mnWhx0yYYsUekKl@xg+LR7*LFp6zk90TZJ(NrO|3NNQM>+2TNdJ@Ob;%@c1w5#6 zGp$YlFI;bn^?BA@uek_3FoWxO@l0|2)_h$-Ca-#BE?I!D^>%jJcBsQDmG)@!+Cz`UVMeg~7#vOsHjlfwF&gdVlZUJ9dYpJ~mm%32SSsZ$RhMs0Y`I zv)yAID&tmK9>TqE(I>FoKXcy(+%5#9sCe!J9D5?qKH-(1J<9~Z<@!yo*JT{llaQn2feLupLq3Mj`ttwaIcw;673}P+e@5A3 z6&{xHfA1i_8+0FoI&=W-AMxdVTG@cxZ-7twf^pon5YN-M=4+&ty<+~qd_ez%7?iOM z`zoBzqFn5!$OE4|t!^3a+oJsx0lo$S_5$SkUC8po~Ymm+O%co7#Fd=-x_QFZ^7*3!J_KoZbge2e^QI5wtaboIeMSw}3CWXUGYF zTsKNSd6|H3p2W9Tf6j8*5VDWf>&+u}u3 zoMS}n-O~&B{2jR63HiDKeDMi&>=*I=tnv=*1LaFw59tq}zD+)e$F&VyV~g}BfUf~} zdH-bszVQoqpeJA`U>d%)6OagCneGC<%FJK+U*MYkK~d1ZD*FFQc+VsLM*rg7S>~s} zYhl0~;92euBJr((@1bm$AcMXDJmL?S zWsv-e7ZkEKKl=NQY4?56C%tVKmNqe(*Q^)6;KKgfDhEDP``l7bEl>Msy zmA>eeykP9H!MHHMcQLfJhJZGJL7>Ad(1&vPrL}$eZoU5zoX6pNTz~XFAPoGN z7l?VT@y-7op;^BGmg?BM8f1aA6TD0wfc)X!)CDrG`N;Zc%+fU=50n7?UwS`)w*Ec! z{w$5H*RSwe#M>_WGTXh`yQk3SgmadUSE`6>dpwY{p7t;C9$==85mE#qzw=eu3 z(SH%X^+wrWUO8$%S|H1uemP5k{{Kb)zcicvj}qYje1$gnQf7VhOB(}1|62jb_UZpW z6FkG`Ez~{1Ub_7kwuTuP`_XSL`*?*gpw7XLfE=Jb@TIhWyqoL28`B2$3ROM3C<+EF zK^q;Q7UKT|*DR~8f51v6rRdvVuCo4-XHF=$Dj!qQwnPF)ViXaIe77;DrNcgXnT>~q$} z0O)ED=%Ct%&#*5*jZE`3knx>y|1yB_$7jHn9F(=%)E}~bjk9ZD zj)7c$xdK4nsQ~)uF&5WZ(0U8f7ggHRzkDyQ)i(cKcKWr!0AmN02YuH9XeZ|Wo%G`| z+KkUbM{J!vwtsEtomoY=Ux*HN0LB=gk2ZC_Lx8WL_ZPtr-nj9d;E{@&2gsE2Cz_}; z%QLHSFRsPcmD#(Z}rUm-w!J|aWD z%#-q0M4b9OA}in+BI4BNk>4gnRAl1c`QgY{GUg*HrE8IQNvxo}KcZ49cqGa;kqeo@ zFNu}nkdMJx=CyH^V(@&TMPfiSFSx{2iUJ8zH4yB_CAu2 zR__b-{?qasGCXg}D+5uB40+8E$Xrkue{oJa4-Hw0yMkyn;=< za)ZoUnDWYfB_9o%L`U%fE$&BEiULh98^68oYJ|C@B^AH8h zN7)+l(eeXP0mgisS{<%OFb+W)b~#06i>m|5NDaL|~~$JU=yyhQPj z?)VPuf4BFQUVB@meR!)^?5cg!2SkLH*wp&-^_ML@tbSM71$SH3%IDY2%w`0q6KDTP_g6S3&N5$of>HN7@w8cj;h2!$a^$Aa# zykK3iT^-67ITi6?abM?Sl}>allXKab*0wSCkG76HHLriJvw6mB8>I2MGj&&hm{BQ5 zbb-FDW_TYxR`JAVJv@)jEc1Sk^NTbA)sj5p+`DU!RqynnLmP*i+E~kOmWyYu{j#S= z`;%o$4@^07^1UGY>B%?eX%eqDU!T0PQBu`)_1r7n)~*~K+hXvovuQfbzDfxd>J-ZL z?fJ>QKCL#|Cn{j`JLx`A^@gs9($%&4YJ_*?TB94p^ouEBch+)nz3S^OPg#>REHcs7 zrDxBn_49t0(yYkEH3faVqia>3c4nY1s@>$;V?4vAUA~b&d1m1BT8VBkDNzL~H*j@n z7g~0%rfs12<}Xf9{z5t!Ab2kU)ol44PEh-Te)K&i?+u~Eln9{*Hzm$>XRDZ zMcEzG6doS?i>32Vjuo_1G~Kj)!#>+KF1~1krqv5OZToISi_oa=q9Wq$hK|^3*Rsaq z`PX|zhyRqrkbCX7cS_cXY#x5r^dYMBH4yuj1{d>qf_p#S2}QjxP}3uEM!CBC0_1#lZ_sg!Oh_X?4%S<>t(_ zi)OYCU%bBLt`3VoS=8;@-~0Y?q0ET0U$^L8u(p3E%TIPY%-0r}XuWjCf~oNlEe2ek z)NyW&u*7YHZu{IhJ;t%)?c9MUQ%(+y-?l#Vd~l6U_6N7@PW0S$H#Ri+_XX>mZ{EIr z^IMx?77PD8d}_e@QQe!?SpVzQ1~Hn6U$iV8)nw8y{bPTu5mx=ofVJ!QTLf6IYOD!p zJ84gc-5pA#U#N6HVEOXjF1WV+eDcgbUGJ1H7+1P#S?}*X#>X!&J*`G{-%+)K>lF70 zv5hXgvEmQb+by=VwQ~N+`mY+nUSmcUyX!XA(I=)_!9&+Pqq+^euOHiSRmA;<2kd@K zwESw~bXR-#g-750v*v`-MS4Z_-o7Kw%B_m4YqRjTt6D_V@M`Q9KhZtREi5L;E)lP> zj>|Rr(?9C{aWYRG_bI)qhgJ8iQGZ_3wFx(G-tM&g%wM4+90IFtZ(O{qSEC_0mVCRU z@R)mNB2q?;d#l0nJ$o9g|MO4h#+wGFmmA&2q56>e;kAN_TWwu1*`if=Sa_+@rNa&< z^%!B-qieN~-_rGnh)7&nPy7bb>|SULDwxxTSy3dQZN-ad5u*I#{8 zJ-GSd)e9Cayy|S&H*(yK&SO8V9`;wc3Kf3QeiGg?>~MI&PRF*6|FUiTFl}tww1o6B zyOzawEmh!7g?XVPwhc=5xzlnqnB1^4Il0_#?~fY1e&5I<>$Vq3J6I(CV7(#--n&+N z`JumlyI-ul9uT2T zSz7l^yKh`-{pEAdrvBM2d1mO=cYEw>>15k@)6j^c zOTKQYoz~>=2k)*}Z(G56pdq4EJJ)6high>qKBL#7`A2Vz*=6WA?Z-iyh}f`jH@7Lh zu1{%N`9hh(BWCWOp8WIlwq3g12hf4G^nL+)R%Y9gq%rfR=vrU7Wso3 zx_VW2>slf#Y{P)o1qL3dmAEo6(kjv77t199LkEQT>f%~O>|WaYoQ;KjXl?Dpe5AyC3(1wc{jYv-c{fBTD178L)}{89^8B&N83B8|LQq zUB2=8x}TqN(qqh^n9y%8ge^S0bL7nZj`d;&?)bv8jBS&ehdM5wG`sWlgKvFeF`?Q0 zavPn;H~6Jp=_;XzdQMxFD@WAiywUat!|vtl+NaUR6!&Q%6N4fq+Rb%Jj@}kkFKA}y zs=#q=8bqD53JP;8*yr6g4l%yR+{g5t721E+fljA-?VZ|Ub??r}mx@GA@@W-uu1D)T zj`lrgX!Z{I>qehu-=l?vdG-kNT(}@PUyNkKps!*Kvmo^(VRk-<{{luuG))w6wm(A(&jn`@0R&Ha&^j^0Us?YWvRr9z@+J2X$ zk%2KuV;rr!=LoLjt{Y`DeE9H&)jwXa+J2xeaZ~gz-<)FDc)Nv;od(;c_-aPQ*ce7^ z@9{(XO~Zx_+dW|M*$soy!G!vhiRu;DdGeh*yNZNz zw(cDM?w8Y=9v&R#+4+a|i>%yRZoe>rUz~ls>zzjFJ1SXx&~I1%weyw)O}W=2;A}u+ zt4_1JYJOaB;?9|EHNv*nD0V<8;OzA$O4ncKab-%q>cL*YX+;;M&fA>t{kvz5HY-_p zMAf>U=d8!Z4cuPCDaKHH@nLP{x8_@1xRSDe-(ZcWZS@83+c*}vXi!KvFGtkfhFeB% zD>P(a#hXn!*jzsRxlhl*7dKj$vF+J=V6%#aLUvsQ#iWp=xe=qM&56pUSLX@p_5X>CKYt> z9O@fUyYd{Xf~~gq8!b_MWMXi=ufFPjee=j#m2~BPs-EL=luyX@Db+@Pw|Ui|U|E7? z>z3^;lD^ba)G ze~Vky{QihWPSz0@wmQem+!k|TJd1a+wRo`JX4vleWfK%|bRU)_gW@1O%N+McyEHCrSh_8= zhhh2ezi(Zi=p(8gIeK(lV}};iDiyCc>+{K9uJ$f!TnF! z#)XZYH+}Hr@4xrj`SBkc;-mVW88rO+!(PoV)|lOC=;#r(_mn+aaKy|$5xtMyUQzyS zxA#28)bky4Wn}ccDCeZ^DVnBLzO^pvJ;o!j`QhMQ%O>|)6kM=#8?W!i%yg<%Ep2m1 z#VU)B&N}_+p^twnQ?~3mpTXyR5=Wg)9JOsw#elM_x?8NXZPMhUZnwI|Pl^@als{I! zc2vza*YYJLykB|K*9$+~tj{wj!Di##p(~2kjvM}7oq03Dt}TzYnzQg~shSNZf0(H0 z($Z%5P{W+g{qL;WH|Wy`zbzg#>OiO7VI>pa{=wz$^`4(zJn>tX=%6+}HO5$3?wdUS zEyqIT&ZMlH>0f2CT}!uXjt4j7jdYwb>R{N40^9S%*lgXA@Zmd}6615Xu{~FK$(*Z& zw|Y0T{Jdw3-KD}iPgfoJ$>Q|DZ_kGOT{;ORe7hzoKX#K_Ng%-w82j~%Lw->b$>XMHz3`&!8c2WzE?qi ztZt>~uc~b*IK`=q>qURBK_}0haUb)?)LyauV-x;NI24e6E}-z>guampZO=!#T-<1x zn4{fCA8Dh?d>ytpxO32g$#%;eDh#{xOJM0{_b1P6zU1q(nsRo7e)?jpWsj??Kln2s zz`9`cTF)}6EsNVum^$1!L=!UfRJB!?;$t<7%U)S`ZOWRTTlJm0{953?UeklWw(M1I z>94>3x^2*4O_?j*_vYF=WABjs5iVygR2w(pZhq9QZhh?%{&`BSa{cJNq05GJcKopE z=#EPY*Blf(NT1k$^4xxt=X%5r$`#b^y$2F|SDIL?EVH#nm~YOxx59s|uyvPTeZNE3 zQqsPT9WmZw!PJxG|I+vQLkS6xB1T=;ncK zE!vLHvC(GhS@B^9+sPKgquu6Sv^zRu$f~lP+uvwAWPimYOIG9!%GcQHo9NS#5B7=U zG5gX&BELO-`t-J5rAB@E_T{o=ntxk<_{YIBYFU1tGUv0K?Ux*Cx^Pl^*K(mHa;}f9 zmiXzeRn0U3-aYoVxcAu~x5kG4JflU`B*R7*`#aj`K3#tbu{fnI<@;rH&Yk|B{}?)8 z#-T<|Z@DeXHTzigq@vp_=GuJ_T|V@K5qme5{T~1TLI1wBQCcFhf=&3cGjSutLM|y< zQ`PeP9WeRvKlbC}qobpubiE#%lqP=JYf+@ZvSNz>6&MQ~Y8m&iMv9O3*iXshRRB*v zZIAo=kf!!%7d1j8k;UrZ0Iz0BP7%ov$>X9>1)jQooxQz1Mx)UOa@dzHT>=&gAW4%N zuqnv%z-KT@@9WsA2IdEku!(o{~lo@ZZf-MaOGoQ%tJoCx_B zzX)Icas;%*`&D3FMb|Ep=|+r|Agi3iABiXo1J80N#1W9y=Yl zdKLEfOCPa?Y9EnRY;{Pn!E_94qq+~AsK{fzl2K$CIuU6BA5+!OpnM9OyiK62>z`oF zIBx+r6@3TgHt@EpzKL?+@kk_XC+-T+MHvI`L(ZO&14pWQOGLf_+zNFL)}BmQ4dAC| zP+kCD6yyc0cVPtdLx-eXkbB}}2)u#06d$~l?&p`k98nYv5!n!t7l7v!t|PK8&_pB- z=e{HGDk85Taud~UydP0^S~>^vcojhDQ+Y4E@kUING{pNAL^c(!p?Ve7OW2Aq5!eyc zn+orU>YW6Rx{%%7+QN%4_p!dmL83AW-Dos;?X}mqcI{eEL|($W_*}<){&Mvi16#nW zz_(TPRp2(l5tjZs0;VV;NG3=I6pEIn?}u4Dkc-F!IEkX@0GQ-OjxT&FPPP#4pxjk- zTaclmWlFS-SphYyxyWaLJzyK<=zFRbI~{ur_c;}znIsJM_r=9=B9lo+MK*!w5cxjf zMMR!Kbq#X~BA`IzuG&4EB68qVk5HYI`#lf2fyb)=$^f@ru!cCdE^q^p=Ybd1R?K@4fl%(UCMUZ)=l>|@QR|ptLQg@ zcYu+IGzDo3(nSyvaR_x$SyS|Kuqr<63Lp}suA*I()uJed&betE#TgN~S3gs)cUAR3 zM0OON3bLwb8-qlrAq!%?5|wS>n>4w_@B_Fy-h&9WR*S=}E#fGusmO|`UPJZcz$Xy- zBydBmiAx)ZP?b;t83DT@@~%KLQZ=HNpa5o%`|j|V6~O%dsS0Kmc15Jb!hZ%+06jz! z?0Rh=(#Fi|NTKdk?|IcjFEYis4CnH=(a^$sTd7synP=ecT^J6(E5}`${wuweL`3?S z4|@Z+!lIvE0qg_c0KN?T9&j7TM5G(Ov#z2Y%*ByV1k}c$v+vcx_ro9wLLcQiFi_Q1 zRh_BoOpq}K2|mDrWLvoIxN3Yt#wC+vdw!oIE9-umKjYd|C$M8MEk(a^0V6I*Po(?U% z%XZuzGyQKX`rE+osOq+&H4*uQsy-(o&nUVXF#k!o3=%Q{t9&#RKwVX@V0eI4;8<15 zpxzWX5Ruy=^0tWFRMkCI{e8@Zx{ggOhbhQaETwuGWff(I1)}h~DjF=)U%P%?7z|Q@ zRe@&|KBngCy?`m{D_G#KA?EjMIKPnP(YFwJ4b`_i>}XL81qwY!(2mDz0R#mQuv59h zMD5}YF%6nyH$SG#k%#=T+G8J)4N-kd;2l-HE2;o-m~t#IC2)jt z8(}S&%v)HX^bJK1Lp&sA(KE3ZvIgr|K=`|=0RAt3_RsP^|Kop*%x2C-Q6k&hO%>^j z>eDLn3E&d~F9p}M#A%fLjXoUD3FfN44SW;TSFm93K~WUt`JWqktO}qAlTy9+Dl%5* z_7Hg!=m@N+st8;~btSj~k*d~!zSyyMfprnNg6dNOcNBI7wpHXFmV!FQx!L!~@Ty^} z)CNKwgDwHgRCODRdf!#lJ8>@nUqJah@SGq$7OAe&OF~Hu`eBzinDPyTH5RboavLdn z9c3k4v#*KWw|gkZie{8~b-O}kO~D4{PJLJ9fBwh+7{Fnr%3V}1i|P%`CH*93`d=4V zMWl|!dOT35DYYX^fxQO&KC0gW-c;3lsygBvHhg)k3Sc&yhe14$xmxYmiQL8}d0ik= z)iDMQ?O|>}93B%EWwl+Pr)H8JDBKa%n*whu+|n#N#QXd|`EULk|Nr`He~s+BhrmjJ zUWfyjA8#@Rc7VGmCt^U6j{~2>%*RXEz&<~=;P*yK^Zh{`j(eEpgj#C%5V#8*D0&Qx zf|XVJ)}=+U#&@x6a*`yhbi3^B?ZwWyriiSm$aUa3?0!8To<9|SyT+o9VG2x89U*cT z_!jW{VhZ4<_kPc-4q3pV&y}2uRc7<~=NazrgEhs-M0F}KKy`#_j)whc za0O@w3*Z3Z8wlS-csqRVCxFj~=M6sklmAgfLiJ;5IUNCSglqe85qSzTvuojM4Mk*E zMDB^mO%XX15sr_KzoUa_Hk%wC9f@i}q;=L#cnZd;t2s>m1^srgkyEaG~Z#f@;7 z6%d8Ldq5!;8;A)AJh8@^0$BFqA5ow}A&U{}$7jOVw9 ztsa+oGFJZgBTkU6qE|5`T36Mni0q>5VJm(IyJjgCH|fD2oHi|vW8yewGMNbAM#Et( zj-nRMT?Rge$jiWIv58+s_)Dtxa_p3W-b3IHs;>jDi0U_hH&k_drQ011hQq3*d+y|1 zD1eGsCmHX@h~#QPya6J6i0q)cg_&g6C?n+}H5QYq9un{Z3=tWiIu^?wbOi3H?dJs3 z%6}k}Oc(Kl2y;i8g4ngRsXas4SJgvofE(e*M0%b5ol!t)K_ ztD^df!cDy2Rn_TmG{T>AMx*nl005MFy&#H;B3yJ)9Xc3dNU9?&cl-<@mocBLS06 z;4?%%4m^YE26k^fmeo1NCVZKlDzN+eCh&^DE2zGS>Tcov81U1_RHgcU<|(^0 zEhvhyI=8E;nW&zix{K;Pg?)jS0uR@QEL;MBnyRi~6EPQi9Eiv3mGFS&QEXTbUluZeKx#)~}wq)33Ism2tI=qdg1;^^~fMRWp;_h8Y5)dg^SsD51GnQh_K+17n|G}A|q7Ch>TQqstYY%Hn4o}6iu+^p1$^U>oHkKKQa?}NvrTuI-;VA4ZX9gD~|!Zs{f z09Gz-3Lu?LXT)(l2WCA`(zJR|393C*pF`x+zz?dW`8HTo^Es?~^F+(~1GtANfC*OO zdk2v_PW6t!uBslED?jj|nEd4hJ-<>~0)O}qD_{CjhU$r`&VnB{!UoPZ@B*sONr2ZP z(v(GxJ;3(Vt4IM1LVfm#g+C`OBDD{kab;IqK@vqf2rDYu#Ih45Ja7h#RCTbuw>^6P zx#!)<@yWKT{+<|&`QunH@O@$ZOGQwfSdULRoHh9u$5OH)KY|h{6g>jo1-=HnqN+#J z>2%U)G0rqPnXh&x=SEqFD=%Rerq;5yiPez^nJe01~Q#RL&HY zIXh7aLMnfwM1$p+aybSj&)<07zxn!`eq(cE7aQO)3lp-8Zda`bTcN;-2vHOf$1zC~ z&+8Ae%)%B?6wNp1tiGh|u4CJx*2;;6(4^$4fxtXg{^YWhor@OlX;3zWZJMzd@0N)b zN%88)b5EA#%x2lV8&a^}eK_|qmL(~#)jh0U;%mT7@BQH5;2^6tY9|g?uDGS8R^kd8 zK>Oa^d(G?5T_4RRvxiQ+5^<{FM71V{>3d&pE_Ha3$3gEPa!S9cc>EF1a2e)r$Se^M)RF4ta#<@Gd9aSwM zqBnqkpq1*WO{|(gOQ3~FP84N;*P_T^X-1rRH%kQ&u3JY$I>;RUE5Iz2`5pq((Rf6E zwU1ih@)Z^k;oGYEG;jrB9UIUdNCjCDky;c*G#U+7R=RY%9a^m>s-ENH0p5EWjRvh& zlUA!i94C11DT<=9c|-)~9C0MX5ja=QL$UfRaS*sjRV_T>oWU!ci{>l1sz)^kouWDd zJwkLJo6j}GbrHADEUz&bWO{gb%+AgpgTX+uEJsy6$`o5U-vPD}juqX*5Pfd~-^P@| z+JcIn0q-Me+`LmSvR-wNRHaOuwiI5;?fDDsHR$n#8L zEJ-quC^|xQ7>Y3_VY`myjmyRxWz;?wGwrq!Inwu=mp4MoHnOyWn*uOXabHAcMNtsP zF>rG{b!&?}FPP0{S`>v81%z14v8cR?RsHQE>>%6| zWEdOyQVpE72|)`z^a@%dH&4-*m6Yu9vk>sGF+L-jrv0|8BeeGF6naa3OfuF8UH z%3%SYHBs$gHQHqg>Y*+`#ium|03AV=6o4SR2)n?v^n*t_@=e#AB}H-|NTH$!ioT1v z0h@sp=aQB(HglUUe0IFl~$Ed%H7G2O0 z;#QF83NE_NWZGnR?}SF<57y^UFhDQ1~`3wRwXD%(~#3M;<( zJGQXQgT-l@o@;>5qsoU<0Tcxk1vBG<)BobX`|t99_?Q23kwlS~D4MCo1`bd?zy{C} zw$e5*ZNC*T1!W+x^fAwz{6F#|KLS{uIRv;$MA|Bg3P1=5z_86qg|Sww;nn-@Nq3wj zSw5@HhCcDT2zNsdf$QNIuAt=3S?AzRr_I%;F7dJFu5tD1C7R79vzf=Mki-db5|boJ zhyj?N?!5++9Sr*55C)*Bu`2r9-g7N*EsK3g4a6bXhB(AU_B=os6laI16V$JudAO9a zf{Qkg=nAu}Mz23)GR=AIwKu5MY7!#!WhEyFC&0Fd+)>fP@nk%E`r6ZeJRWm+c*xGq z4vms2YB@c42@9OULIJmVgfZ+F3cv$Xr@ABNk_|D}a2JtRRP|jo|Gun7&G5dQ^h=1I zkHX|HFXV?_{_lVFuYyIYrl=mOt-eRVp1>9&Z>tq_G=SXz*!X?*{%tH~@P4t{HOz|G z44r+-YPFf7dnkK~&dQX(7mtdvHeSH}=lgRrL9AKU~>5yxFX? zy4q!Zy-mO8Xf_>Q4I1yF2oVbaRz)i`>V~j)pn?`hsITFR2*^>HAaVfghocPigAy=H zVu3xRVP?Lxu*6Cow1#RE)eMmq;yQ?Q@IIwhYs>ELJ}WC->h&7V&9j{`)+?}v5Q~Vf z)oOXG-SWfXkfJDX&JjgX)zUIj(QrL0r|GcFOTwdMQc5(Qsz+oZB6pq0?<&0EjlpMt z>RnV1muS4J3&V>q!mF=-z$5#6FCPg7FggkxkisSCeh)KC3lY%=rPZE$?l}Oy^{sD_=Q)i=BURNkLCOGLIS6Mc2Z|nI$^d}0 zNTGncy}PKWYNAD?QPG?>i!9E>Cq6?0UABQ;h@yyky+*Isrq^rHXhziP&XCX22rO!g zD1B8dlNH9VX6C!QV|=lX%5H!|9I%L6mJ>vn$JsOits$;yhb7(ME;Pe$7z4A2DGhNk zA`x*?r`PM!>#fk~blBfNU_2g|^%FIOuA;`Cqll=;crsQI;m#ei=F6H37Bv;8$v(<`K@Cc3(GYMUZ)o}hkVEsRIdHnBGK2i!G%X6|kFHJ~Qm=)gVM9~!Ij!_l7&#|=M z43T2a(E1R@pjTdb1%Sh&Lq7A_&qN1%2W3{Fr1L5zM;T~YY!t@y>-)fke+xi!=Hw%Ia#J zW-}#8B8no%7xQ}lO`z?)FL4n|%6IO@nAyK4$etk8)w!m|z*|zr190`~RRC_^ekfOn zdb?0Qsn?kW>+ylP-;cO_q!hqu`RTv^_w`r*;$JMOnk?79STY~EOUp2jiC(&VDcaiF zYDF&UirK=cm@QsP4vf6_V^u941W~Qm>uhdr^7`wq$0E{DbOl&P=nIlWk)xJ6>S;mZ zPKY=`=mvCMVJo}{sH?aFf}$uH>W}dLZ8U!qikqPCg5CqaZh^0lhb3ni6X*Pr;c1$g!Pzx>3E~#ouRby4Xul->q zj$t(wPD;>0IZG2kR(>Do3~% zcN+10`Z}mha933erf?P+&k~pyfgS)ms2<>TKrEZAv>WtSBAShaIGzXgYA73l*RMkr zLS6m!a9;{vWKP@>$}vK<*+ZEH=mO2Y_BoWpM^*q?mMwNR8o|AL-{DC6@XAM80o*6v zL$hhk8Q%wsOu0fuF0%S@rHPV+JlPxor06!vNtR{PZ^Yrr~66Co0TDE2h!8I8u2D4u}J!6gW(xvGAIFHQ{J8?_Lbp?=FIzQ2PO zcfj8RzXiIDWkQ$mE6gXcl+R7n0J9bm2n|{1pAqc)<}94jjg z8jTupl9UBIW%lHfh;%{}Umg^DIX*rvR=O)gghAj3EG0rin|0{!$zD@MCr;fH=fi)e z@?md0@o17|Fqv34*3syFO@3Hl;{`F2%li?4Stx2bRCJu>S?&w3Wo~|P7#UfX@wfl> z-)1lv2o}s*Lk$irSku`wDQLIGw3>&+@m*Bz0{bWf&>^Zr&;h>KMf11t**DPa%V_pH zX!bjh{~lVr2L86K&T~A=>G@H``&V8Q0Vj4aF8n>urB^>&Q2@*gP3yOaqdnSf&q_C? z-bjfTsl9~2-$J&o)R$FS7H36;AUaaSEe_(=; zCt02h1@PUI1T)!tr)mHSlrhRkM6z0~X2t7)da9JkWWr=J;qU#uzsJGBfq3st0SK#z zq=wx$(|@Hqq}#bi9KC`TzlSfri553e-3dXwDd-+P{~DV8TXgm%$iEE5s{y9Fi|7fW zGo)hj&*$>u$?_(PwO#LmuPY=$I&)&~6ExPD-O52Nz(zm$I0rAtE(Z zjsE+;{ok{;wl?4AY&K&yo1v=w?4SBsRCS`3qd%sM

c%3v0@{AxLde#B?E#Do>UR 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 + + + + + + + + + + + + + + diff --git a/src/interface/desktop/assets/icons/github.svg b/src/interface/desktop/assets/icons/github.svg new file mode 100644 index 00000000..98d74c33 --- /dev/null +++ b/src/interface/desktop/assets/icons/github.svg @@ -0,0 +1 @@ + diff --git a/src/interface/desktop/assets/icons/khoj-logo-sideways-200.png b/src/interface/desktop/assets/icons/khoj-logo-sideways-200.png new file mode 100644 index 0000000000000000000000000000000000000000..52ccdfa4ad8d26dd0a15b092246390000df12c50 GIT binary patch literal 13011 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 A{r~^~ literal 0 HcmV?d00001 diff --git a/src/interface/desktop/assets/icons/khoj-logo-sideways-500.png b/src/interface/desktop/assets/icons/khoj-logo-sideways-500.png new file mode 100644 index 0000000000000000000000000000000000000000..56648932e1322e8f06ace593302da32b2f278aca GIT binary patch literal 36306 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%0Dy0)*K>{#>@8mNG;5v~ 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

9E}bC5#_G-db`4mK7`V<6Ndd-S^Gp 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 literal 0 HcmV?d00001 diff --git a/src/interface/desktop/assets/icons/khoj-logo-sideways.svg b/src/interface/desktop/assets/icons/khoj-logo-sideways.svg new file mode 100644 index 00000000..12e065ba --- /dev/null +++ b/src/interface/desktop/assets/icons/khoj-logo-sideways.svg @@ -0,0 +1,5385 @@ + + + + + + + + + + + Logo 1024px + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Logo 1024px + + + + diff --git a/src/interface/desktop/assets/icons/link.svg b/src/interface/desktop/assets/icons/link.svg new file mode 100644 index 00000000..ef484368 --- /dev/null +++ b/src/interface/desktop/assets/icons/link.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/src/interface/desktop/assets/icons/logotype.svg b/src/interface/desktop/assets/icons/logotype.svg new file mode 100644 index 00000000..397deceb --- /dev/null +++ b/src/interface/desktop/assets/icons/logotype.svg @@ -0,0 +1 @@ +logotype diff --git a/src/interface/desktop/assets/icons/markdown.svg b/src/interface/desktop/assets/icons/markdown.svg new file mode 100644 index 00000000..65b2d2af --- /dev/null +++ b/src/interface/desktop/assets/icons/markdown.svg @@ -0,0 +1 @@ + diff --git a/src/interface/desktop/assets/icons/notion.svg b/src/interface/desktop/assets/icons/notion.svg new file mode 100644 index 00000000..bf6442f7 --- /dev/null +++ b/src/interface/desktop/assets/icons/notion.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/interface/desktop/assets/icons/openai-logomark.svg b/src/interface/desktop/assets/icons/openai-logomark.svg new file mode 100644 index 00000000..c0bcb8bc --- /dev/null +++ b/src/interface/desktop/assets/icons/openai-logomark.svg @@ -0,0 +1 @@ + diff --git a/src/interface/desktop/assets/icons/org.svg b/src/interface/desktop/assets/icons/org.svg new file mode 100644 index 00000000..8696d3d5 --- /dev/null +++ b/src/interface/desktop/assets/icons/org.svg @@ -0,0 +1 @@ + diff --git a/src/interface/desktop/assets/icons/pdf.svg b/src/interface/desktop/assets/icons/pdf.svg new file mode 100644 index 00000000..7c8598b3 --- /dev/null +++ b/src/interface/desktop/assets/icons/pdf.svg @@ -0,0 +1,23 @@ + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/interface/desktop/assets/icons/plaintext.svg b/src/interface/desktop/assets/icons/plaintext.svg new file mode 100644 index 00000000..92233c24 --- /dev/null +++ b/src/interface/desktop/assets/icons/plaintext.svg @@ -0,0 +1 @@ + diff --git a/src/interface/desktop/assets/icons/question-mark-icon.svg b/src/interface/desktop/assets/icons/question-mark-icon.svg new file mode 100644 index 00000000..fde2e972 --- /dev/null +++ b/src/interface/desktop/assets/icons/question-mark-icon.svg @@ -0,0 +1 @@ + diff --git a/src/interface/desktop/assets/khoj.css b/src/interface/desktop/assets/khoj.css new file mode 100644 index 00000000..02f2493f --- /dev/null +++ b/src/interface/desktop/assets/khoj.css @@ -0,0 +1,116 @@ +/* Amber Light scheme (Default) */ +/* Can be forced with data-theme="light" */ +[data-theme="light"], +:root:not([data-theme="dark"]) { + --primary: #ffb300; + --primary-hover: #ffa000; + --primary-focus: rgba(255, 179, 0, 0.125); + --primary-inverse: rgba(0, 0, 0, 0.75); +} + +/* Amber Dark scheme (Auto) */ +/* Automatically enabled if user has Dark mode enabled */ +@media only screen and (prefers-color-scheme: dark) { + :root:not([data-theme]) { + --primary: #ffb300; + --primary-hover: #ffc107; + --primary-focus: rgba(255, 179, 0, 0.25); + --primary-inverse: rgba(0, 0, 0, 0.75); + } +} +/* Amber Dark scheme (Forced) */ +/* Enabled if forced with data-theme="dark" */ +[data-theme="dark"] { + --primary: #ffb300; + --primary-hover: #ffc107; + --primary-focus: rgba(255, 179, 0, 0.25); + --primary-inverse: rgba(0, 0, 0, 0.75); +} +/* Amber (Common styles) */ +:root { + --form-element-active-border-color: var(--primary); + --form-element-focus-color: var(--primary-focus); + --switch-color: var(--primary-inverse); + --switch-checked-background-color: var(--primary); +} + +.khoj-configure { + display: grid; + grid-template-columns: 1fr; + padding: 0 24px; +} +.khoj-header { + display: grid; + grid-auto-flow: column; + gap: 20px; + padding: 16px 0; + margin: 0 0 16px 0; +} + +nav.khoj-nav { + display: grid; + grid-auto-flow: column; + grid-gap: 32px; + justify-self: right; +} + +a.khoj-nav { + display: flex; + align-items: center; +} + +a.khoj-logo { + justify-self: left; +} + +.khoj-nav a { + color: #333; + text-decoration: none; + font-size: small; + font-weight: normal; + padding: 0 4px; + border-radius: 4px; + justify-self: center; + margin: 0; +} +.khoj-nav a:hover { + background-color: var(--primary-hover); +} +.khoj-nav-selected { + background-color: var(--primary); +} +img.khoj-logo { + width: min(60vw, 111px); + max-width: 100%; + justify-self: center; +} + +a.khoj-banner { + color: black; + text-decoration: none; +} + +p.khoj-banner { + font-size: small; + margin: 0; + padding: 10px; +} + +p#khoj-banner { + display: inline; +} + +@media only screen and (max-width: 600px) { + div.khoj-header { + display: grid; + grid-auto-flow: column; + gap: 20px; + padding: 16px 10px; + margin: 0 0 16px 0; + } + + nav.khoj-nav { + grid-gap: 0px; + justify-content: space-between; + } +} diff --git a/src/interface/desktop/assets/markdown-it.min.js b/src/interface/desktop/assets/markdown-it.min.js new file mode 100644 index 00000000..5b37e28e --- /dev/null +++ b/src/interface/desktop/assets/markdown-it.min.js @@ -0,0 +1,8476 @@ +/*! markdown-it 13.0.1 https://github.com/markdown-it/markdown-it @license MIT */ +(function(global, factory) { + typeof exports === "object" && typeof module !== "undefined" ? module.exports = factory() : typeof define === "function" && define.amd ? define(factory) : (global = typeof globalThis !== "undefined" ? globalThis : global || self, + global.markdownit = factory()); +})(this, (function() { + "use strict"; + function createCommonjsModule(fn, basedir, module) { + return module = { + path: basedir, + exports: {}, + require: function(path, base) { + return commonjsRequire(path, base === undefined || base === null ? module.path : base); + } + }, fn(module, module.exports), module.exports; + } + function getAugmentedNamespace(n) { + if (n.__esModule) return n; + var a = Object.defineProperty({}, "__esModule", { + value: true + }); + Object.keys(n).forEach((function(k) { + var d = Object.getOwnPropertyDescriptor(n, k); + Object.defineProperty(a, k, d.get ? d : { + enumerable: true, + get: function() { + return n[k]; + } + }); + })); + return a; + } + function commonjsRequire() { + throw new Error("Dynamic requires are not currently supported by @rollup/plugin-commonjs"); + } + var require$$0 = { + Aacute: "\xc1", + aacute: "\xe1", + Abreve: "\u0102", + abreve: "\u0103", + ac: "\u223e", + acd: "\u223f", + acE: "\u223e\u0333", + Acirc: "\xc2", + acirc: "\xe2", + acute: "\xb4", + Acy: "\u0410", + acy: "\u0430", + AElig: "\xc6", + aelig: "\xe6", + af: "\u2061", + Afr: "\ud835\udd04", + afr: "\ud835\udd1e", + Agrave: "\xc0", + agrave: "\xe0", + alefsym: "\u2135", + aleph: "\u2135", + Alpha: "\u0391", + alpha: "\u03b1", + Amacr: "\u0100", + amacr: "\u0101", + amalg: "\u2a3f", + amp: "&", + AMP: "&", + andand: "\u2a55", + And: "\u2a53", + and: "\u2227", + andd: "\u2a5c", + andslope: "\u2a58", + andv: "\u2a5a", + ang: "\u2220", + ange: "\u29a4", + angle: "\u2220", + angmsdaa: "\u29a8", + angmsdab: "\u29a9", + angmsdac: "\u29aa", + angmsdad: "\u29ab", + angmsdae: "\u29ac", + angmsdaf: "\u29ad", + angmsdag: "\u29ae", + angmsdah: "\u29af", + angmsd: "\u2221", + angrt: "\u221f", + angrtvb: "\u22be", + angrtvbd: "\u299d", + angsph: "\u2222", + angst: "\xc5", + angzarr: "\u237c", + Aogon: "\u0104", + aogon: "\u0105", + Aopf: "\ud835\udd38", + aopf: "\ud835\udd52", + apacir: "\u2a6f", + ap: "\u2248", + apE: "\u2a70", + ape: "\u224a", + apid: "\u224b", + apos: "'", + ApplyFunction: "\u2061", + approx: "\u2248", + approxeq: "\u224a", + Aring: "\xc5", + aring: "\xe5", + Ascr: "\ud835\udc9c", + ascr: "\ud835\udcb6", + Assign: "\u2254", + ast: "*", + asymp: "\u2248", + asympeq: "\u224d", + Atilde: "\xc3", + atilde: "\xe3", + Auml: "\xc4", + auml: "\xe4", + awconint: "\u2233", + awint: "\u2a11", + backcong: "\u224c", + backepsilon: "\u03f6", + backprime: "\u2035", + backsim: "\u223d", + backsimeq: "\u22cd", + Backslash: "\u2216", + Barv: "\u2ae7", + barvee: "\u22bd", + barwed: "\u2305", + Barwed: "\u2306", + barwedge: "\u2305", + bbrk: "\u23b5", + bbrktbrk: "\u23b6", + bcong: "\u224c", + Bcy: "\u0411", + bcy: "\u0431", + bdquo: "\u201e", + becaus: "\u2235", + because: "\u2235", + Because: "\u2235", + bemptyv: "\u29b0", + bepsi: "\u03f6", + bernou: "\u212c", + Bernoullis: "\u212c", + Beta: "\u0392", + beta: "\u03b2", + beth: "\u2136", + between: "\u226c", + Bfr: "\ud835\udd05", + bfr: "\ud835\udd1f", + bigcap: "\u22c2", + bigcirc: "\u25ef", + bigcup: "\u22c3", + bigodot: "\u2a00", + bigoplus: "\u2a01", + bigotimes: "\u2a02", + bigsqcup: "\u2a06", + bigstar: "\u2605", + bigtriangledown: "\u25bd", + bigtriangleup: "\u25b3", + biguplus: "\u2a04", + bigvee: "\u22c1", + bigwedge: "\u22c0", + bkarow: "\u290d", + blacklozenge: "\u29eb", + blacksquare: "\u25aa", + blacktriangle: "\u25b4", + blacktriangledown: "\u25be", + blacktriangleleft: "\u25c2", + blacktriangleright: "\u25b8", + blank: "\u2423", + blk12: "\u2592", + blk14: "\u2591", + blk34: "\u2593", + block: "\u2588", + bne: "=\u20e5", + bnequiv: "\u2261\u20e5", + bNot: "\u2aed", + bnot: "\u2310", + Bopf: "\ud835\udd39", + bopf: "\ud835\udd53", + bot: "\u22a5", + bottom: "\u22a5", + bowtie: "\u22c8", + boxbox: "\u29c9", + boxdl: "\u2510", + boxdL: "\u2555", + boxDl: "\u2556", + boxDL: "\u2557", + boxdr: "\u250c", + boxdR: "\u2552", + boxDr: "\u2553", + boxDR: "\u2554", + boxh: "\u2500", + boxH: "\u2550", + boxhd: "\u252c", + boxHd: "\u2564", + boxhD: "\u2565", + boxHD: "\u2566", + boxhu: "\u2534", + boxHu: "\u2567", + boxhU: "\u2568", + boxHU: "\u2569", + boxminus: "\u229f", + boxplus: "\u229e", + boxtimes: "\u22a0", + boxul: "\u2518", + boxuL: "\u255b", + boxUl: "\u255c", + boxUL: "\u255d", + boxur: "\u2514", + boxuR: "\u2558", + boxUr: "\u2559", + boxUR: "\u255a", + boxv: "\u2502", + boxV: "\u2551", + boxvh: "\u253c", + boxvH: "\u256a", + boxVh: "\u256b", + boxVH: "\u256c", + boxvl: "\u2524", + boxvL: "\u2561", + boxVl: "\u2562", + boxVL: "\u2563", + boxvr: "\u251c", + boxvR: "\u255e", + boxVr: "\u255f", + boxVR: "\u2560", + bprime: "\u2035", + breve: "\u02d8", + Breve: "\u02d8", + brvbar: "\xa6", + bscr: "\ud835\udcb7", + Bscr: "\u212c", + bsemi: "\u204f", + bsim: "\u223d", + bsime: "\u22cd", + bsolb: "\u29c5", + bsol: "\\", + bsolhsub: "\u27c8", + bull: "\u2022", + bullet: "\u2022", + bump: "\u224e", + bumpE: "\u2aae", + bumpe: "\u224f", + Bumpeq: "\u224e", + bumpeq: "\u224f", + Cacute: "\u0106", + cacute: "\u0107", + capand: "\u2a44", + capbrcup: "\u2a49", + capcap: "\u2a4b", + cap: "\u2229", + Cap: "\u22d2", + capcup: "\u2a47", + capdot: "\u2a40", + CapitalDifferentialD: "\u2145", + caps: "\u2229\ufe00", + caret: "\u2041", + caron: "\u02c7", + Cayleys: "\u212d", + ccaps: "\u2a4d", + Ccaron: "\u010c", + ccaron: "\u010d", + Ccedil: "\xc7", + ccedil: "\xe7", + Ccirc: "\u0108", + ccirc: "\u0109", + Cconint: "\u2230", + ccups: "\u2a4c", + ccupssm: "\u2a50", + Cdot: "\u010a", + cdot: "\u010b", + cedil: "\xb8", + Cedilla: "\xb8", + cemptyv: "\u29b2", + cent: "\xa2", + centerdot: "\xb7", + CenterDot: "\xb7", + cfr: "\ud835\udd20", + Cfr: "\u212d", + CHcy: "\u0427", + chcy: "\u0447", + check: "\u2713", + checkmark: "\u2713", + Chi: "\u03a7", + chi: "\u03c7", + circ: "\u02c6", + circeq: "\u2257", + circlearrowleft: "\u21ba", + circlearrowright: "\u21bb", + circledast: "\u229b", + circledcirc: "\u229a", + circleddash: "\u229d", + CircleDot: "\u2299", + circledR: "\xae", + circledS: "\u24c8", + CircleMinus: "\u2296", + CirclePlus: "\u2295", + CircleTimes: "\u2297", + cir: "\u25cb", + cirE: "\u29c3", + cire: "\u2257", + cirfnint: "\u2a10", + cirmid: "\u2aef", + cirscir: "\u29c2", + ClockwiseContourIntegral: "\u2232", + CloseCurlyDoubleQuote: "\u201d", + CloseCurlyQuote: "\u2019", + clubs: "\u2663", + clubsuit: "\u2663", + colon: ":", + Colon: "\u2237", + Colone: "\u2a74", + colone: "\u2254", + coloneq: "\u2254", + comma: ",", + commat: "@", + comp: "\u2201", + compfn: "\u2218", + complement: "\u2201", + complexes: "\u2102", + cong: "\u2245", + congdot: "\u2a6d", + Congruent: "\u2261", + conint: "\u222e", + Conint: "\u222f", + ContourIntegral: "\u222e", + copf: "\ud835\udd54", + Copf: "\u2102", + coprod: "\u2210", + Coproduct: "\u2210", + copy: "\xa9", + COPY: "\xa9", + copysr: "\u2117", + CounterClockwiseContourIntegral: "\u2233", + crarr: "\u21b5", + cross: "\u2717", + Cross: "\u2a2f", + Cscr: "\ud835\udc9e", + cscr: "\ud835\udcb8", + csub: "\u2acf", + csube: "\u2ad1", + csup: "\u2ad0", + csupe: "\u2ad2", + ctdot: "\u22ef", + cudarrl: "\u2938", + cudarrr: "\u2935", + cuepr: "\u22de", + cuesc: "\u22df", + cularr: "\u21b6", + cularrp: "\u293d", + cupbrcap: "\u2a48", + cupcap: "\u2a46", + CupCap: "\u224d", + cup: "\u222a", + Cup: "\u22d3", + cupcup: "\u2a4a", + cupdot: "\u228d", + cupor: "\u2a45", + cups: "\u222a\ufe00", + curarr: "\u21b7", + curarrm: "\u293c", + curlyeqprec: "\u22de", + curlyeqsucc: "\u22df", + curlyvee: "\u22ce", + curlywedge: "\u22cf", + curren: "\xa4", + curvearrowleft: "\u21b6", + curvearrowright: "\u21b7", + cuvee: "\u22ce", + cuwed: "\u22cf", + cwconint: "\u2232", + cwint: "\u2231", + cylcty: "\u232d", + dagger: "\u2020", + Dagger: "\u2021", + daleth: "\u2138", + darr: "\u2193", + Darr: "\u21a1", + dArr: "\u21d3", + dash: "\u2010", + Dashv: "\u2ae4", + dashv: "\u22a3", + dbkarow: "\u290f", + dblac: "\u02dd", + Dcaron: "\u010e", + dcaron: "\u010f", + Dcy: "\u0414", + dcy: "\u0434", + ddagger: "\u2021", + ddarr: "\u21ca", + DD: "\u2145", + dd: "\u2146", + DDotrahd: "\u2911", + ddotseq: "\u2a77", + deg: "\xb0", + Del: "\u2207", + Delta: "\u0394", + delta: "\u03b4", + demptyv: "\u29b1", + dfisht: "\u297f", + Dfr: "\ud835\udd07", + dfr: "\ud835\udd21", + dHar: "\u2965", + dharl: "\u21c3", + dharr: "\u21c2", + DiacriticalAcute: "\xb4", + DiacriticalDot: "\u02d9", + DiacriticalDoubleAcute: "\u02dd", + DiacriticalGrave: "`", + DiacriticalTilde: "\u02dc", + diam: "\u22c4", + diamond: "\u22c4", + Diamond: "\u22c4", + diamondsuit: "\u2666", + diams: "\u2666", + die: "\xa8", + DifferentialD: "\u2146", + digamma: "\u03dd", + disin: "\u22f2", + div: "\xf7", + divide: "\xf7", + divideontimes: "\u22c7", + divonx: "\u22c7", + DJcy: "\u0402", + djcy: "\u0452", + dlcorn: "\u231e", + dlcrop: "\u230d", + dollar: "$", + Dopf: "\ud835\udd3b", + dopf: "\ud835\udd55", + Dot: "\xa8", + dot: "\u02d9", + DotDot: "\u20dc", + doteq: "\u2250", + doteqdot: "\u2251", + DotEqual: "\u2250", + dotminus: "\u2238", + dotplus: "\u2214", + dotsquare: "\u22a1", + doublebarwedge: "\u2306", + DoubleContourIntegral: "\u222f", + DoubleDot: "\xa8", + DoubleDownArrow: "\u21d3", + DoubleLeftArrow: "\u21d0", + DoubleLeftRightArrow: "\u21d4", + DoubleLeftTee: "\u2ae4", + DoubleLongLeftArrow: "\u27f8", + DoubleLongLeftRightArrow: "\u27fa", + DoubleLongRightArrow: "\u27f9", + DoubleRightArrow: "\u21d2", + DoubleRightTee: "\u22a8", + DoubleUpArrow: "\u21d1", + DoubleUpDownArrow: "\u21d5", + DoubleVerticalBar: "\u2225", + DownArrowBar: "\u2913", + downarrow: "\u2193", + DownArrow: "\u2193", + Downarrow: "\u21d3", + DownArrowUpArrow: "\u21f5", + DownBreve: "\u0311", + downdownarrows: "\u21ca", + downharpoonleft: "\u21c3", + downharpoonright: "\u21c2", + DownLeftRightVector: "\u2950", + DownLeftTeeVector: "\u295e", + DownLeftVectorBar: "\u2956", + DownLeftVector: "\u21bd", + DownRightTeeVector: "\u295f", + DownRightVectorBar: "\u2957", + DownRightVector: "\u21c1", + DownTeeArrow: "\u21a7", + DownTee: "\u22a4", + drbkarow: "\u2910", + drcorn: "\u231f", + drcrop: "\u230c", + Dscr: "\ud835\udc9f", + dscr: "\ud835\udcb9", + DScy: "\u0405", + dscy: "\u0455", + dsol: "\u29f6", + Dstrok: "\u0110", + dstrok: "\u0111", + dtdot: "\u22f1", + dtri: "\u25bf", + dtrif: "\u25be", + duarr: "\u21f5", + duhar: "\u296f", + dwangle: "\u29a6", + DZcy: "\u040f", + dzcy: "\u045f", + dzigrarr: "\u27ff", + Eacute: "\xc9", + eacute: "\xe9", + easter: "\u2a6e", + Ecaron: "\u011a", + ecaron: "\u011b", + Ecirc: "\xca", + ecirc: "\xea", + ecir: "\u2256", + ecolon: "\u2255", + Ecy: "\u042d", + ecy: "\u044d", + eDDot: "\u2a77", + Edot: "\u0116", + edot: "\u0117", + eDot: "\u2251", + ee: "\u2147", + efDot: "\u2252", + Efr: "\ud835\udd08", + efr: "\ud835\udd22", + eg: "\u2a9a", + Egrave: "\xc8", + egrave: "\xe8", + egs: "\u2a96", + egsdot: "\u2a98", + el: "\u2a99", + Element: "\u2208", + elinters: "\u23e7", + ell: "\u2113", + els: "\u2a95", + elsdot: "\u2a97", + Emacr: "\u0112", + emacr: "\u0113", + empty: "\u2205", + emptyset: "\u2205", + EmptySmallSquare: "\u25fb", + emptyv: "\u2205", + EmptyVerySmallSquare: "\u25ab", + emsp13: "\u2004", + emsp14: "\u2005", + emsp: "\u2003", + ENG: "\u014a", + eng: "\u014b", + ensp: "\u2002", + Eogon: "\u0118", + eogon: "\u0119", + Eopf: "\ud835\udd3c", + eopf: "\ud835\udd56", + epar: "\u22d5", + eparsl: "\u29e3", + eplus: "\u2a71", + epsi: "\u03b5", + Epsilon: "\u0395", + epsilon: "\u03b5", + epsiv: "\u03f5", + eqcirc: "\u2256", + eqcolon: "\u2255", + eqsim: "\u2242", + eqslantgtr: "\u2a96", + eqslantless: "\u2a95", + Equal: "\u2a75", + equals: "=", + EqualTilde: "\u2242", + equest: "\u225f", + Equilibrium: "\u21cc", + equiv: "\u2261", + equivDD: "\u2a78", + eqvparsl: "\u29e5", + erarr: "\u2971", + erDot: "\u2253", + escr: "\u212f", + Escr: "\u2130", + esdot: "\u2250", + Esim: "\u2a73", + esim: "\u2242", + Eta: "\u0397", + eta: "\u03b7", + ETH: "\xd0", + eth: "\xf0", + Euml: "\xcb", + euml: "\xeb", + euro: "\u20ac", + excl: "!", + exist: "\u2203", + Exists: "\u2203", + expectation: "\u2130", + exponentiale: "\u2147", + ExponentialE: "\u2147", + fallingdotseq: "\u2252", + Fcy: "\u0424", + fcy: "\u0444", + female: "\u2640", + ffilig: "\ufb03", + fflig: "\ufb00", + ffllig: "\ufb04", + Ffr: "\ud835\udd09", + ffr: "\ud835\udd23", + filig: "\ufb01", + FilledSmallSquare: "\u25fc", + FilledVerySmallSquare: "\u25aa", + fjlig: "fj", + flat: "\u266d", + fllig: "\ufb02", + fltns: "\u25b1", + fnof: "\u0192", + Fopf: "\ud835\udd3d", + fopf: "\ud835\udd57", + forall: "\u2200", + ForAll: "\u2200", + fork: "\u22d4", + forkv: "\u2ad9", + Fouriertrf: "\u2131", + fpartint: "\u2a0d", + frac12: "\xbd", + frac13: "\u2153", + frac14: "\xbc", + frac15: "\u2155", + frac16: "\u2159", + frac18: "\u215b", + frac23: "\u2154", + frac25: "\u2156", + frac34: "\xbe", + frac35: "\u2157", + frac38: "\u215c", + frac45: "\u2158", + frac56: "\u215a", + frac58: "\u215d", + frac78: "\u215e", + frasl: "\u2044", + frown: "\u2322", + fscr: "\ud835\udcbb", + Fscr: "\u2131", + gacute: "\u01f5", + Gamma: "\u0393", + gamma: "\u03b3", + Gammad: "\u03dc", + gammad: "\u03dd", + gap: "\u2a86", + Gbreve: "\u011e", + gbreve: "\u011f", + Gcedil: "\u0122", + Gcirc: "\u011c", + gcirc: "\u011d", + Gcy: "\u0413", + gcy: "\u0433", + Gdot: "\u0120", + gdot: "\u0121", + ge: "\u2265", + gE: "\u2267", + gEl: "\u2a8c", + gel: "\u22db", + geq: "\u2265", + geqq: "\u2267", + geqslant: "\u2a7e", + gescc: "\u2aa9", + ges: "\u2a7e", + gesdot: "\u2a80", + gesdoto: "\u2a82", + gesdotol: "\u2a84", + gesl: "\u22db\ufe00", + gesles: "\u2a94", + Gfr: "\ud835\udd0a", + gfr: "\ud835\udd24", + gg: "\u226b", + Gg: "\u22d9", + ggg: "\u22d9", + gimel: "\u2137", + GJcy: "\u0403", + gjcy: "\u0453", + gla: "\u2aa5", + gl: "\u2277", + glE: "\u2a92", + glj: "\u2aa4", + gnap: "\u2a8a", + gnapprox: "\u2a8a", + gne: "\u2a88", + gnE: "\u2269", + gneq: "\u2a88", + gneqq: "\u2269", + gnsim: "\u22e7", + Gopf: "\ud835\udd3e", + gopf: "\ud835\udd58", + grave: "`", + GreaterEqual: "\u2265", + GreaterEqualLess: "\u22db", + GreaterFullEqual: "\u2267", + GreaterGreater: "\u2aa2", + GreaterLess: "\u2277", + GreaterSlantEqual: "\u2a7e", + GreaterTilde: "\u2273", + Gscr: "\ud835\udca2", + gscr: "\u210a", + gsim: "\u2273", + gsime: "\u2a8e", + gsiml: "\u2a90", + gtcc: "\u2aa7", + gtcir: "\u2a7a", + gt: ">", + GT: ">", + Gt: "\u226b", + gtdot: "\u22d7", + gtlPar: "\u2995", + gtquest: "\u2a7c", + gtrapprox: "\u2a86", + gtrarr: "\u2978", + gtrdot: "\u22d7", + gtreqless: "\u22db", + gtreqqless: "\u2a8c", + gtrless: "\u2277", + gtrsim: "\u2273", + gvertneqq: "\u2269\ufe00", + gvnE: "\u2269\ufe00", + Hacek: "\u02c7", + hairsp: "\u200a", + half: "\xbd", + hamilt: "\u210b", + HARDcy: "\u042a", + hardcy: "\u044a", + harrcir: "\u2948", + harr: "\u2194", + hArr: "\u21d4", + harrw: "\u21ad", + Hat: "^", + hbar: "\u210f", + Hcirc: "\u0124", + hcirc: "\u0125", + hearts: "\u2665", + heartsuit: "\u2665", + hellip: "\u2026", + hercon: "\u22b9", + hfr: "\ud835\udd25", + Hfr: "\u210c", + HilbertSpace: "\u210b", + hksearow: "\u2925", + hkswarow: "\u2926", + hoarr: "\u21ff", + homtht: "\u223b", + hookleftarrow: "\u21a9", + hookrightarrow: "\u21aa", + hopf: "\ud835\udd59", + Hopf: "\u210d", + horbar: "\u2015", + HorizontalLine: "\u2500", + hscr: "\ud835\udcbd", + Hscr: "\u210b", + hslash: "\u210f", + Hstrok: "\u0126", + hstrok: "\u0127", + HumpDownHump: "\u224e", + HumpEqual: "\u224f", + hybull: "\u2043", + hyphen: "\u2010", + Iacute: "\xcd", + iacute: "\xed", + ic: "\u2063", + Icirc: "\xce", + icirc: "\xee", + Icy: "\u0418", + icy: "\u0438", + Idot: "\u0130", + IEcy: "\u0415", + iecy: "\u0435", + iexcl: "\xa1", + iff: "\u21d4", + ifr: "\ud835\udd26", + Ifr: "\u2111", + Igrave: "\xcc", + igrave: "\xec", + ii: "\u2148", + iiiint: "\u2a0c", + iiint: "\u222d", + iinfin: "\u29dc", + iiota: "\u2129", + IJlig: "\u0132", + ijlig: "\u0133", + Imacr: "\u012a", + imacr: "\u012b", + image: "\u2111", + ImaginaryI: "\u2148", + imagline: "\u2110", + imagpart: "\u2111", + imath: "\u0131", + Im: "\u2111", + imof: "\u22b7", + imped: "\u01b5", + Implies: "\u21d2", + incare: "\u2105", + in: "\u2208", + infin: "\u221e", + infintie: "\u29dd", + inodot: "\u0131", + intcal: "\u22ba", + int: "\u222b", + Int: "\u222c", + integers: "\u2124", + Integral: "\u222b", + intercal: "\u22ba", + Intersection: "\u22c2", + intlarhk: "\u2a17", + intprod: "\u2a3c", + InvisibleComma: "\u2063", + InvisibleTimes: "\u2062", + IOcy: "\u0401", + iocy: "\u0451", + Iogon: "\u012e", + iogon: "\u012f", + Iopf: "\ud835\udd40", + iopf: "\ud835\udd5a", + Iota: "\u0399", + iota: "\u03b9", + iprod: "\u2a3c", + iquest: "\xbf", + iscr: "\ud835\udcbe", + Iscr: "\u2110", + isin: "\u2208", + isindot: "\u22f5", + isinE: "\u22f9", + isins: "\u22f4", + isinsv: "\u22f3", + isinv: "\u2208", + it: "\u2062", + Itilde: "\u0128", + itilde: "\u0129", + Iukcy: "\u0406", + iukcy: "\u0456", + Iuml: "\xcf", + iuml: "\xef", + Jcirc: "\u0134", + jcirc: "\u0135", + Jcy: "\u0419", + jcy: "\u0439", + Jfr: "\ud835\udd0d", + jfr: "\ud835\udd27", + jmath: "\u0237", + Jopf: "\ud835\udd41", + jopf: "\ud835\udd5b", + Jscr: "\ud835\udca5", + jscr: "\ud835\udcbf", + Jsercy: "\u0408", + jsercy: "\u0458", + Jukcy: "\u0404", + jukcy: "\u0454", + Kappa: "\u039a", + kappa: "\u03ba", + kappav: "\u03f0", + Kcedil: "\u0136", + kcedil: "\u0137", + Kcy: "\u041a", + kcy: "\u043a", + Kfr: "\ud835\udd0e", + kfr: "\ud835\udd28", + kgreen: "\u0138", + KHcy: "\u0425", + khcy: "\u0445", + KJcy: "\u040c", + kjcy: "\u045c", + Kopf: "\ud835\udd42", + kopf: "\ud835\udd5c", + Kscr: "\ud835\udca6", + kscr: "\ud835\udcc0", + lAarr: "\u21da", + Lacute: "\u0139", + lacute: "\u013a", + laemptyv: "\u29b4", + lagran: "\u2112", + Lambda: "\u039b", + lambda: "\u03bb", + lang: "\u27e8", + Lang: "\u27ea", + langd: "\u2991", + langle: "\u27e8", + lap: "\u2a85", + Laplacetrf: "\u2112", + laquo: "\xab", + larrb: "\u21e4", + larrbfs: "\u291f", + larr: "\u2190", + Larr: "\u219e", + lArr: "\u21d0", + larrfs: "\u291d", + larrhk: "\u21a9", + larrlp: "\u21ab", + larrpl: "\u2939", + larrsim: "\u2973", + larrtl: "\u21a2", + latail: "\u2919", + lAtail: "\u291b", + lat: "\u2aab", + late: "\u2aad", + lates: "\u2aad\ufe00", + lbarr: "\u290c", + lBarr: "\u290e", + lbbrk: "\u2772", + lbrace: "{", + lbrack: "[", + lbrke: "\u298b", + lbrksld: "\u298f", + lbrkslu: "\u298d", + Lcaron: "\u013d", + lcaron: "\u013e", + Lcedil: "\u013b", + lcedil: "\u013c", + lceil: "\u2308", + lcub: "{", + Lcy: "\u041b", + lcy: "\u043b", + ldca: "\u2936", + ldquo: "\u201c", + ldquor: "\u201e", + ldrdhar: "\u2967", + ldrushar: "\u294b", + ldsh: "\u21b2", + le: "\u2264", + lE: "\u2266", + LeftAngleBracket: "\u27e8", + LeftArrowBar: "\u21e4", + leftarrow: "\u2190", + LeftArrow: "\u2190", + Leftarrow: "\u21d0", + LeftArrowRightArrow: "\u21c6", + leftarrowtail: "\u21a2", + LeftCeiling: "\u2308", + LeftDoubleBracket: "\u27e6", + LeftDownTeeVector: "\u2961", + LeftDownVectorBar: "\u2959", + LeftDownVector: "\u21c3", + LeftFloor: "\u230a", + leftharpoondown: "\u21bd", + leftharpoonup: "\u21bc", + leftleftarrows: "\u21c7", + leftrightarrow: "\u2194", + LeftRightArrow: "\u2194", + Leftrightarrow: "\u21d4", + leftrightarrows: "\u21c6", + leftrightharpoons: "\u21cb", + leftrightsquigarrow: "\u21ad", + LeftRightVector: "\u294e", + LeftTeeArrow: "\u21a4", + LeftTee: "\u22a3", + LeftTeeVector: "\u295a", + leftthreetimes: "\u22cb", + LeftTriangleBar: "\u29cf", + LeftTriangle: "\u22b2", + LeftTriangleEqual: "\u22b4", + LeftUpDownVector: "\u2951", + LeftUpTeeVector: "\u2960", + LeftUpVectorBar: "\u2958", + LeftUpVector: "\u21bf", + LeftVectorBar: "\u2952", + LeftVector: "\u21bc", + lEg: "\u2a8b", + leg: "\u22da", + leq: "\u2264", + leqq: "\u2266", + leqslant: "\u2a7d", + lescc: "\u2aa8", + les: "\u2a7d", + lesdot: "\u2a7f", + lesdoto: "\u2a81", + lesdotor: "\u2a83", + lesg: "\u22da\ufe00", + lesges: "\u2a93", + lessapprox: "\u2a85", + lessdot: "\u22d6", + lesseqgtr: "\u22da", + lesseqqgtr: "\u2a8b", + LessEqualGreater: "\u22da", + LessFullEqual: "\u2266", + LessGreater: "\u2276", + lessgtr: "\u2276", + LessLess: "\u2aa1", + lesssim: "\u2272", + LessSlantEqual: "\u2a7d", + LessTilde: "\u2272", + lfisht: "\u297c", + lfloor: "\u230a", + Lfr: "\ud835\udd0f", + lfr: "\ud835\udd29", + lg: "\u2276", + lgE: "\u2a91", + lHar: "\u2962", + lhard: "\u21bd", + lharu: "\u21bc", + lharul: "\u296a", + lhblk: "\u2584", + LJcy: "\u0409", + ljcy: "\u0459", + llarr: "\u21c7", + ll: "\u226a", + Ll: "\u22d8", + llcorner: "\u231e", + Lleftarrow: "\u21da", + llhard: "\u296b", + lltri: "\u25fa", + Lmidot: "\u013f", + lmidot: "\u0140", + lmoustache: "\u23b0", + lmoust: "\u23b0", + lnap: "\u2a89", + lnapprox: "\u2a89", + lne: "\u2a87", + lnE: "\u2268", + lneq: "\u2a87", + lneqq: "\u2268", + lnsim: "\u22e6", + loang: "\u27ec", + loarr: "\u21fd", + lobrk: "\u27e6", + longleftarrow: "\u27f5", + LongLeftArrow: "\u27f5", + Longleftarrow: "\u27f8", + longleftrightarrow: "\u27f7", + LongLeftRightArrow: "\u27f7", + Longleftrightarrow: "\u27fa", + longmapsto: "\u27fc", + longrightarrow: "\u27f6", + LongRightArrow: "\u27f6", + Longrightarrow: "\u27f9", + looparrowleft: "\u21ab", + looparrowright: "\u21ac", + lopar: "\u2985", + Lopf: "\ud835\udd43", + lopf: "\ud835\udd5d", + loplus: "\u2a2d", + lotimes: "\u2a34", + lowast: "\u2217", + lowbar: "_", + LowerLeftArrow: "\u2199", + LowerRightArrow: "\u2198", + loz: "\u25ca", + lozenge: "\u25ca", + lozf: "\u29eb", + lpar: "(", + lparlt: "\u2993", + lrarr: "\u21c6", + lrcorner: "\u231f", + lrhar: "\u21cb", + lrhard: "\u296d", + lrm: "\u200e", + lrtri: "\u22bf", + lsaquo: "\u2039", + lscr: "\ud835\udcc1", + Lscr: "\u2112", + lsh: "\u21b0", + Lsh: "\u21b0", + lsim: "\u2272", + lsime: "\u2a8d", + lsimg: "\u2a8f", + lsqb: "[", + lsquo: "\u2018", + lsquor: "\u201a", + Lstrok: "\u0141", + lstrok: "\u0142", + ltcc: "\u2aa6", + ltcir: "\u2a79", + lt: "<", + LT: "<", + Lt: "\u226a", + ltdot: "\u22d6", + lthree: "\u22cb", + ltimes: "\u22c9", + ltlarr: "\u2976", + ltquest: "\u2a7b", + ltri: "\u25c3", + ltrie: "\u22b4", + ltrif: "\u25c2", + ltrPar: "\u2996", + lurdshar: "\u294a", + luruhar: "\u2966", + lvertneqq: "\u2268\ufe00", + lvnE: "\u2268\ufe00", + macr: "\xaf", + male: "\u2642", + malt: "\u2720", + maltese: "\u2720", + Map: "\u2905", + map: "\u21a6", + mapsto: "\u21a6", + mapstodown: "\u21a7", + mapstoleft: "\u21a4", + mapstoup: "\u21a5", + marker: "\u25ae", + mcomma: "\u2a29", + Mcy: "\u041c", + mcy: "\u043c", + mdash: "\u2014", + mDDot: "\u223a", + measuredangle: "\u2221", + MediumSpace: "\u205f", + Mellintrf: "\u2133", + Mfr: "\ud835\udd10", + mfr: "\ud835\udd2a", + mho: "\u2127", + micro: "\xb5", + midast: "*", + midcir: "\u2af0", + mid: "\u2223", + middot: "\xb7", + minusb: "\u229f", + minus: "\u2212", + minusd: "\u2238", + minusdu: "\u2a2a", + MinusPlus: "\u2213", + mlcp: "\u2adb", + mldr: "\u2026", + mnplus: "\u2213", + models: "\u22a7", + Mopf: "\ud835\udd44", + mopf: "\ud835\udd5e", + mp: "\u2213", + mscr: "\ud835\udcc2", + Mscr: "\u2133", + mstpos: "\u223e", + Mu: "\u039c", + mu: "\u03bc", + multimap: "\u22b8", + mumap: "\u22b8", + nabla: "\u2207", + Nacute: "\u0143", + nacute: "\u0144", + nang: "\u2220\u20d2", + nap: "\u2249", + napE: "\u2a70\u0338", + napid: "\u224b\u0338", + napos: "\u0149", + napprox: "\u2249", + natural: "\u266e", + naturals: "\u2115", + natur: "\u266e", + nbsp: "\xa0", + nbump: "\u224e\u0338", + nbumpe: "\u224f\u0338", + ncap: "\u2a43", + Ncaron: "\u0147", + ncaron: "\u0148", + Ncedil: "\u0145", + ncedil: "\u0146", + ncong: "\u2247", + ncongdot: "\u2a6d\u0338", + ncup: "\u2a42", + Ncy: "\u041d", + ncy: "\u043d", + ndash: "\u2013", + nearhk: "\u2924", + nearr: "\u2197", + neArr: "\u21d7", + nearrow: "\u2197", + ne: "\u2260", + nedot: "\u2250\u0338", + NegativeMediumSpace: "\u200b", + NegativeThickSpace: "\u200b", + NegativeThinSpace: "\u200b", + NegativeVeryThinSpace: "\u200b", + nequiv: "\u2262", + nesear: "\u2928", + nesim: "\u2242\u0338", + NestedGreaterGreater: "\u226b", + NestedLessLess: "\u226a", + NewLine: "\n", + nexist: "\u2204", + nexists: "\u2204", + Nfr: "\ud835\udd11", + nfr: "\ud835\udd2b", + ngE: "\u2267\u0338", + nge: "\u2271", + ngeq: "\u2271", + ngeqq: "\u2267\u0338", + ngeqslant: "\u2a7e\u0338", + nges: "\u2a7e\u0338", + nGg: "\u22d9\u0338", + ngsim: "\u2275", + nGt: "\u226b\u20d2", + ngt: "\u226f", + ngtr: "\u226f", + nGtv: "\u226b\u0338", + nharr: "\u21ae", + nhArr: "\u21ce", + nhpar: "\u2af2", + ni: "\u220b", + nis: "\u22fc", + nisd: "\u22fa", + niv: "\u220b", + NJcy: "\u040a", + njcy: "\u045a", + nlarr: "\u219a", + nlArr: "\u21cd", + nldr: "\u2025", + nlE: "\u2266\u0338", + nle: "\u2270", + nleftarrow: "\u219a", + nLeftarrow: "\u21cd", + nleftrightarrow: "\u21ae", + nLeftrightarrow: "\u21ce", + nleq: "\u2270", + nleqq: "\u2266\u0338", + nleqslant: "\u2a7d\u0338", + nles: "\u2a7d\u0338", + nless: "\u226e", + nLl: "\u22d8\u0338", + nlsim: "\u2274", + nLt: "\u226a\u20d2", + nlt: "\u226e", + nltri: "\u22ea", + nltrie: "\u22ec", + nLtv: "\u226a\u0338", + nmid: "\u2224", + NoBreak: "\u2060", + NonBreakingSpace: "\xa0", + nopf: "\ud835\udd5f", + Nopf: "\u2115", + Not: "\u2aec", + not: "\xac", + NotCongruent: "\u2262", + NotCupCap: "\u226d", + NotDoubleVerticalBar: "\u2226", + NotElement: "\u2209", + NotEqual: "\u2260", + NotEqualTilde: "\u2242\u0338", + NotExists: "\u2204", + NotGreater: "\u226f", + NotGreaterEqual: "\u2271", + NotGreaterFullEqual: "\u2267\u0338", + NotGreaterGreater: "\u226b\u0338", + NotGreaterLess: "\u2279", + NotGreaterSlantEqual: "\u2a7e\u0338", + NotGreaterTilde: "\u2275", + NotHumpDownHump: "\u224e\u0338", + NotHumpEqual: "\u224f\u0338", + notin: "\u2209", + notindot: "\u22f5\u0338", + notinE: "\u22f9\u0338", + notinva: "\u2209", + notinvb: "\u22f7", + notinvc: "\u22f6", + NotLeftTriangleBar: "\u29cf\u0338", + NotLeftTriangle: "\u22ea", + NotLeftTriangleEqual: "\u22ec", + NotLess: "\u226e", + NotLessEqual: "\u2270", + NotLessGreater: "\u2278", + NotLessLess: "\u226a\u0338", + NotLessSlantEqual: "\u2a7d\u0338", + NotLessTilde: "\u2274", + NotNestedGreaterGreater: "\u2aa2\u0338", + NotNestedLessLess: "\u2aa1\u0338", + notni: "\u220c", + notniva: "\u220c", + notnivb: "\u22fe", + notnivc: "\u22fd", + NotPrecedes: "\u2280", + NotPrecedesEqual: "\u2aaf\u0338", + NotPrecedesSlantEqual: "\u22e0", + NotReverseElement: "\u220c", + NotRightTriangleBar: "\u29d0\u0338", + NotRightTriangle: "\u22eb", + NotRightTriangleEqual: "\u22ed", + NotSquareSubset: "\u228f\u0338", + NotSquareSubsetEqual: "\u22e2", + NotSquareSuperset: "\u2290\u0338", + NotSquareSupersetEqual: "\u22e3", + NotSubset: "\u2282\u20d2", + NotSubsetEqual: "\u2288", + NotSucceeds: "\u2281", + NotSucceedsEqual: "\u2ab0\u0338", + NotSucceedsSlantEqual: "\u22e1", + NotSucceedsTilde: "\u227f\u0338", + NotSuperset: "\u2283\u20d2", + NotSupersetEqual: "\u2289", + NotTilde: "\u2241", + NotTildeEqual: "\u2244", + NotTildeFullEqual: "\u2247", + NotTildeTilde: "\u2249", + NotVerticalBar: "\u2224", + nparallel: "\u2226", + npar: "\u2226", + nparsl: "\u2afd\u20e5", + npart: "\u2202\u0338", + npolint: "\u2a14", + npr: "\u2280", + nprcue: "\u22e0", + nprec: "\u2280", + npreceq: "\u2aaf\u0338", + npre: "\u2aaf\u0338", + nrarrc: "\u2933\u0338", + nrarr: "\u219b", + nrArr: "\u21cf", + nrarrw: "\u219d\u0338", + nrightarrow: "\u219b", + nRightarrow: "\u21cf", + nrtri: "\u22eb", + nrtrie: "\u22ed", + nsc: "\u2281", + nsccue: "\u22e1", + nsce: "\u2ab0\u0338", + Nscr: "\ud835\udca9", + nscr: "\ud835\udcc3", + nshortmid: "\u2224", + nshortparallel: "\u2226", + nsim: "\u2241", + nsime: "\u2244", + nsimeq: "\u2244", + nsmid: "\u2224", + nspar: "\u2226", + nsqsube: "\u22e2", + nsqsupe: "\u22e3", + nsub: "\u2284", + nsubE: "\u2ac5\u0338", + nsube: "\u2288", + nsubset: "\u2282\u20d2", + nsubseteq: "\u2288", + nsubseteqq: "\u2ac5\u0338", + nsucc: "\u2281", + nsucceq: "\u2ab0\u0338", + nsup: "\u2285", + nsupE: "\u2ac6\u0338", + nsupe: "\u2289", + nsupset: "\u2283\u20d2", + nsupseteq: "\u2289", + nsupseteqq: "\u2ac6\u0338", + ntgl: "\u2279", + Ntilde: "\xd1", + ntilde: "\xf1", + ntlg: "\u2278", + ntriangleleft: "\u22ea", + ntrianglelefteq: "\u22ec", + ntriangleright: "\u22eb", + ntrianglerighteq: "\u22ed", + Nu: "\u039d", + nu: "\u03bd", + num: "#", + numero: "\u2116", + numsp: "\u2007", + nvap: "\u224d\u20d2", + nvdash: "\u22ac", + nvDash: "\u22ad", + nVdash: "\u22ae", + nVDash: "\u22af", + nvge: "\u2265\u20d2", + nvgt: ">\u20d2", + nvHarr: "\u2904", + nvinfin: "\u29de", + nvlArr: "\u2902", + nvle: "\u2264\u20d2", + nvlt: "<\u20d2", + nvltrie: "\u22b4\u20d2", + nvrArr: "\u2903", + nvrtrie: "\u22b5\u20d2", + nvsim: "\u223c\u20d2", + nwarhk: "\u2923", + nwarr: "\u2196", + nwArr: "\u21d6", + nwarrow: "\u2196", + nwnear: "\u2927", + Oacute: "\xd3", + oacute: "\xf3", + oast: "\u229b", + Ocirc: "\xd4", + ocirc: "\xf4", + ocir: "\u229a", + Ocy: "\u041e", + ocy: "\u043e", + odash: "\u229d", + Odblac: "\u0150", + odblac: "\u0151", + odiv: "\u2a38", + odot: "\u2299", + odsold: "\u29bc", + OElig: "\u0152", + oelig: "\u0153", + ofcir: "\u29bf", + Ofr: "\ud835\udd12", + ofr: "\ud835\udd2c", + ogon: "\u02db", + Ograve: "\xd2", + ograve: "\xf2", + ogt: "\u29c1", + ohbar: "\u29b5", + ohm: "\u03a9", + oint: "\u222e", + olarr: "\u21ba", + olcir: "\u29be", + olcross: "\u29bb", + oline: "\u203e", + olt: "\u29c0", + Omacr: "\u014c", + omacr: "\u014d", + Omega: "\u03a9", + omega: "\u03c9", + Omicron: "\u039f", + omicron: "\u03bf", + omid: "\u29b6", + ominus: "\u2296", + Oopf: "\ud835\udd46", + oopf: "\ud835\udd60", + opar: "\u29b7", + OpenCurlyDoubleQuote: "\u201c", + OpenCurlyQuote: "\u2018", + operp: "\u29b9", + oplus: "\u2295", + orarr: "\u21bb", + Or: "\u2a54", + or: "\u2228", + ord: "\u2a5d", + order: "\u2134", + orderof: "\u2134", + ordf: "\xaa", + ordm: "\xba", + origof: "\u22b6", + oror: "\u2a56", + orslope: "\u2a57", + orv: "\u2a5b", + oS: "\u24c8", + Oscr: "\ud835\udcaa", + oscr: "\u2134", + Oslash: "\xd8", + oslash: "\xf8", + osol: "\u2298", + Otilde: "\xd5", + otilde: "\xf5", + otimesas: "\u2a36", + Otimes: "\u2a37", + otimes: "\u2297", + Ouml: "\xd6", + ouml: "\xf6", + ovbar: "\u233d", + OverBar: "\u203e", + OverBrace: "\u23de", + OverBracket: "\u23b4", + OverParenthesis: "\u23dc", + para: "\xb6", + parallel: "\u2225", + par: "\u2225", + parsim: "\u2af3", + parsl: "\u2afd", + part: "\u2202", + PartialD: "\u2202", + Pcy: "\u041f", + pcy: "\u043f", + percnt: "%", + period: ".", + permil: "\u2030", + perp: "\u22a5", + pertenk: "\u2031", + Pfr: "\ud835\udd13", + pfr: "\ud835\udd2d", + Phi: "\u03a6", + phi: "\u03c6", + phiv: "\u03d5", + phmmat: "\u2133", + phone: "\u260e", + Pi: "\u03a0", + pi: "\u03c0", + pitchfork: "\u22d4", + piv: "\u03d6", + planck: "\u210f", + planckh: "\u210e", + plankv: "\u210f", + plusacir: "\u2a23", + plusb: "\u229e", + pluscir: "\u2a22", + plus: "+", + plusdo: "\u2214", + plusdu: "\u2a25", + pluse: "\u2a72", + PlusMinus: "\xb1", + plusmn: "\xb1", + plussim: "\u2a26", + plustwo: "\u2a27", + pm: "\xb1", + Poincareplane: "\u210c", + pointint: "\u2a15", + popf: "\ud835\udd61", + Popf: "\u2119", + pound: "\xa3", + prap: "\u2ab7", + Pr: "\u2abb", + pr: "\u227a", + prcue: "\u227c", + precapprox: "\u2ab7", + prec: "\u227a", + preccurlyeq: "\u227c", + Precedes: "\u227a", + PrecedesEqual: "\u2aaf", + PrecedesSlantEqual: "\u227c", + PrecedesTilde: "\u227e", + preceq: "\u2aaf", + precnapprox: "\u2ab9", + precneqq: "\u2ab5", + precnsim: "\u22e8", + pre: "\u2aaf", + prE: "\u2ab3", + precsim: "\u227e", + prime: "\u2032", + Prime: "\u2033", + primes: "\u2119", + prnap: "\u2ab9", + prnE: "\u2ab5", + prnsim: "\u22e8", + prod: "\u220f", + Product: "\u220f", + profalar: "\u232e", + profline: "\u2312", + profsurf: "\u2313", + prop: "\u221d", + Proportional: "\u221d", + Proportion: "\u2237", + propto: "\u221d", + prsim: "\u227e", + prurel: "\u22b0", + Pscr: "\ud835\udcab", + pscr: "\ud835\udcc5", + Psi: "\u03a8", + psi: "\u03c8", + puncsp: "\u2008", + Qfr: "\ud835\udd14", + qfr: "\ud835\udd2e", + qint: "\u2a0c", + qopf: "\ud835\udd62", + Qopf: "\u211a", + qprime: "\u2057", + Qscr: "\ud835\udcac", + qscr: "\ud835\udcc6", + quaternions: "\u210d", + quatint: "\u2a16", + quest: "?", + questeq: "\u225f", + quot: '"', + QUOT: '"', + rAarr: "\u21db", + race: "\u223d\u0331", + Racute: "\u0154", + racute: "\u0155", + radic: "\u221a", + raemptyv: "\u29b3", + rang: "\u27e9", + Rang: "\u27eb", + rangd: "\u2992", + range: "\u29a5", + rangle: "\u27e9", + raquo: "\xbb", + rarrap: "\u2975", + rarrb: "\u21e5", + rarrbfs: "\u2920", + rarrc: "\u2933", + rarr: "\u2192", + Rarr: "\u21a0", + rArr: "\u21d2", + rarrfs: "\u291e", + rarrhk: "\u21aa", + rarrlp: "\u21ac", + rarrpl: "\u2945", + rarrsim: "\u2974", + Rarrtl: "\u2916", + rarrtl: "\u21a3", + rarrw: "\u219d", + ratail: "\u291a", + rAtail: "\u291c", + ratio: "\u2236", + rationals: "\u211a", + rbarr: "\u290d", + rBarr: "\u290f", + RBarr: "\u2910", + rbbrk: "\u2773", + rbrace: "}", + rbrack: "]", + rbrke: "\u298c", + rbrksld: "\u298e", + rbrkslu: "\u2990", + Rcaron: "\u0158", + rcaron: "\u0159", + Rcedil: "\u0156", + rcedil: "\u0157", + rceil: "\u2309", + rcub: "}", + Rcy: "\u0420", + rcy: "\u0440", + rdca: "\u2937", + rdldhar: "\u2969", + rdquo: "\u201d", + rdquor: "\u201d", + rdsh: "\u21b3", + real: "\u211c", + realine: "\u211b", + realpart: "\u211c", + reals: "\u211d", + Re: "\u211c", + rect: "\u25ad", + reg: "\xae", + REG: "\xae", + ReverseElement: "\u220b", + ReverseEquilibrium: "\u21cb", + ReverseUpEquilibrium: "\u296f", + rfisht: "\u297d", + rfloor: "\u230b", + rfr: "\ud835\udd2f", + Rfr: "\u211c", + rHar: "\u2964", + rhard: "\u21c1", + rharu: "\u21c0", + rharul: "\u296c", + Rho: "\u03a1", + rho: "\u03c1", + rhov: "\u03f1", + RightAngleBracket: "\u27e9", + RightArrowBar: "\u21e5", + rightarrow: "\u2192", + RightArrow: "\u2192", + Rightarrow: "\u21d2", + RightArrowLeftArrow: "\u21c4", + rightarrowtail: "\u21a3", + RightCeiling: "\u2309", + RightDoubleBracket: "\u27e7", + RightDownTeeVector: "\u295d", + RightDownVectorBar: "\u2955", + RightDownVector: "\u21c2", + RightFloor: "\u230b", + rightharpoondown: "\u21c1", + rightharpoonup: "\u21c0", + rightleftarrows: "\u21c4", + rightleftharpoons: "\u21cc", + rightrightarrows: "\u21c9", + rightsquigarrow: "\u219d", + RightTeeArrow: "\u21a6", + RightTee: "\u22a2", + RightTeeVector: "\u295b", + rightthreetimes: "\u22cc", + RightTriangleBar: "\u29d0", + RightTriangle: "\u22b3", + RightTriangleEqual: "\u22b5", + RightUpDownVector: "\u294f", + RightUpTeeVector: "\u295c", + RightUpVectorBar: "\u2954", + RightUpVector: "\u21be", + RightVectorBar: "\u2953", + RightVector: "\u21c0", + ring: "\u02da", + risingdotseq: "\u2253", + rlarr: "\u21c4", + rlhar: "\u21cc", + rlm: "\u200f", + rmoustache: "\u23b1", + rmoust: "\u23b1", + rnmid: "\u2aee", + roang: "\u27ed", + roarr: "\u21fe", + robrk: "\u27e7", + ropar: "\u2986", + ropf: "\ud835\udd63", + Ropf: "\u211d", + roplus: "\u2a2e", + rotimes: "\u2a35", + RoundImplies: "\u2970", + rpar: ")", + rpargt: "\u2994", + rppolint: "\u2a12", + rrarr: "\u21c9", + Rrightarrow: "\u21db", + rsaquo: "\u203a", + rscr: "\ud835\udcc7", + Rscr: "\u211b", + rsh: "\u21b1", + Rsh: "\u21b1", + rsqb: "]", + rsquo: "\u2019", + rsquor: "\u2019", + rthree: "\u22cc", + rtimes: "\u22ca", + rtri: "\u25b9", + rtrie: "\u22b5", + rtrif: "\u25b8", + rtriltri: "\u29ce", + RuleDelayed: "\u29f4", + ruluhar: "\u2968", + rx: "\u211e", + Sacute: "\u015a", + sacute: "\u015b", + sbquo: "\u201a", + scap: "\u2ab8", + Scaron: "\u0160", + scaron: "\u0161", + Sc: "\u2abc", + sc: "\u227b", + sccue: "\u227d", + sce: "\u2ab0", + scE: "\u2ab4", + Scedil: "\u015e", + scedil: "\u015f", + Scirc: "\u015c", + scirc: "\u015d", + scnap: "\u2aba", + scnE: "\u2ab6", + scnsim: "\u22e9", + scpolint: "\u2a13", + scsim: "\u227f", + Scy: "\u0421", + scy: "\u0441", + sdotb: "\u22a1", + sdot: "\u22c5", + sdote: "\u2a66", + searhk: "\u2925", + searr: "\u2198", + seArr: "\u21d8", + searrow: "\u2198", + sect: "\xa7", + semi: ";", + seswar: "\u2929", + setminus: "\u2216", + setmn: "\u2216", + sext: "\u2736", + Sfr: "\ud835\udd16", + sfr: "\ud835\udd30", + sfrown: "\u2322", + sharp: "\u266f", + SHCHcy: "\u0429", + shchcy: "\u0449", + SHcy: "\u0428", + shcy: "\u0448", + ShortDownArrow: "\u2193", + ShortLeftArrow: "\u2190", + shortmid: "\u2223", + shortparallel: "\u2225", + ShortRightArrow: "\u2192", + ShortUpArrow: "\u2191", + shy: "\xad", + Sigma: "\u03a3", + sigma: "\u03c3", + sigmaf: "\u03c2", + sigmav: "\u03c2", + sim: "\u223c", + simdot: "\u2a6a", + sime: "\u2243", + simeq: "\u2243", + simg: "\u2a9e", + simgE: "\u2aa0", + siml: "\u2a9d", + simlE: "\u2a9f", + simne: "\u2246", + simplus: "\u2a24", + simrarr: "\u2972", + slarr: "\u2190", + SmallCircle: "\u2218", + smallsetminus: "\u2216", + smashp: "\u2a33", + smeparsl: "\u29e4", + smid: "\u2223", + smile: "\u2323", + smt: "\u2aaa", + smte: "\u2aac", + smtes: "\u2aac\ufe00", + SOFTcy: "\u042c", + softcy: "\u044c", + solbar: "\u233f", + solb: "\u29c4", + sol: "/", + Sopf: "\ud835\udd4a", + sopf: "\ud835\udd64", + spades: "\u2660", + spadesuit: "\u2660", + spar: "\u2225", + sqcap: "\u2293", + sqcaps: "\u2293\ufe00", + sqcup: "\u2294", + sqcups: "\u2294\ufe00", + Sqrt: "\u221a", + sqsub: "\u228f", + sqsube: "\u2291", + sqsubset: "\u228f", + sqsubseteq: "\u2291", + sqsup: "\u2290", + sqsupe: "\u2292", + sqsupset: "\u2290", + sqsupseteq: "\u2292", + square: "\u25a1", + Square: "\u25a1", + SquareIntersection: "\u2293", + SquareSubset: "\u228f", + SquareSubsetEqual: "\u2291", + SquareSuperset: "\u2290", + SquareSupersetEqual: "\u2292", + SquareUnion: "\u2294", + squarf: "\u25aa", + squ: "\u25a1", + squf: "\u25aa", + srarr: "\u2192", + Sscr: "\ud835\udcae", + sscr: "\ud835\udcc8", + ssetmn: "\u2216", + ssmile: "\u2323", + sstarf: "\u22c6", + Star: "\u22c6", + star: "\u2606", + starf: "\u2605", + straightepsilon: "\u03f5", + straightphi: "\u03d5", + strns: "\xaf", + sub: "\u2282", + Sub: "\u22d0", + subdot: "\u2abd", + subE: "\u2ac5", + sube: "\u2286", + subedot: "\u2ac3", + submult: "\u2ac1", + subnE: "\u2acb", + subne: "\u228a", + subplus: "\u2abf", + subrarr: "\u2979", + subset: "\u2282", + Subset: "\u22d0", + subseteq: "\u2286", + subseteqq: "\u2ac5", + SubsetEqual: "\u2286", + subsetneq: "\u228a", + subsetneqq: "\u2acb", + subsim: "\u2ac7", + subsub: "\u2ad5", + subsup: "\u2ad3", + succapprox: "\u2ab8", + succ: "\u227b", + succcurlyeq: "\u227d", + Succeeds: "\u227b", + SucceedsEqual: "\u2ab0", + SucceedsSlantEqual: "\u227d", + SucceedsTilde: "\u227f", + succeq: "\u2ab0", + succnapprox: "\u2aba", + succneqq: "\u2ab6", + succnsim: "\u22e9", + succsim: "\u227f", + SuchThat: "\u220b", + sum: "\u2211", + Sum: "\u2211", + sung: "\u266a", + sup1: "\xb9", + sup2: "\xb2", + sup3: "\xb3", + sup: "\u2283", + Sup: "\u22d1", + supdot: "\u2abe", + supdsub: "\u2ad8", + supE: "\u2ac6", + supe: "\u2287", + supedot: "\u2ac4", + Superset: "\u2283", + SupersetEqual: "\u2287", + suphsol: "\u27c9", + suphsub: "\u2ad7", + suplarr: "\u297b", + supmult: "\u2ac2", + supnE: "\u2acc", + supne: "\u228b", + supplus: "\u2ac0", + supset: "\u2283", + Supset: "\u22d1", + supseteq: "\u2287", + supseteqq: "\u2ac6", + supsetneq: "\u228b", + supsetneqq: "\u2acc", + supsim: "\u2ac8", + supsub: "\u2ad4", + supsup: "\u2ad6", + swarhk: "\u2926", + swarr: "\u2199", + swArr: "\u21d9", + swarrow: "\u2199", + swnwar: "\u292a", + szlig: "\xdf", + Tab: "\t", + target: "\u2316", + Tau: "\u03a4", + tau: "\u03c4", + tbrk: "\u23b4", + Tcaron: "\u0164", + tcaron: "\u0165", + Tcedil: "\u0162", + tcedil: "\u0163", + Tcy: "\u0422", + tcy: "\u0442", + tdot: "\u20db", + telrec: "\u2315", + Tfr: "\ud835\udd17", + tfr: "\ud835\udd31", + there4: "\u2234", + therefore: "\u2234", + Therefore: "\u2234", + Theta: "\u0398", + theta: "\u03b8", + thetasym: "\u03d1", + thetav: "\u03d1", + thickapprox: "\u2248", + thicksim: "\u223c", + ThickSpace: "\u205f\u200a", + ThinSpace: "\u2009", + thinsp: "\u2009", + thkap: "\u2248", + thksim: "\u223c", + THORN: "\xde", + thorn: "\xfe", + tilde: "\u02dc", + Tilde: "\u223c", + TildeEqual: "\u2243", + TildeFullEqual: "\u2245", + TildeTilde: "\u2248", + timesbar: "\u2a31", + timesb: "\u22a0", + times: "\xd7", + timesd: "\u2a30", + tint: "\u222d", + toea: "\u2928", + topbot: "\u2336", + topcir: "\u2af1", + top: "\u22a4", + Topf: "\ud835\udd4b", + topf: "\ud835\udd65", + topfork: "\u2ada", + tosa: "\u2929", + tprime: "\u2034", + trade: "\u2122", + TRADE: "\u2122", + triangle: "\u25b5", + triangledown: "\u25bf", + triangleleft: "\u25c3", + trianglelefteq: "\u22b4", + triangleq: "\u225c", + triangleright: "\u25b9", + trianglerighteq: "\u22b5", + tridot: "\u25ec", + trie: "\u225c", + triminus: "\u2a3a", + TripleDot: "\u20db", + triplus: "\u2a39", + trisb: "\u29cd", + tritime: "\u2a3b", + trpezium: "\u23e2", + Tscr: "\ud835\udcaf", + tscr: "\ud835\udcc9", + TScy: "\u0426", + tscy: "\u0446", + TSHcy: "\u040b", + tshcy: "\u045b", + Tstrok: "\u0166", + tstrok: "\u0167", + twixt: "\u226c", + twoheadleftarrow: "\u219e", + twoheadrightarrow: "\u21a0", + Uacute: "\xda", + uacute: "\xfa", + uarr: "\u2191", + Uarr: "\u219f", + uArr: "\u21d1", + Uarrocir: "\u2949", + Ubrcy: "\u040e", + ubrcy: "\u045e", + Ubreve: "\u016c", + ubreve: "\u016d", + Ucirc: "\xdb", + ucirc: "\xfb", + Ucy: "\u0423", + ucy: "\u0443", + udarr: "\u21c5", + Udblac: "\u0170", + udblac: "\u0171", + udhar: "\u296e", + ufisht: "\u297e", + Ufr: "\ud835\udd18", + ufr: "\ud835\udd32", + Ugrave: "\xd9", + ugrave: "\xf9", + uHar: "\u2963", + uharl: "\u21bf", + uharr: "\u21be", + uhblk: "\u2580", + ulcorn: "\u231c", + ulcorner: "\u231c", + ulcrop: "\u230f", + ultri: "\u25f8", + Umacr: "\u016a", + umacr: "\u016b", + uml: "\xa8", + UnderBar: "_", + UnderBrace: "\u23df", + UnderBracket: "\u23b5", + UnderParenthesis: "\u23dd", + Union: "\u22c3", + UnionPlus: "\u228e", + Uogon: "\u0172", + uogon: "\u0173", + Uopf: "\ud835\udd4c", + uopf: "\ud835\udd66", + UpArrowBar: "\u2912", + uparrow: "\u2191", + UpArrow: "\u2191", + Uparrow: "\u21d1", + UpArrowDownArrow: "\u21c5", + updownarrow: "\u2195", + UpDownArrow: "\u2195", + Updownarrow: "\u21d5", + UpEquilibrium: "\u296e", + upharpoonleft: "\u21bf", + upharpoonright: "\u21be", + uplus: "\u228e", + UpperLeftArrow: "\u2196", + UpperRightArrow: "\u2197", + upsi: "\u03c5", + Upsi: "\u03d2", + upsih: "\u03d2", + Upsilon: "\u03a5", + upsilon: "\u03c5", + UpTeeArrow: "\u21a5", + UpTee: "\u22a5", + upuparrows: "\u21c8", + urcorn: "\u231d", + urcorner: "\u231d", + urcrop: "\u230e", + Uring: "\u016e", + uring: "\u016f", + urtri: "\u25f9", + Uscr: "\ud835\udcb0", + uscr: "\ud835\udcca", + utdot: "\u22f0", + Utilde: "\u0168", + utilde: "\u0169", + utri: "\u25b5", + utrif: "\u25b4", + uuarr: "\u21c8", + Uuml: "\xdc", + uuml: "\xfc", + uwangle: "\u29a7", + vangrt: "\u299c", + varepsilon: "\u03f5", + varkappa: "\u03f0", + varnothing: "\u2205", + varphi: "\u03d5", + varpi: "\u03d6", + varpropto: "\u221d", + varr: "\u2195", + vArr: "\u21d5", + varrho: "\u03f1", + varsigma: "\u03c2", + varsubsetneq: "\u228a\ufe00", + varsubsetneqq: "\u2acb\ufe00", + varsupsetneq: "\u228b\ufe00", + varsupsetneqq: "\u2acc\ufe00", + vartheta: "\u03d1", + vartriangleleft: "\u22b2", + vartriangleright: "\u22b3", + vBar: "\u2ae8", + Vbar: "\u2aeb", + vBarv: "\u2ae9", + Vcy: "\u0412", + vcy: "\u0432", + vdash: "\u22a2", + vDash: "\u22a8", + Vdash: "\u22a9", + VDash: "\u22ab", + Vdashl: "\u2ae6", + veebar: "\u22bb", + vee: "\u2228", + Vee: "\u22c1", + veeeq: "\u225a", + vellip: "\u22ee", + verbar: "|", + Verbar: "\u2016", + vert: "|", + Vert: "\u2016", + VerticalBar: "\u2223", + VerticalLine: "|", + VerticalSeparator: "\u2758", + VerticalTilde: "\u2240", + VeryThinSpace: "\u200a", + Vfr: "\ud835\udd19", + vfr: "\ud835\udd33", + vltri: "\u22b2", + vnsub: "\u2282\u20d2", + vnsup: "\u2283\u20d2", + Vopf: "\ud835\udd4d", + vopf: "\ud835\udd67", + vprop: "\u221d", + vrtri: "\u22b3", + Vscr: "\ud835\udcb1", + vscr: "\ud835\udccb", + vsubnE: "\u2acb\ufe00", + vsubne: "\u228a\ufe00", + vsupnE: "\u2acc\ufe00", + vsupne: "\u228b\ufe00", + Vvdash: "\u22aa", + vzigzag: "\u299a", + Wcirc: "\u0174", + wcirc: "\u0175", + wedbar: "\u2a5f", + wedge: "\u2227", + Wedge: "\u22c0", + wedgeq: "\u2259", + weierp: "\u2118", + Wfr: "\ud835\udd1a", + wfr: "\ud835\udd34", + Wopf: "\ud835\udd4e", + wopf: "\ud835\udd68", + wp: "\u2118", + wr: "\u2240", + wreath: "\u2240", + Wscr: "\ud835\udcb2", + wscr: "\ud835\udccc", + xcap: "\u22c2", + xcirc: "\u25ef", + xcup: "\u22c3", + xdtri: "\u25bd", + Xfr: "\ud835\udd1b", + xfr: "\ud835\udd35", + xharr: "\u27f7", + xhArr: "\u27fa", + Xi: "\u039e", + xi: "\u03be", + xlarr: "\u27f5", + xlArr: "\u27f8", + xmap: "\u27fc", + xnis: "\u22fb", + xodot: "\u2a00", + Xopf: "\ud835\udd4f", + xopf: "\ud835\udd69", + xoplus: "\u2a01", + xotime: "\u2a02", + xrarr: "\u27f6", + xrArr: "\u27f9", + Xscr: "\ud835\udcb3", + xscr: "\ud835\udccd", + xsqcup: "\u2a06", + xuplus: "\u2a04", + xutri: "\u25b3", + xvee: "\u22c1", + xwedge: "\u22c0", + Yacute: "\xdd", + yacute: "\xfd", + YAcy: "\u042f", + yacy: "\u044f", + Ycirc: "\u0176", + ycirc: "\u0177", + Ycy: "\u042b", + ycy: "\u044b", + yen: "\xa5", + Yfr: "\ud835\udd1c", + yfr: "\ud835\udd36", + YIcy: "\u0407", + yicy: "\u0457", + Yopf: "\ud835\udd50", + yopf: "\ud835\udd6a", + Yscr: "\ud835\udcb4", + yscr: "\ud835\udcce", + YUcy: "\u042e", + yucy: "\u044e", + yuml: "\xff", + Yuml: "\u0178", + Zacute: "\u0179", + zacute: "\u017a", + Zcaron: "\u017d", + zcaron: "\u017e", + Zcy: "\u0417", + zcy: "\u0437", + Zdot: "\u017b", + zdot: "\u017c", + zeetrf: "\u2128", + ZeroWidthSpace: "\u200b", + Zeta: "\u0396", + zeta: "\u03b6", + zfr: "\ud835\udd37", + Zfr: "\u2128", + ZHcy: "\u0416", + zhcy: "\u0436", + zigrarr: "\u21dd", + zopf: "\ud835\udd6b", + Zopf: "\u2124", + Zscr: "\ud835\udcb5", + zscr: "\ud835\udccf", + zwj: "\u200d", + zwnj: "\u200c" + }; + /*eslint quotes:0*/ var entities = require$$0; + var regex$4 = /[!-#%-\*,-\/:;\?@\[-\]_\{\}\xA1\xA7\xAB\xB6\xB7\xBB\xBF\u037E\u0387\u055A-\u055F\u0589\u058A\u05BE\u05C0\u05C3\u05C6\u05F3\u05F4\u0609\u060A\u060C\u060D\u061B\u061E\u061F\u066A-\u066D\u06D4\u0700-\u070D\u07F7-\u07F9\u0830-\u083E\u085E\u0964\u0965\u0970\u09FD\u0A76\u0AF0\u0C84\u0DF4\u0E4F\u0E5A\u0E5B\u0F04-\u0F12\u0F14\u0F3A-\u0F3D\u0F85\u0FD0-\u0FD4\u0FD9\u0FDA\u104A-\u104F\u10FB\u1360-\u1368\u1400\u166D\u166E\u169B\u169C\u16EB-\u16ED\u1735\u1736\u17D4-\u17D6\u17D8-\u17DA\u1800-\u180A\u1944\u1945\u1A1E\u1A1F\u1AA0-\u1AA6\u1AA8-\u1AAD\u1B5A-\u1B60\u1BFC-\u1BFF\u1C3B-\u1C3F\u1C7E\u1C7F\u1CC0-\u1CC7\u1CD3\u2010-\u2027\u2030-\u2043\u2045-\u2051\u2053-\u205E\u207D\u207E\u208D\u208E\u2308-\u230B\u2329\u232A\u2768-\u2775\u27C5\u27C6\u27E6-\u27EF\u2983-\u2998\u29D8-\u29DB\u29FC\u29FD\u2CF9-\u2CFC\u2CFE\u2CFF\u2D70\u2E00-\u2E2E\u2E30-\u2E4E\u3001-\u3003\u3008-\u3011\u3014-\u301F\u3030\u303D\u30A0\u30FB\uA4FE\uA4FF\uA60D-\uA60F\uA673\uA67E\uA6F2-\uA6F7\uA874-\uA877\uA8CE\uA8CF\uA8F8-\uA8FA\uA8FC\uA92E\uA92F\uA95F\uA9C1-\uA9CD\uA9DE\uA9DF\uAA5C-\uAA5F\uAADE\uAADF\uAAF0\uAAF1\uABEB\uFD3E\uFD3F\uFE10-\uFE19\uFE30-\uFE52\uFE54-\uFE61\uFE63\uFE68\uFE6A\uFE6B\uFF01-\uFF03\uFF05-\uFF0A\uFF0C-\uFF0F\uFF1A\uFF1B\uFF1F\uFF20\uFF3B-\uFF3D\uFF3F\uFF5B\uFF5D\uFF5F-\uFF65]|\uD800[\uDD00-\uDD02\uDF9F\uDFD0]|\uD801\uDD6F|\uD802[\uDC57\uDD1F\uDD3F\uDE50-\uDE58\uDE7F\uDEF0-\uDEF6\uDF39-\uDF3F\uDF99-\uDF9C]|\uD803[\uDF55-\uDF59]|\uD804[\uDC47-\uDC4D\uDCBB\uDCBC\uDCBE-\uDCC1\uDD40-\uDD43\uDD74\uDD75\uDDC5-\uDDC8\uDDCD\uDDDB\uDDDD-\uDDDF\uDE38-\uDE3D\uDEA9]|\uD805[\uDC4B-\uDC4F\uDC5B\uDC5D\uDCC6\uDDC1-\uDDD7\uDE41-\uDE43\uDE60-\uDE6C\uDF3C-\uDF3E]|\uD806[\uDC3B\uDE3F-\uDE46\uDE9A-\uDE9C\uDE9E-\uDEA2]|\uD807[\uDC41-\uDC45\uDC70\uDC71\uDEF7\uDEF8]|\uD809[\uDC70-\uDC74]|\uD81A[\uDE6E\uDE6F\uDEF5\uDF37-\uDF3B\uDF44]|\uD81B[\uDE97-\uDE9A]|\uD82F\uDC9F|\uD836[\uDE87-\uDE8B]|\uD83A[\uDD5E\uDD5F]/; + var encodeCache = {}; + // Create a lookup array where anything but characters in `chars` string + // and alphanumeric chars is percent-encoded. + + function getEncodeCache(exclude) { + var i, ch, cache = encodeCache[exclude]; + if (cache) { + return cache; + } + cache = encodeCache[exclude] = []; + for (i = 0; i < 128; i++) { + ch = String.fromCharCode(i); + if (/^[0-9a-z]$/i.test(ch)) { + // always allow unencoded alphanumeric characters + cache.push(ch); + } else { + cache.push("%" + ("0" + i.toString(16).toUpperCase()).slice(-2)); + } + } + for (i = 0; i < exclude.length; i++) { + cache[exclude.charCodeAt(i)] = exclude[i]; + } + return cache; + } + // Encode unsafe characters with percent-encoding, skipping already + // encoded sequences. + + // - string - string to encode + // - exclude - list of characters to ignore (in addition to a-zA-Z0-9) + // - keepEscaped - don't encode '%' in a correct escape sequence (default: true) + + function encode$2(string, exclude, keepEscaped) { + var i, l, code, nextCode, cache, result = ""; + if (typeof exclude !== "string") { + // encode(string, keepEscaped) + keepEscaped = exclude; + exclude = encode$2.defaultChars; + } + if (typeof keepEscaped === "undefined") { + keepEscaped = true; + } + cache = getEncodeCache(exclude); + for (i = 0, l = string.length; i < l; i++) { + code = string.charCodeAt(i); + if (keepEscaped && code === 37 /* % */ && i + 2 < l) { + if (/^[0-9a-f]{2}$/i.test(string.slice(i + 1, i + 3))) { + result += string.slice(i, i + 3); + i += 2; + continue; + } + } + if (code < 128) { + result += cache[code]; + continue; + } + if (code >= 55296 && code <= 57343) { + if (code >= 55296 && code <= 56319 && i + 1 < l) { + nextCode = string.charCodeAt(i + 1); + if (nextCode >= 56320 && nextCode <= 57343) { + result += encodeURIComponent(string[i] + string[i + 1]); + i++; + continue; + } + } + result += "%EF%BF%BD"; + continue; + } + result += encodeURIComponent(string[i]); + } + return result; + } + encode$2.defaultChars = ";/?:@&=+$,-_.!~*'()#"; + encode$2.componentChars = "-_.!~*'()"; + var encode_1 = encode$2; + /* eslint-disable no-bitwise */ var decodeCache = {}; + function getDecodeCache(exclude) { + var i, ch, cache = decodeCache[exclude]; + if (cache) { + return cache; + } + cache = decodeCache[exclude] = []; + for (i = 0; i < 128; i++) { + ch = String.fromCharCode(i); + cache.push(ch); + } + for (i = 0; i < exclude.length; i++) { + ch = exclude.charCodeAt(i); + cache[ch] = "%" + ("0" + ch.toString(16).toUpperCase()).slice(-2); + } + return cache; + } + // Decode percent-encoded string. + + function decode$2(string, exclude) { + var cache; + if (typeof exclude !== "string") { + exclude = decode$2.defaultChars; + } + cache = getDecodeCache(exclude); + return string.replace(/(%[a-f0-9]{2})+/gi, (function(seq) { + var i, l, b1, b2, b3, b4, chr, result = ""; + for (i = 0, l = seq.length; i < l; i += 3) { + b1 = parseInt(seq.slice(i + 1, i + 3), 16); + if (b1 < 128) { + result += cache[b1]; + continue; + } + if ((b1 & 224) === 192 && i + 3 < l) { + // 110xxxxx 10xxxxxx + b2 = parseInt(seq.slice(i + 4, i + 6), 16); + if ((b2 & 192) === 128) { + chr = b1 << 6 & 1984 | b2 & 63; + if (chr < 128) { + result += "\ufffd\ufffd"; + } else { + result += String.fromCharCode(chr); + } + i += 3; + continue; + } + } + if ((b1 & 240) === 224 && i + 6 < l) { + // 1110xxxx 10xxxxxx 10xxxxxx + b2 = parseInt(seq.slice(i + 4, i + 6), 16); + b3 = parseInt(seq.slice(i + 7, i + 9), 16); + if ((b2 & 192) === 128 && (b3 & 192) === 128) { + chr = b1 << 12 & 61440 | b2 << 6 & 4032 | b3 & 63; + if (chr < 2048 || chr >= 55296 && chr <= 57343) { + result += "\ufffd\ufffd\ufffd"; + } else { + result += String.fromCharCode(chr); + } + i += 6; + continue; + } + } + if ((b1 & 248) === 240 && i + 9 < l) { + // 111110xx 10xxxxxx 10xxxxxx 10xxxxxx + b2 = parseInt(seq.slice(i + 4, i + 6), 16); + b3 = parseInt(seq.slice(i + 7, i + 9), 16); + b4 = parseInt(seq.slice(i + 10, i + 12), 16); + if ((b2 & 192) === 128 && (b3 & 192) === 128 && (b4 & 192) === 128) { + chr = b1 << 18 & 1835008 | b2 << 12 & 258048 | b3 << 6 & 4032 | b4 & 63; + if (chr < 65536 || chr > 1114111) { + result += "\ufffd\ufffd\ufffd\ufffd"; + } else { + chr -= 65536; + result += String.fromCharCode(55296 + (chr >> 10), 56320 + (chr & 1023)); + } + i += 9; + continue; + } + } + result += "\ufffd"; + } + return result; + })); + } + decode$2.defaultChars = ";/?:@&=+$,#"; + decode$2.componentChars = ""; + var decode_1 = decode$2; + var format$1 = function format(url) { + var result = ""; + result += url.protocol || ""; + result += url.slashes ? "//" : ""; + result += url.auth ? url.auth + "@" : ""; + if (url.hostname && url.hostname.indexOf(":") !== -1) { + // ipv6 address + result += "[" + url.hostname + "]"; + } else { + result += url.hostname || ""; + } + result += url.port ? ":" + url.port : ""; + result += url.pathname || ""; + result += url.search || ""; + result += url.hash || ""; + return result; + }; + // Copyright Joyent, Inc. and other Node contributors. + + // Changes from joyent/node: + + // 1. No leading slash in paths, + // e.g. in `url.parse('http://foo?bar')` pathname is ``, not `/` + + // 2. Backslashes are not replaced with slashes, + // so `http:\\example.org\` is treated like a relative path + + // 3. Trailing colon is treated like a part of the path, + // i.e. in `http://example.org:foo` pathname is `:foo` + + // 4. Nothing is URL-encoded in the resulting object, + // (in joyent/node some chars in auth and paths are encoded) + + // 5. `url.parse()` does not have `parseQueryString` argument + + // 6. Removed extraneous result properties: `host`, `path`, `query`, etc., + // which can be constructed using other parts of the url. + + function Url() { + this.protocol = null; + this.slashes = null; + this.auth = null; + this.port = null; + this.hostname = null; + this.hash = null; + this.search = null; + this.pathname = null; + } + // Reference: RFC 3986, RFC 1808, RFC 2396 + // define these here so at least they only have to be + // compiled once on the first module load. + var protocolPattern = /^([a-z0-9.+-]+:)/i, portPattern = /:[0-9]*$/, + // Special case for a simple path URL + simplePathPattern = /^(\/\/?(?!\/)[^\?\s]*)(\?[^\s]*)?$/, + // RFC 2396: characters reserved for delimiting URLs. + // We actually just auto-escape these. + delims = [ "<", ">", '"', "`", " ", "\r", "\n", "\t" ], + // RFC 2396: characters not allowed for various reasons. + unwise = [ "{", "}", "|", "\\", "^", "`" ].concat(delims), + // Allowed by RFCs, but cause of XSS attacks. Always escape these. + autoEscape = [ "'" ].concat(unwise), + // Characters that are never ever allowed in a hostname. + // Note that any invalid chars are also handled, but these + // are the ones that are *expected* to be seen, so we fast-path + // them. + nonHostChars = [ "%", "/", "?", ";", "#" ].concat(autoEscape), hostEndingChars = [ "/", "?", "#" ], hostnameMaxLen = 255, hostnamePartPattern = /^[+a-z0-9A-Z_-]{0,63}$/, hostnamePartStart = /^([+a-z0-9A-Z_-]{0,63})(.*)$/, + // protocols that can allow "unsafe" and "unwise" chars. + /* eslint-disable no-script-url */ + // protocols that never have a hostname. + hostlessProtocol = { + javascript: true, + "javascript:": true + }, + // protocols that always contain a // bit. + slashedProtocol = { + http: true, + https: true, + ftp: true, + gopher: true, + file: true, + "http:": true, + "https:": true, + "ftp:": true, + "gopher:": true, + "file:": true + }; + /* eslint-enable no-script-url */ function urlParse(url, slashesDenoteHost) { + if (url && url instanceof Url) { + return url; + } + var u = new Url; + u.parse(url, slashesDenoteHost); + return u; + } + Url.prototype.parse = function(url, slashesDenoteHost) { + var i, l, lowerProto, hec, slashes, rest = url; + // trim before proceeding. + // This is to support parse stuff like " http://foo.com \n" + rest = rest.trim(); + if (!slashesDenoteHost && url.split("#").length === 1) { + // Try fast path regexp + var simplePath = simplePathPattern.exec(rest); + if (simplePath) { + this.pathname = simplePath[1]; + if (simplePath[2]) { + this.search = simplePath[2]; + } + return this; + } + } + var proto = protocolPattern.exec(rest); + if (proto) { + proto = proto[0]; + lowerProto = proto.toLowerCase(); + this.protocol = proto; + rest = rest.substr(proto.length); + } + // figure out if it's got a host + // user@server is *always* interpreted as a hostname, and url + // resolution will treat //foo/bar as host=foo,path=bar because that's + // how the browser resolves relative URLs. + if (slashesDenoteHost || proto || rest.match(/^\/\/[^@\/]+@[^@\/]+/)) { + slashes = rest.substr(0, 2) === "//"; + if (slashes && !(proto && hostlessProtocol[proto])) { + rest = rest.substr(2); + this.slashes = true; + } + } + if (!hostlessProtocol[proto] && (slashes || proto && !slashedProtocol[proto])) { + // there's a hostname. + // the first instance of /, ?, ;, or # ends the host. + // If there is an @ in the hostname, then non-host chars *are* allowed + // to the left of the last @ sign, unless some host-ending character + // comes *before* the @-sign. + // URLs are obnoxious. + // ex: + // http://a@b@c/ => user:a@b host:c + // http://a@b?@c => user:a host:c path:/?@c + // v0.12 TODO(isaacs): This is not quite how Chrome does things. + // Review our test case against browsers more comprehensively. + // find the first instance of any hostEndingChars + var hostEnd = -1; + for (i = 0; i < hostEndingChars.length; i++) { + hec = rest.indexOf(hostEndingChars[i]); + if (hec !== -1 && (hostEnd === -1 || hec < hostEnd)) { + hostEnd = hec; + } + } + // at this point, either we have an explicit point where the + // auth portion cannot go past, or the last @ char is the decider. + var auth, atSign; + if (hostEnd === -1) { + // atSign can be anywhere. + atSign = rest.lastIndexOf("@"); + } else { + // atSign must be in auth portion. + // http://a@b/c@d => host:b auth:a path:/c@d + atSign = rest.lastIndexOf("@", hostEnd); + } + // Now we have a portion which is definitely the auth. + // Pull that off. + if (atSign !== -1) { + auth = rest.slice(0, atSign); + rest = rest.slice(atSign + 1); + this.auth = auth; + } + // the host is the remaining to the left of the first non-host char + hostEnd = -1; + for (i = 0; i < nonHostChars.length; i++) { + hec = rest.indexOf(nonHostChars[i]); + if (hec !== -1 && (hostEnd === -1 || hec < hostEnd)) { + hostEnd = hec; + } + } + // if we still have not hit it, then the entire thing is a host. + if (hostEnd === -1) { + hostEnd = rest.length; + } + if (rest[hostEnd - 1] === ":") { + hostEnd--; + } + var host = rest.slice(0, hostEnd); + rest = rest.slice(hostEnd); + // pull out port. + this.parseHost(host); + // we've indicated that there is a hostname, + // so even if it's empty, it has to be present. + this.hostname = this.hostname || ""; + // if hostname begins with [ and ends with ] + // assume that it's an IPv6 address. + var ipv6Hostname = this.hostname[0] === "[" && this.hostname[this.hostname.length - 1] === "]"; + // validate a little. + if (!ipv6Hostname) { + var hostparts = this.hostname.split(/\./); + for (i = 0, l = hostparts.length; i < l; i++) { + var part = hostparts[i]; + if (!part) { + continue; + } + if (!part.match(hostnamePartPattern)) { + var newpart = ""; + for (var j = 0, k = part.length; j < k; j++) { + if (part.charCodeAt(j) > 127) { + // we replace non-ASCII char with a temporary placeholder + // we need this to make sure size of hostname is not + // broken by replacing non-ASCII by nothing + newpart += "x"; + } else { + newpart += part[j]; + } + } + // we test again with ASCII char only + if (!newpart.match(hostnamePartPattern)) { + var validParts = hostparts.slice(0, i); + var notHost = hostparts.slice(i + 1); + var bit = part.match(hostnamePartStart); + if (bit) { + validParts.push(bit[1]); + notHost.unshift(bit[2]); + } + if (notHost.length) { + rest = notHost.join(".") + rest; + } + this.hostname = validParts.join("."); + break; + } + } + } + } + if (this.hostname.length > hostnameMaxLen) { + this.hostname = ""; + } + // strip [ and ] from the hostname + // the host field still retains them, though + if (ipv6Hostname) { + this.hostname = this.hostname.substr(1, this.hostname.length - 2); + } + } + // chop off from the tail first. + var hash = rest.indexOf("#"); + if (hash !== -1) { + // got a fragment string. + this.hash = rest.substr(hash); + rest = rest.slice(0, hash); + } + var qm = rest.indexOf("?"); + if (qm !== -1) { + this.search = rest.substr(qm); + rest = rest.slice(0, qm); + } + if (rest) { + this.pathname = rest; + } + if (slashedProtocol[lowerProto] && this.hostname && !this.pathname) { + this.pathname = ""; + } + return this; + }; + Url.prototype.parseHost = function(host) { + var port = portPattern.exec(host); + if (port) { + port = port[0]; + if (port !== ":") { + this.port = port.substr(1); + } + host = host.substr(0, host.length - port.length); + } + if (host) { + this.hostname = host; + } + }; + var parse$1 = urlParse; + var encode$1 = encode_1; + var decode$1 = decode_1; + var format = format$1; + var parse = parse$1; + var mdurl = { + encode: encode$1, + decode: decode$1, + format: format, + parse: parse + }; + var regex$3 = /[\0-\uD7FF\uE000-\uFFFF]|[\uD800-\uDBFF][\uDC00-\uDFFF]|[\uD800-\uDBFF](?![\uDC00-\uDFFF])|(?:[^\uD800-\uDBFF]|^)[\uDC00-\uDFFF]/; + var regex$2 = /[\0-\x1F\x7F-\x9F]/; + var regex$1 = /[\xAD\u0600-\u0605\u061C\u06DD\u070F\u08E2\u180E\u200B-\u200F\u202A-\u202E\u2060-\u2064\u2066-\u206F\uFEFF\uFFF9-\uFFFB]|\uD804[\uDCBD\uDCCD]|\uD82F[\uDCA0-\uDCA3]|\uD834[\uDD73-\uDD7A]|\uDB40[\uDC01\uDC20-\uDC7F]/; + var regex = /[ \xA0\u1680\u2000-\u200A\u2028\u2029\u202F\u205F\u3000]/; + var Any = regex$3; + var Cc = regex$2; + var Cf = regex$1; + var P = regex$4; + var Z = regex; + var uc_micro = { + Any: Any, + Cc: Cc, + Cf: Cf, + P: P, + Z: Z + }; + var utils = createCommonjsModule((function(module, exports) { + function _class(obj) { + return Object.prototype.toString.call(obj); + } + function isString(obj) { + return _class(obj) === "[object String]"; + } + var _hasOwnProperty = Object.prototype.hasOwnProperty; + function has(object, key) { + return _hasOwnProperty.call(object, key); + } + // Merge objects + + function assign(obj /*from1, from2, from3, ...*/) { + var sources = Array.prototype.slice.call(arguments, 1); + sources.forEach((function(source) { + if (!source) { + return; + } + if (typeof source !== "object") { + throw new TypeError(source + "must be object"); + } + Object.keys(source).forEach((function(key) { + obj[key] = source[key]; + })); + })); + return obj; + } + // Remove element from array and put another array at those position. + // Useful for some operations with tokens + function arrayReplaceAt(src, pos, newElements) { + return [].concat(src.slice(0, pos), newElements, src.slice(pos + 1)); + } + //////////////////////////////////////////////////////////////////////////////// + function isValidEntityCode(c) { + /*eslint no-bitwise:0*/ + // broken sequence + if (c >= 55296 && c <= 57343) { + return false; + } + // never used + if (c >= 64976 && c <= 65007) { + return false; + } + if ((c & 65535) === 65535 || (c & 65535) === 65534) { + return false; + } + // control codes + if (c >= 0 && c <= 8) { + return false; + } + if (c === 11) { + return false; + } + if (c >= 14 && c <= 31) { + return false; + } + if (c >= 127 && c <= 159) { + return false; + } + // out of range + if (c > 1114111) { + return false; + } + return true; + } + function fromCodePoint(c) { + /*eslint no-bitwise:0*/ + if (c > 65535) { + c -= 65536; + var surrogate1 = 55296 + (c >> 10), surrogate2 = 56320 + (c & 1023); + return String.fromCharCode(surrogate1, surrogate2); + } + return String.fromCharCode(c); + } + var UNESCAPE_MD_RE = /\\([!"#$%&'()*+,\-.\/:;<=>?@[\\\]^_`{|}~])/g; + var ENTITY_RE = /&([a-z#][a-z0-9]{1,31});/gi; + var UNESCAPE_ALL_RE = new RegExp(UNESCAPE_MD_RE.source + "|" + ENTITY_RE.source, "gi"); + var DIGITAL_ENTITY_TEST_RE = /^#((?:x[a-f0-9]{1,8}|[0-9]{1,8}))/i; + function replaceEntityPattern(match, name) { + var code = 0; + if (has(entities, name)) { + return entities[name]; + } + if (name.charCodeAt(0) === 35 /* # */ && DIGITAL_ENTITY_TEST_RE.test(name)) { + code = name[1].toLowerCase() === "x" ? parseInt(name.slice(2), 16) : parseInt(name.slice(1), 10); + if (isValidEntityCode(code)) { + return fromCodePoint(code); + } + } + return match; + } + /*function replaceEntities(str) { + if (str.indexOf('&') < 0) { return str; } + + return str.replace(ENTITY_RE, replaceEntityPattern); + }*/ function unescapeMd(str) { + if (str.indexOf("\\") < 0) { + return str; + } + return str.replace(UNESCAPE_MD_RE, "$1"); + } + function unescapeAll(str) { + if (str.indexOf("\\") < 0 && str.indexOf("&") < 0) { + return str; + } + return str.replace(UNESCAPE_ALL_RE, (function(match, escaped, entity) { + if (escaped) { + return escaped; + } + return replaceEntityPattern(match, entity); + })); + } + //////////////////////////////////////////////////////////////////////////////// + var HTML_ESCAPE_TEST_RE = /[&<>"]/; + var HTML_ESCAPE_REPLACE_RE = /[&<>"]/g; + var HTML_REPLACEMENTS = { + "&": "&", + "<": "<", + ">": ">", + '"': """ + }; + function replaceUnsafeChar(ch) { + return HTML_REPLACEMENTS[ch]; + } + function escapeHtml(str) { + if (HTML_ESCAPE_TEST_RE.test(str)) { + return str.replace(HTML_ESCAPE_REPLACE_RE, replaceUnsafeChar); + } + return str; + } + //////////////////////////////////////////////////////////////////////////////// + var REGEXP_ESCAPE_RE = /[.?*+^$[\]\\(){}|-]/g; + function escapeRE(str) { + return str.replace(REGEXP_ESCAPE_RE, "\\$&"); + } + //////////////////////////////////////////////////////////////////////////////// + function isSpace(code) { + switch (code) { + case 9: + case 32: + return true; + } + return false; + } + // Zs (unicode class) || [\t\f\v\r\n] + function isWhiteSpace(code) { + if (code >= 8192 && code <= 8202) { + return true; + } + switch (code) { + case 9: + // \t + case 10: + // \n + case 11: + // \v + case 12: + // \f + case 13: + // \r + case 32: + case 160: + case 5760: + case 8239: + case 8287: + case 12288: + return true; + } + return false; + } + //////////////////////////////////////////////////////////////////////////////// + /*eslint-disable max-len*/ + // Currently without astral characters support. + function isPunctChar(ch) { + return regex$4.test(ch); + } + // Markdown ASCII punctuation characters. + + // !, ", #, $, %, &, ', (, ), *, +, ,, -, ., /, :, ;, <, =, >, ?, @, [, \, ], ^, _, `, {, |, }, or ~ + // http://spec.commonmark.org/0.15/#ascii-punctuation-character + + // Don't confuse with unicode punctuation !!! It lacks some chars in ascii range. + + function isMdAsciiPunct(ch) { + switch (ch) { + case 33 /* ! */ : + case 34 /* " */ : + case 35 /* # */ : + case 36 /* $ */ : + case 37 /* % */ : + case 38 /* & */ : + case 39 /* ' */ : + case 40 /* ( */ : + case 41 /* ) */ : + case 42 /* * */ : + case 43 /* + */ : + case 44 /* , */ : + case 45 /* - */ : + case 46 /* . */ : + case 47 /* / */ : + case 58 /* : */ : + case 59 /* ; */ : + case 60 /* < */ : + case 61 /* = */ : + case 62 /* > */ : + case 63 /* ? */ : + case 64 /* @ */ : + case 91 /* [ */ : + case 92 /* \ */ : + case 93 /* ] */ : + case 94 /* ^ */ : + case 95 /* _ */ : + case 96 /* ` */ : + case 123 /* { */ : + case 124 /* | */ : + case 125 /* } */ : + case 126 /* ~ */ : + return true; + + default: + return false; + } + } + // Hepler to unify [reference labels]. + + function normalizeReference(str) { + // Trim and collapse whitespace + str = str.trim().replace(/\s+/g, " "); + // In node v10 'ẞ'.toLowerCase() === 'Ṿ', which is presumed to be a bug + // fixed in v12 (couldn't find any details). + + // So treat this one as a special case + // (remove this when node v10 is no longer supported). + + if ("\u1e9e".toLowerCase() === "\u1e7e") { + str = str.replace(/\u1e9e/g, "\xdf"); + } + // .toLowerCase().toUpperCase() should get rid of all differences + // between letter variants. + + // Simple .toLowerCase() doesn't normalize 125 code points correctly, + // and .toUpperCase doesn't normalize 6 of them (list of exceptions: + // İ, ϴ, ẞ, Ω, K, Å - those are already uppercased, but have differently + // uppercased versions). + + // Here's an example showing how it happens. Lets take greek letter omega: + // uppercase U+0398 (Θ), U+03f4 (ϴ) and lowercase U+03b8 (θ), U+03d1 (ϑ) + + // Unicode entries: + // 0398;GREEK CAPITAL LETTER THETA;Lu;0;L;;;;;N;;;;03B8; + // 03B8;GREEK SMALL LETTER THETA;Ll;0;L;;;;;N;;;0398;;0398 + // 03D1;GREEK THETA SYMBOL;Ll;0;L; 03B8;;;;N;GREEK SMALL LETTER SCRIPT THETA;;0398;;0398 + // 03F4;GREEK CAPITAL THETA SYMBOL;Lu;0;L; 0398;;;;N;;;;03B8; + + // Case-insensitive comparison should treat all of them as equivalent. + + // But .toLowerCase() doesn't change ϑ (it's already lowercase), + // and .toUpperCase() doesn't change ϴ (already uppercase). + + // Applying first lower then upper case normalizes any character: + // '\u0398\u03f4\u03b8\u03d1'.toLowerCase().toUpperCase() === '\u0398\u0398\u0398\u0398' + + // Note: this is equivalent to unicode case folding; unicode normalization + // is a different step that is not required here. + + // Final result should be uppercased, because it's later stored in an object + // (this avoid a conflict with Object.prototype members, + // most notably, `__proto__`) + + return str.toLowerCase().toUpperCase(); + } + //////////////////////////////////////////////////////////////////////////////// + // Re-export libraries commonly used in both markdown-it and its plugins, + // so plugins won't have to depend on them explicitly, which reduces their + // bundled size (e.g. a browser build). + + exports.lib = {}; + exports.lib.mdurl = mdurl; + exports.lib.ucmicro = uc_micro; + exports.assign = assign; + exports.isString = isString; + exports.has = has; + exports.unescapeMd = unescapeMd; + exports.unescapeAll = unescapeAll; + exports.isValidEntityCode = isValidEntityCode; + exports.fromCodePoint = fromCodePoint; + // exports.replaceEntities = replaceEntities; + exports.escapeHtml = escapeHtml; + exports.arrayReplaceAt = arrayReplaceAt; + exports.isSpace = isSpace; + exports.isWhiteSpace = isWhiteSpace; + exports.isMdAsciiPunct = isMdAsciiPunct; + exports.isPunctChar = isPunctChar; + exports.escapeRE = escapeRE; + exports.normalizeReference = normalizeReference; + })); + // Parse link label + var parse_link_label = function parseLinkLabel(state, start, disableNested) { + var level, found, marker, prevPos, labelEnd = -1, max = state.posMax, oldPos = state.pos; + state.pos = start + 1; + level = 1; + while (state.pos < max) { + marker = state.src.charCodeAt(state.pos); + if (marker === 93 /* ] */) { + level--; + if (level === 0) { + found = true; + break; + } + } + prevPos = state.pos; + state.md.inline.skipToken(state); + if (marker === 91 /* [ */) { + if (prevPos === state.pos - 1) { + // increase level if we find text `[`, which is not a part of any token + level++; + } else if (disableNested) { + state.pos = oldPos; + return -1; + } + } + } + if (found) { + labelEnd = state.pos; + } + // restore old state + state.pos = oldPos; + return labelEnd; + }; + var unescapeAll$2 = utils.unescapeAll; + var parse_link_destination = function parseLinkDestination(str, pos, max) { + var code, level, lines = 0, start = pos, result = { + ok: false, + pos: 0, + lines: 0, + str: "" + }; + if (str.charCodeAt(pos) === 60 /* < */) { + pos++; + while (pos < max) { + code = str.charCodeAt(pos); + if (code === 10 /* \n */) { + return result; + } + if (code === 60 /* < */) { + return result; + } + if (code === 62 /* > */) { + result.pos = pos + 1; + result.str = unescapeAll$2(str.slice(start + 1, pos)); + result.ok = true; + return result; + } + if (code === 92 /* \ */ && pos + 1 < max) { + pos += 2; + continue; + } + pos++; + } + // no closing '>' + return result; + } + // this should be ... } else { ... branch + level = 0; + while (pos < max) { + code = str.charCodeAt(pos); + if (code === 32) { + break; + } + // ascii control characters + if (code < 32 || code === 127) { + break; + } + if (code === 92 /* \ */ && pos + 1 < max) { + if (str.charCodeAt(pos + 1) === 32) { + break; + } + pos += 2; + continue; + } + if (code === 40 /* ( */) { + level++; + if (level > 32) { + return result; + } + } + if (code === 41 /* ) */) { + if (level === 0) { + break; + } + level--; + } + pos++; + } + if (start === pos) { + return result; + } + if (level !== 0) { + return result; + } + result.str = unescapeAll$2(str.slice(start, pos)); + result.lines = lines; + result.pos = pos; + result.ok = true; + return result; + }; + var unescapeAll$1 = utils.unescapeAll; + var parse_link_title = function parseLinkTitle(str, pos, max) { + var code, marker, lines = 0, start = pos, result = { + ok: false, + pos: 0, + lines: 0, + str: "" + }; + if (pos >= max) { + return result; + } + marker = str.charCodeAt(pos); + if (marker !== 34 /* " */ && marker !== 39 /* ' */ && marker !== 40 /* ( */) { + return result; + } + pos++; + // if opening marker is "(", switch it to closing marker ")" + if (marker === 40) { + marker = 41; + } + while (pos < max) { + code = str.charCodeAt(pos); + if (code === marker) { + result.pos = pos + 1; + result.lines = lines; + result.str = unescapeAll$1(str.slice(start + 1, pos)); + result.ok = true; + return result; + } else if (code === 40 /* ( */ && marker === 41 /* ) */) { + return result; + } else if (code === 10) { + lines++; + } else if (code === 92 /* \ */ && pos + 1 < max) { + pos++; + if (str.charCodeAt(pos) === 10) { + lines++; + } + } + pos++; + } + return result; + }; + var parseLinkLabel = parse_link_label; + var parseLinkDestination = parse_link_destination; + var parseLinkTitle = parse_link_title; + var helpers = { + parseLinkLabel: parseLinkLabel, + parseLinkDestination: parseLinkDestination, + parseLinkTitle: parseLinkTitle + }; + var assign$1 = utils.assign; + var unescapeAll = utils.unescapeAll; + var escapeHtml = utils.escapeHtml; + //////////////////////////////////////////////////////////////////////////////// + var default_rules = {}; + default_rules.code_inline = function(tokens, idx, options, env, slf) { + var token = tokens[idx]; + return "" + escapeHtml(tokens[idx].content) + ""; + }; + default_rules.code_block = function(tokens, idx, options, env, slf) { + var token = tokens[idx]; + return "" + escapeHtml(tokens[idx].content) + "\n"; + }; + default_rules.fence = function(tokens, idx, options, env, slf) { + var token = tokens[idx], info = token.info ? unescapeAll(token.info).trim() : "", langName = "", langAttrs = "", highlighted, i, arr, tmpAttrs, tmpToken; + if (info) { + arr = info.split(/(\s+)/g); + langName = arr[0]; + langAttrs = arr.slice(2).join(""); + } + if (options.highlight) { + highlighted = options.highlight(token.content, langName, langAttrs) || escapeHtml(token.content); + } else { + highlighted = escapeHtml(token.content); + } + if (highlighted.indexOf("" + highlighted + "\n"; + } + return "

" + highlighted + "
\n"; + }; + default_rules.image = function(tokens, idx, options, env, slf) { + var token = tokens[idx]; + // "alt" attr MUST be set, even if empty. Because it's mandatory and + // should be placed on proper position for tests. + + // Replace content with actual value + token.attrs[token.attrIndex("alt")][1] = slf.renderInlineAsText(token.children, options, env); + return slf.renderToken(tokens, idx, options); + }; + default_rules.hardbreak = function(tokens, idx, options /*, env */) { + return options.xhtmlOut ? "
\n" : "
\n"; + }; + default_rules.softbreak = function(tokens, idx, options /*, env */) { + return options.breaks ? options.xhtmlOut ? "
\n" : "
\n" : "\n"; + }; + default_rules.text = function(tokens, idx /*, options, env */) { + return escapeHtml(tokens[idx].content); + }; + default_rules.html_block = function(tokens, idx /*, options, env */) { + return tokens[idx].content; + }; + default_rules.html_inline = function(tokens, idx /*, options, env */) { + return tokens[idx].content; + }; + /** + * new Renderer() + * + * Creates new [[Renderer]] instance and fill [[Renderer#rules]] with defaults. + **/ function Renderer() { + /** + * Renderer#rules -> Object + * + * Contains render rules for tokens. Can be updated and extended. + * + * ##### Example + * + * ```javascript + * var md = require('markdown-it')(); + * + * md.renderer.rules.strong_open = function () { return ''; }; + * md.renderer.rules.strong_close = function () { return ''; }; + * + * var result = md.renderInline(...); + * ``` + * + * Each rule is called as independent static function with fixed signature: + * + * ```javascript + * function my_token_render(tokens, idx, options, env, renderer) { + * // ... + * return renderedHTML; + * } + * ``` + * + * See [source code](https://github.com/markdown-it/markdown-it/blob/master/lib/renderer.js) + * for more details and examples. + **/ + this.rules = assign$1({}, default_rules); + } + /** + * Renderer.renderAttrs(token) -> String + * + * Render token attributes to string. + **/ Renderer.prototype.renderAttrs = function renderAttrs(token) { + var i, l, result; + if (!token.attrs) { + return ""; + } + result = ""; + for (i = 0, l = token.attrs.length; i < l; i++) { + result += " " + escapeHtml(token.attrs[i][0]) + '="' + escapeHtml(token.attrs[i][1]) + '"'; + } + return result; + }; + /** + * Renderer.renderToken(tokens, idx, options) -> String + * - tokens (Array): list of tokens + * - idx (Numbed): token index to render + * - options (Object): params of parser instance + * + * Default token renderer. Can be overriden by custom function + * in [[Renderer#rules]]. + **/ Renderer.prototype.renderToken = function renderToken(tokens, idx, options) { + var nextToken, result = "", needLf = false, token = tokens[idx]; + // Tight list paragraphs + if (token.hidden) { + return ""; + } + // Insert a newline between hidden paragraph and subsequent opening + // block-level tag. + + // For example, here we should insert a newline before blockquote: + // - a + // > + + if (token.block && token.nesting !== -1 && idx && tokens[idx - 1].hidden) { + result += "\n"; + } + // Add token name, e.g. ``. + needLf = false; + } + } + } + } + result += needLf ? ">\n" : ">"; + return result; + }; + /** + * Renderer.renderInline(tokens, options, env) -> String + * - tokens (Array): list on block tokens to render + * - options (Object): params of parser instance + * - env (Object): additional data from parsed input (references, for example) + * + * The same as [[Renderer.render]], but for single token of `inline` type. + **/ Renderer.prototype.renderInline = function(tokens, options, env) { + var type, result = "", rules = this.rules; + for (var i = 0, len = tokens.length; i < len; i++) { + type = tokens[i].type; + if (typeof rules[type] !== "undefined") { + result += rules[type](tokens, i, options, env, this); + } else { + result += this.renderToken(tokens, i, options); + } + } + return result; + }; + /** internal + * Renderer.renderInlineAsText(tokens, options, env) -> String + * - tokens (Array): list on block tokens to render + * - options (Object): params of parser instance + * - env (Object): additional data from parsed input (references, for example) + * + * Special kludge for image `alt` attributes to conform CommonMark spec. + * Don't try to use it! Spec requires to show `alt` content with stripped markup, + * instead of simple escaping. + **/ Renderer.prototype.renderInlineAsText = function(tokens, options, env) { + var result = ""; + for (var i = 0, len = tokens.length; i < len; i++) { + if (tokens[i].type === "text") { + result += tokens[i].content; + } else if (tokens[i].type === "image") { + result += this.renderInlineAsText(tokens[i].children, options, env); + } else if (tokens[i].type === "softbreak") { + result += "\n"; + } + } + return result; + }; + /** + * Renderer.render(tokens, options, env) -> String + * - tokens (Array): list on block tokens to render + * - options (Object): params of parser instance + * - env (Object): additional data from parsed input (references, for example) + * + * Takes token stream and generates HTML. Probably, you will never need to call + * this method directly. + **/ Renderer.prototype.render = function(tokens, options, env) { + var i, len, type, result = "", rules = this.rules; + for (i = 0, len = tokens.length; i < len; i++) { + type = tokens[i].type; + if (type === "inline") { + result += this.renderInline(tokens[i].children, options, env); + } else if (typeof rules[type] !== "undefined") { + result += rules[tokens[i].type](tokens, i, options, env, this); + } else { + result += this.renderToken(tokens, i, options, env); + } + } + return result; + }; + var renderer = Renderer; + /** + * class Ruler + * + * Helper class, used by [[MarkdownIt#core]], [[MarkdownIt#block]] and + * [[MarkdownIt#inline]] to manage sequences of functions (rules): + * + * - keep rules in defined order + * - assign the name to each rule + * - enable/disable rules + * - add/replace rules + * - allow assign rules to additional named chains (in the same) + * - cacheing lists of active rules + * + * You will not need use this class directly until write plugins. For simple + * rules control use [[MarkdownIt.disable]], [[MarkdownIt.enable]] and + * [[MarkdownIt.use]]. + **/ + /** + * new Ruler() + **/ function Ruler() { + // List of added rules. Each element is: + // { + // name: XXX, + // enabled: Boolean, + // fn: Function(), + // alt: [ name2, name3 ] + // } + this.__rules__ = []; + // Cached rule chains. + + // First level - chain name, '' for default. + // Second level - diginal anchor for fast filtering by charcodes. + + this.__cache__ = null; + } + //////////////////////////////////////////////////////////////////////////////// + // Helper methods, should not be used directly + // Find rule index by name + + Ruler.prototype.__find__ = function(name) { + for (var i = 0; i < this.__rules__.length; i++) { + if (this.__rules__[i].name === name) { + return i; + } + } + return -1; + }; + // Build rules lookup cache + + Ruler.prototype.__compile__ = function() { + var self = this; + var chains = [ "" ]; + // collect unique names + self.__rules__.forEach((function(rule) { + if (!rule.enabled) { + return; + } + rule.alt.forEach((function(altName) { + if (chains.indexOf(altName) < 0) { + chains.push(altName); + } + })); + })); + self.__cache__ = {}; + chains.forEach((function(chain) { + self.__cache__[chain] = []; + self.__rules__.forEach((function(rule) { + if (!rule.enabled) { + return; + } + if (chain && rule.alt.indexOf(chain) < 0) { + return; + } + self.__cache__[chain].push(rule.fn); + })); + })); + }; + /** + * Ruler.at(name, fn [, options]) + * - name (String): rule name to replace. + * - fn (Function): new rule function. + * - options (Object): new rule options (not mandatory). + * + * Replace rule by name with new function & options. Throws error if name not + * found. + * + * ##### Options: + * + * - __alt__ - array with names of "alternate" chains. + * + * ##### Example + * + * Replace existing typographer replacement rule with new one: + * + * ```javascript + * var md = require('markdown-it')(); + * + * md.core.ruler.at('replacements', function replace(state) { + * //... + * }); + * ``` + **/ Ruler.prototype.at = function(name, fn, options) { + var index = this.__find__(name); + var opt = options || {}; + if (index === -1) { + throw new Error("Parser rule not found: " + name); + } + this.__rules__[index].fn = fn; + this.__rules__[index].alt = opt.alt || []; + this.__cache__ = null; + }; + /** + * Ruler.before(beforeName, ruleName, fn [, options]) + * - beforeName (String): new rule will be added before this one. + * - ruleName (String): name of added rule. + * - fn (Function): rule function. + * - options (Object): rule options (not mandatory). + * + * Add new rule to chain before one with given name. See also + * [[Ruler.after]], [[Ruler.push]]. + * + * ##### Options: + * + * - __alt__ - array with names of "alternate" chains. + * + * ##### Example + * + * ```javascript + * var md = require('markdown-it')(); + * + * md.block.ruler.before('paragraph', 'my_rule', function replace(state) { + * //... + * }); + * ``` + **/ Ruler.prototype.before = function(beforeName, ruleName, fn, options) { + var index = this.__find__(beforeName); + var opt = options || {}; + if (index === -1) { + throw new Error("Parser rule not found: " + beforeName); + } + this.__rules__.splice(index, 0, { + name: ruleName, + enabled: true, + fn: fn, + alt: opt.alt || [] + }); + this.__cache__ = null; + }; + /** + * Ruler.after(afterName, ruleName, fn [, options]) + * - afterName (String): new rule will be added after this one. + * - ruleName (String): name of added rule. + * - fn (Function): rule function. + * - options (Object): rule options (not mandatory). + * + * Add new rule to chain after one with given name. See also + * [[Ruler.before]], [[Ruler.push]]. + * + * ##### Options: + * + * - __alt__ - array with names of "alternate" chains. + * + * ##### Example + * + * ```javascript + * var md = require('markdown-it')(); + * + * md.inline.ruler.after('text', 'my_rule', function replace(state) { + * //... + * }); + * ``` + **/ Ruler.prototype.after = function(afterName, ruleName, fn, options) { + var index = this.__find__(afterName); + var opt = options || {}; + if (index === -1) { + throw new Error("Parser rule not found: " + afterName); + } + this.__rules__.splice(index + 1, 0, { + name: ruleName, + enabled: true, + fn: fn, + alt: opt.alt || [] + }); + this.__cache__ = null; + }; + /** + * Ruler.push(ruleName, fn [, options]) + * - ruleName (String): name of added rule. + * - fn (Function): rule function. + * - options (Object): rule options (not mandatory). + * + * Push new rule to the end of chain. See also + * [[Ruler.before]], [[Ruler.after]]. + * + * ##### Options: + * + * - __alt__ - array with names of "alternate" chains. + * + * ##### Example + * + * ```javascript + * var md = require('markdown-it')(); + * + * md.core.ruler.push('my_rule', function replace(state) { + * //... + * }); + * ``` + **/ Ruler.prototype.push = function(ruleName, fn, options) { + var opt = options || {}; + this.__rules__.push({ + name: ruleName, + enabled: true, + fn: fn, + alt: opt.alt || [] + }); + this.__cache__ = null; + }; + /** + * Ruler.enable(list [, ignoreInvalid]) -> Array + * - list (String|Array): list of rule names to enable. + * - ignoreInvalid (Boolean): set `true` to ignore errors when rule not found. + * + * Enable rules with given names. If any rule name not found - throw Error. + * Errors can be disabled by second param. + * + * Returns list of found rule names (if no exception happened). + * + * See also [[Ruler.disable]], [[Ruler.enableOnly]]. + **/ Ruler.prototype.enable = function(list, ignoreInvalid) { + if (!Array.isArray(list)) { + list = [ list ]; + } + var result = []; + // Search by name and enable + list.forEach((function(name) { + var idx = this.__find__(name); + if (idx < 0) { + if (ignoreInvalid) { + return; + } + throw new Error("Rules manager: invalid rule name " + name); + } + this.__rules__[idx].enabled = true; + result.push(name); + }), this); + this.__cache__ = null; + return result; + }; + /** + * Ruler.enableOnly(list [, ignoreInvalid]) + * - list (String|Array): list of rule names to enable (whitelist). + * - ignoreInvalid (Boolean): set `true` to ignore errors when rule not found. + * + * Enable rules with given names, and disable everything else. If any rule name + * not found - throw Error. Errors can be disabled by second param. + * + * See also [[Ruler.disable]], [[Ruler.enable]]. + **/ Ruler.prototype.enableOnly = function(list, ignoreInvalid) { + if (!Array.isArray(list)) { + list = [ list ]; + } + this.__rules__.forEach((function(rule) { + rule.enabled = false; + })); + this.enable(list, ignoreInvalid); + }; + /** + * Ruler.disable(list [, ignoreInvalid]) -> Array + * - list (String|Array): list of rule names to disable. + * - ignoreInvalid (Boolean): set `true` to ignore errors when rule not found. + * + * Disable rules with given names. If any rule name not found - throw Error. + * Errors can be disabled by second param. + * + * Returns list of found rule names (if no exception happened). + * + * See also [[Ruler.enable]], [[Ruler.enableOnly]]. + **/ Ruler.prototype.disable = function(list, ignoreInvalid) { + if (!Array.isArray(list)) { + list = [ list ]; + } + var result = []; + // Search by name and disable + list.forEach((function(name) { + var idx = this.__find__(name); + if (idx < 0) { + if (ignoreInvalid) { + return; + } + throw new Error("Rules manager: invalid rule name " + name); + } + this.__rules__[idx].enabled = false; + result.push(name); + }), this); + this.__cache__ = null; + return result; + }; + /** + * Ruler.getRules(chainName) -> Array + * + * Return array of active functions (rules) for given chain name. It analyzes + * rules configuration, compiles caches if not exists and returns result. + * + * Default chain name is `''` (empty string). It can't be skipped. That's + * done intentionally, to keep signature monomorphic for high speed. + **/ Ruler.prototype.getRules = function(chainName) { + if (this.__cache__ === null) { + this.__compile__(); + } + // Chain can be empty, if rules disabled. But we still have to return Array. + return this.__cache__[chainName] || []; + }; + var ruler = Ruler; + // Normalize input string + // https://spec.commonmark.org/0.29/#line-ending + var NEWLINES_RE = /\r\n?|\n/g; + var NULL_RE = /\0/g; + var normalize = function normalize(state) { + var str; + // Normalize newlines + str = state.src.replace(NEWLINES_RE, "\n"); + // Replace NULL characters + str = str.replace(NULL_RE, "\ufffd"); + state.src = str; + }; + var block = function block(state) { + var token; + if (state.inlineMode) { + token = new state.Token("inline", "", 0); + token.content = state.src; + token.map = [ 0, 1 ]; + token.children = []; + state.tokens.push(token); + } else { + state.md.block.parse(state.src, state.md, state.env, state.tokens); + } + }; + var inline = function inline(state) { + var tokens = state.tokens, tok, i, l; + // Parse inlines + for (i = 0, l = tokens.length; i < l; i++) { + tok = tokens[i]; + if (tok.type === "inline") { + state.md.inline.parse(tok.content, state.md, state.env, tok.children); + } + } + }; + var arrayReplaceAt = utils.arrayReplaceAt; + function isLinkOpen$1(str) { + return /^\s]/i.test(str); + } + function isLinkClose$1(str) { + return /^<\/a\s*>/i.test(str); + } + var linkify$1 = function linkify(state) { + var i, j, l, tokens, token, currentToken, nodes, ln, text, pos, lastPos, level, htmlLinkLevel, url, fullUrl, urlText, blockTokens = state.tokens, links; + if (!state.md.options.linkify) { + return; + } + for (j = 0, l = blockTokens.length; j < l; j++) { + if (blockTokens[j].type !== "inline" || !state.md.linkify.pretest(blockTokens[j].content)) { + continue; + } + tokens = blockTokens[j].children; + htmlLinkLevel = 0; + // We scan from the end, to keep position when new tags added. + // Use reversed logic in links start/end match + for (i = tokens.length - 1; i >= 0; i--) { + currentToken = tokens[i]; + // Skip content of markdown links + if (currentToken.type === "link_close") { + i--; + while (tokens[i].level !== currentToken.level && tokens[i].type !== "link_open") { + i--; + } + continue; + } + // Skip content of html tag links + if (currentToken.type === "html_inline") { + if (isLinkOpen$1(currentToken.content) && htmlLinkLevel > 0) { + htmlLinkLevel--; + } + if (isLinkClose$1(currentToken.content)) { + htmlLinkLevel++; + } + } + if (htmlLinkLevel > 0) { + continue; + } + if (currentToken.type === "text" && state.md.linkify.test(currentToken.content)) { + text = currentToken.content; + links = state.md.linkify.match(text); + // Now split string to nodes + nodes = []; + level = currentToken.level; + lastPos = 0; + // forbid escape sequence at the start of the string, + // this avoids http\://example.com/ from being linkified as + // http://example.com/ + if (links.length > 0 && links[0].index === 0 && i > 0 && tokens[i - 1].type === "text_special") { + links = links.slice(1); + } + for (ln = 0; ln < links.length; ln++) { + url = links[ln].url; + fullUrl = state.md.normalizeLink(url); + if (!state.md.validateLink(fullUrl)) { + continue; + } + urlText = links[ln].text; + // Linkifier might send raw hostnames like "example.com", where url + // starts with domain name. So we prepend http:// in those cases, + // and remove it afterwards. + + if (!links[ln].schema) { + urlText = state.md.normalizeLinkText("http://" + urlText).replace(/^http:\/\//, ""); + } else if (links[ln].schema === "mailto:" && !/^mailto:/i.test(urlText)) { + urlText = state.md.normalizeLinkText("mailto:" + urlText).replace(/^mailto:/, ""); + } else { + urlText = state.md.normalizeLinkText(urlText); + } + pos = links[ln].index; + if (pos > lastPos) { + token = new state.Token("text", "", 0); + token.content = text.slice(lastPos, pos); + token.level = level; + nodes.push(token); + } + token = new state.Token("link_open", "a", 1); + token.attrs = [ [ "href", fullUrl ] ]; + token.level = level++; + token.markup = "linkify"; + token.info = "auto"; + nodes.push(token); + token = new state.Token("text", "", 0); + token.content = urlText; + token.level = level; + nodes.push(token); + token = new state.Token("link_close", "a", -1); + token.level = --level; + token.markup = "linkify"; + token.info = "auto"; + nodes.push(token); + lastPos = links[ln].lastIndex; + } + if (lastPos < text.length) { + token = new state.Token("text", "", 0); + token.content = text.slice(lastPos); + token.level = level; + nodes.push(token); + } + // replace current node + blockTokens[j].children = tokens = arrayReplaceAt(tokens, i, nodes); + } + } + } + }; + // Simple typographic replacements + // TODO: + // - fractionals 1/2, 1/4, 3/4 -> ½, ¼, ¾ + // - multiplications 2 x 4 -> 2 × 4 + var RARE_RE = /\+-|\.\.|\?\?\?\?|!!!!|,,|--/; + // Workaround for phantomjs - need regex without /g flag, + // or root check will fail every second time + var SCOPED_ABBR_TEST_RE = /\((c|tm|r)\)/i; + var SCOPED_ABBR_RE = /\((c|tm|r)\)/gi; + var SCOPED_ABBR = { + c: "\xa9", + r: "\xae", + tm: "\u2122" + }; + function replaceFn(match, name) { + return SCOPED_ABBR[name.toLowerCase()]; + } + function replace_scoped(inlineTokens) { + var i, token, inside_autolink = 0; + for (i = inlineTokens.length - 1; i >= 0; i--) { + token = inlineTokens[i]; + if (token.type === "text" && !inside_autolink) { + token.content = token.content.replace(SCOPED_ABBR_RE, replaceFn); + } + if (token.type === "link_open" && token.info === "auto") { + inside_autolink--; + } + if (token.type === "link_close" && token.info === "auto") { + inside_autolink++; + } + } + } + function replace_rare(inlineTokens) { + var i, token, inside_autolink = 0; + for (i = inlineTokens.length - 1; i >= 0; i--) { + token = inlineTokens[i]; + if (token.type === "text" && !inside_autolink) { + if (RARE_RE.test(token.content)) { + token.content = token.content.replace(/\+-/g, "\xb1").replace(/\.{2,}/g, "\u2026").replace(/([?!])\u2026/g, "$1..").replace(/([?!]){4,}/g, "$1$1$1").replace(/,{2,}/g, ",").replace(/(^|[^-])---(?=[^-]|$)/gm, "$1\u2014").replace(/(^|\s)--(?=\s|$)/gm, "$1\u2013").replace(/(^|[^-\s])--(?=[^-\s]|$)/gm, "$1\u2013"); + } + } + if (token.type === "link_open" && token.info === "auto") { + inside_autolink--; + } + if (token.type === "link_close" && token.info === "auto") { + inside_autolink++; + } + } + } + var replacements = function replace(state) { + var blkIdx; + if (!state.md.options.typographer) { + return; + } + for (blkIdx = state.tokens.length - 1; blkIdx >= 0; blkIdx--) { + if (state.tokens[blkIdx].type !== "inline") { + continue; + } + if (SCOPED_ABBR_TEST_RE.test(state.tokens[blkIdx].content)) { + replace_scoped(state.tokens[blkIdx].children); + } + if (RARE_RE.test(state.tokens[blkIdx].content)) { + replace_rare(state.tokens[blkIdx].children); + } + } + }; + var isWhiteSpace$1 = utils.isWhiteSpace; + var isPunctChar$1 = utils.isPunctChar; + var isMdAsciiPunct$1 = utils.isMdAsciiPunct; + var QUOTE_TEST_RE = /['"]/; + var QUOTE_RE = /['"]/g; + var APOSTROPHE = "\u2019"; + /* ’ */ function replaceAt(str, index, ch) { + return str.slice(0, index) + ch + str.slice(index + 1); + } + function process_inlines(tokens, state) { + var i, token, text, t, pos, max, thisLevel, item, lastChar, nextChar, isLastPunctChar, isNextPunctChar, isLastWhiteSpace, isNextWhiteSpace, canOpen, canClose, j, isSingle, stack, openQuote, closeQuote; + stack = []; + for (i = 0; i < tokens.length; i++) { + token = tokens[i]; + thisLevel = tokens[i].level; + for (j = stack.length - 1; j >= 0; j--) { + if (stack[j].level <= thisLevel) { + break; + } + } + stack.length = j + 1; + if (token.type !== "text") { + continue; + } + text = token.content; + pos = 0; + max = text.length; + /*eslint no-labels:0,block-scoped-var:0*/ OUTER: while (pos < max) { + QUOTE_RE.lastIndex = pos; + t = QUOTE_RE.exec(text); + if (!t) { + break; + } + canOpen = canClose = true; + pos = t.index + 1; + isSingle = t[0] === "'"; + // Find previous character, + // default to space if it's the beginning of the line + + lastChar = 32; + if (t.index - 1 >= 0) { + lastChar = text.charCodeAt(t.index - 1); + } else { + for (j = i - 1; j >= 0; j--) { + if (tokens[j].type === "softbreak" || tokens[j].type === "hardbreak") break; + // lastChar defaults to 0x20 + if (!tokens[j].content) continue; + // should skip all tokens except 'text', 'html_inline' or 'code_inline' + lastChar = tokens[j].content.charCodeAt(tokens[j].content.length - 1); + break; + } + } + // Find next character, + // default to space if it's the end of the line + + nextChar = 32; + if (pos < max) { + nextChar = text.charCodeAt(pos); + } else { + for (j = i + 1; j < tokens.length; j++) { + if (tokens[j].type === "softbreak" || tokens[j].type === "hardbreak") break; + // nextChar defaults to 0x20 + if (!tokens[j].content) continue; + // should skip all tokens except 'text', 'html_inline' or 'code_inline' + nextChar = tokens[j].content.charCodeAt(0); + break; + } + } + isLastPunctChar = isMdAsciiPunct$1(lastChar) || isPunctChar$1(String.fromCharCode(lastChar)); + isNextPunctChar = isMdAsciiPunct$1(nextChar) || isPunctChar$1(String.fromCharCode(nextChar)); + isLastWhiteSpace = isWhiteSpace$1(lastChar); + isNextWhiteSpace = isWhiteSpace$1(nextChar); + if (isNextWhiteSpace) { + canOpen = false; + } else if (isNextPunctChar) { + if (!(isLastWhiteSpace || isLastPunctChar)) { + canOpen = false; + } + } + if (isLastWhiteSpace) { + canClose = false; + } else if (isLastPunctChar) { + if (!(isNextWhiteSpace || isNextPunctChar)) { + canClose = false; + } + } + if (nextChar === 34 /* " */ && t[0] === '"') { + if (lastChar >= 48 /* 0 */ && lastChar <= 57 /* 9 */) { + // special case: 1"" - count first quote as an inch + canClose = canOpen = false; + } + } + if (canOpen && canClose) { + // Replace quotes in the middle of punctuation sequence, but not + // in the middle of the words, i.e.: + // 1. foo " bar " baz - not replaced + // 2. foo-"-bar-"-baz - replaced + // 3. foo"bar"baz - not replaced + canOpen = isLastPunctChar; + canClose = isNextPunctChar; + } + if (!canOpen && !canClose) { + // middle of word + if (isSingle) { + token.content = replaceAt(token.content, t.index, APOSTROPHE); + } + continue; + } + if (canClose) { + // this could be a closing quote, rewind the stack to get a match + for (j = stack.length - 1; j >= 0; j--) { + item = stack[j]; + if (stack[j].level < thisLevel) { + break; + } + if (item.single === isSingle && stack[j].level === thisLevel) { + item = stack[j]; + if (isSingle) { + openQuote = state.md.options.quotes[2]; + closeQuote = state.md.options.quotes[3]; + } else { + openQuote = state.md.options.quotes[0]; + closeQuote = state.md.options.quotes[1]; + } + // replace token.content *before* tokens[item.token].content, + // because, if they are pointing at the same token, replaceAt + // could mess up indices when quote length != 1 + token.content = replaceAt(token.content, t.index, closeQuote); + tokens[item.token].content = replaceAt(tokens[item.token].content, item.pos, openQuote); + pos += closeQuote.length - 1; + if (item.token === i) { + pos += openQuote.length - 1; + } + text = token.content; + max = text.length; + stack.length = j; + continue OUTER; + } + } + } + if (canOpen) { + stack.push({ + token: i, + pos: t.index, + single: isSingle, + level: thisLevel + }); + } else if (canClose && isSingle) { + token.content = replaceAt(token.content, t.index, APOSTROPHE); + } + } + } + } + var smartquotes = function smartquotes(state) { + /*eslint max-depth:0*/ + var blkIdx; + if (!state.md.options.typographer) { + return; + } + for (blkIdx = state.tokens.length - 1; blkIdx >= 0; blkIdx--) { + if (state.tokens[blkIdx].type !== "inline" || !QUOTE_TEST_RE.test(state.tokens[blkIdx].content)) { + continue; + } + process_inlines(state.tokens[blkIdx].children, state); + } + }; + // Join raw text tokens with the rest of the text + var text_join = function text_join(state) { + var j, l, tokens, curr, max, last, blockTokens = state.tokens; + for (j = 0, l = blockTokens.length; j < l; j++) { + if (blockTokens[j].type !== "inline") continue; + tokens = blockTokens[j].children; + max = tokens.length; + for (curr = 0; curr < max; curr++) { + if (tokens[curr].type === "text_special") { + tokens[curr].type = "text"; + } + } + for (curr = last = 0; curr < max; curr++) { + if (tokens[curr].type === "text" && curr + 1 < max && tokens[curr + 1].type === "text") { + // collapse two adjacent text nodes + tokens[curr + 1].content = tokens[curr].content + tokens[curr + 1].content; + } else { + if (curr !== last) { + tokens[last] = tokens[curr]; + } + last++; + } + } + if (curr !== last) { + tokens.length = last; + } + } + }; + // Token class + /** + * class Token + **/ + /** + * new Token(type, tag, nesting) + * + * Create new token and fill passed properties. + **/ function Token(type, tag, nesting) { + /** + * Token#type -> String + * + * Type of the token (string, e.g. "paragraph_open") + **/ + this.type = type; + /** + * Token#tag -> String + * + * html tag name, e.g. "p" + **/ this.tag = tag; + /** + * Token#attrs -> Array + * + * Html attributes. Format: `[ [ name1, value1 ], [ name2, value2 ] ]` + **/ this.attrs = null; + /** + * Token#map -> Array + * + * Source map info. Format: `[ line_begin, line_end ]` + **/ this.map = null; + /** + * Token#nesting -> Number + * + * Level change (number in {-1, 0, 1} set), where: + * + * - `1` means the tag is opening + * - `0` means the tag is self-closing + * - `-1` means the tag is closing + **/ this.nesting = nesting; + /** + * Token#level -> Number + * + * nesting level, the same as `state.level` + **/ this.level = 0; + /** + * Token#children -> Array + * + * An array of child nodes (inline and img tokens) + **/ this.children = null; + /** + * Token#content -> String + * + * In a case of self-closing tag (code, html, fence, etc.), + * it has contents of this tag. + **/ this.content = ""; + /** + * Token#markup -> String + * + * '*' or '_' for emphasis, fence string for fence, etc. + **/ this.markup = ""; + /** + * Token#info -> String + * + * Additional information: + * + * - Info string for "fence" tokens + * - The value "auto" for autolink "link_open" and "link_close" tokens + * - The string value of the item marker for ordered-list "list_item_open" tokens + **/ this.info = ""; + /** + * Token#meta -> Object + * + * A place for plugins to store an arbitrary data + **/ this.meta = null; + /** + * Token#block -> Boolean + * + * True for block-level tokens, false for inline tokens. + * Used in renderer to calculate line breaks + **/ this.block = false; + /** + * Token#hidden -> Boolean + * + * If it's true, ignore this element when rendering. Used for tight lists + * to hide paragraphs. + **/ this.hidden = false; + } + /** + * Token.attrIndex(name) -> Number + * + * Search attribute index by name. + **/ Token.prototype.attrIndex = function attrIndex(name) { + var attrs, i, len; + if (!this.attrs) { + return -1; + } + attrs = this.attrs; + for (i = 0, len = attrs.length; i < len; i++) { + if (attrs[i][0] === name) { + return i; + } + } + return -1; + }; + /** + * Token.attrPush(attrData) + * + * Add `[ name, value ]` attribute to list. Init attrs if necessary + **/ Token.prototype.attrPush = function attrPush(attrData) { + if (this.attrs) { + this.attrs.push(attrData); + } else { + this.attrs = [ attrData ]; + } + }; + /** + * Token.attrSet(name, value) + * + * Set `name` attribute to `value`. Override old value if exists. + **/ Token.prototype.attrSet = function attrSet(name, value) { + var idx = this.attrIndex(name), attrData = [ name, value ]; + if (idx < 0) { + this.attrPush(attrData); + } else { + this.attrs[idx] = attrData; + } + }; + /** + * Token.attrGet(name) + * + * Get the value of attribute `name`, or null if it does not exist. + **/ Token.prototype.attrGet = function attrGet(name) { + var idx = this.attrIndex(name), value = null; + if (idx >= 0) { + value = this.attrs[idx][1]; + } + return value; + }; + /** + * Token.attrJoin(name, value) + * + * Join value to existing attribute via space. Or create new attribute if not + * exists. Useful to operate with token classes. + **/ Token.prototype.attrJoin = function attrJoin(name, value) { + var idx = this.attrIndex(name); + if (idx < 0) { + this.attrPush([ name, value ]); + } else { + this.attrs[idx][1] = this.attrs[idx][1] + " " + value; + } + }; + var token = Token; + function StateCore(src, md, env) { + this.src = src; + this.env = env; + this.tokens = []; + this.inlineMode = false; + this.md = md; + // link to parser instance + } + // re-export Token class to use in core rules + StateCore.prototype.Token = token; + var state_core = StateCore; + var _rules$2 = [ [ "normalize", normalize ], [ "block", block ], [ "inline", inline ], [ "linkify", linkify$1 ], [ "replacements", replacements ], [ "smartquotes", smartquotes ], + // `text_join` finds `text_special` tokens (for escape sequences) + // and joins them with the rest of the text + [ "text_join", text_join ] ]; + /** + * new Core() + **/ function Core() { + /** + * Core#ruler -> Ruler + * + * [[Ruler]] instance. Keep configuration of core rules. + **/ + this.ruler = new ruler; + for (var i = 0; i < _rules$2.length; i++) { + this.ruler.push(_rules$2[i][0], _rules$2[i][1]); + } + } + /** + * Core.process(state) + * + * Executes core chain rules. + **/ Core.prototype.process = function(state) { + var i, l, rules; + rules = this.ruler.getRules(""); + for (i = 0, l = rules.length; i < l; i++) { + rules[i](state); + } + }; + Core.prototype.State = state_core; + var parser_core = Core; + var isSpace$a = utils.isSpace; + function getLine(state, line) { + var pos = state.bMarks[line] + state.tShift[line], max = state.eMarks[line]; + return state.src.slice(pos, max); + } + function escapedSplit(str) { + var result = [], pos = 0, max = str.length, ch, isEscaped = false, lastPos = 0, current = ""; + ch = str.charCodeAt(pos); + while (pos < max) { + if (ch === 124 /* | */) { + if (!isEscaped) { + // pipe separating cells, '|' + result.push(current + str.substring(lastPos, pos)); + current = ""; + lastPos = pos + 1; + } else { + // escaped pipe, '\|' + current += str.substring(lastPos, pos - 1); + lastPos = pos; + } + } + isEscaped = ch === 92 /* \ */; + pos++; + ch = str.charCodeAt(pos); + } + result.push(current + str.substring(lastPos)); + return result; + } + var table = function table(state, startLine, endLine, silent) { + var ch, lineText, pos, i, l, nextLine, columns, columnCount, token, aligns, t, tableLines, tbodyLines, oldParentType, terminate, terminatorRules, firstCh, secondCh; + // should have at least two lines + if (startLine + 2 > endLine) { + return false; + } + nextLine = startLine + 1; + if (state.sCount[nextLine] < state.blkIndent) { + return false; + } + // if it's indented more than 3 spaces, it should be a code block + if (state.sCount[nextLine] - state.blkIndent >= 4) { + return false; + } + // first character of the second line should be '|', '-', ':', + // and no other characters are allowed but spaces; + // basically, this is the equivalent of /^[-:|][-:|\s]*$/ regexp + pos = state.bMarks[nextLine] + state.tShift[nextLine]; + if (pos >= state.eMarks[nextLine]) { + return false; + } + firstCh = state.src.charCodeAt(pos++); + if (firstCh !== 124 /* | */ && firstCh !== 45 /* - */ && firstCh !== 58 /* : */) { + return false; + } + if (pos >= state.eMarks[nextLine]) { + return false; + } + secondCh = state.src.charCodeAt(pos++); + if (secondCh !== 124 /* | */ && secondCh !== 45 /* - */ && secondCh !== 58 /* : */ && !isSpace$a(secondCh)) { + return false; + } + // if first character is '-', then second character must not be a space + // (due to parsing ambiguity with list) + if (firstCh === 45 /* - */ && isSpace$a(secondCh)) { + return false; + } + while (pos < state.eMarks[nextLine]) { + ch = state.src.charCodeAt(pos); + if (ch !== 124 /* | */ && ch !== 45 /* - */ && ch !== 58 /* : */ && !isSpace$a(ch)) { + return false; + } + pos++; + } + lineText = getLine(state, startLine + 1); + columns = lineText.split("|"); + aligns = []; + for (i = 0; i < columns.length; i++) { + t = columns[i].trim(); + if (!t) { + // allow empty columns before and after table, but not in between columns; + // e.g. allow ` |---| `, disallow ` ---||--- ` + if (i === 0 || i === columns.length - 1) { + continue; + } else { + return false; + } + } + if (!/^:?-+:?$/.test(t)) { + return false; + } + if (t.charCodeAt(t.length - 1) === 58 /* : */) { + aligns.push(t.charCodeAt(0) === 58 /* : */ ? "center" : "right"); + } else if (t.charCodeAt(0) === 58 /* : */) { + aligns.push("left"); + } else { + aligns.push(""); + } + } + lineText = getLine(state, startLine).trim(); + if (lineText.indexOf("|") === -1) { + return false; + } + if (state.sCount[startLine] - state.blkIndent >= 4) { + return false; + } + columns = escapedSplit(lineText); + if (columns.length && columns[0] === "") columns.shift(); + if (columns.length && columns[columns.length - 1] === "") columns.pop(); + // header row will define an amount of columns in the entire table, + // and align row should be exactly the same (the rest of the rows can differ) + columnCount = columns.length; + if (columnCount === 0 || columnCount !== aligns.length) { + return false; + } + if (silent) { + return true; + } + oldParentType = state.parentType; + state.parentType = "table"; + // use 'blockquote' lists for termination because it's + // the most similar to tables + terminatorRules = state.md.block.ruler.getRules("blockquote"); + token = state.push("table_open", "table", 1); + token.map = tableLines = [ startLine, 0 ]; + token = state.push("thead_open", "thead", 1); + token.map = [ startLine, startLine + 1 ]; + token = state.push("tr_open", "tr", 1); + token.map = [ startLine, startLine + 1 ]; + for (i = 0; i < columns.length; i++) { + token = state.push("th_open", "th", 1); + if (aligns[i]) { + token.attrs = [ [ "style", "text-align:" + aligns[i] ] ]; + } + token = state.push("inline", "", 0); + token.content = columns[i].trim(); + token.children = []; + token = state.push("th_close", "th", -1); + } + token = state.push("tr_close", "tr", -1); + token = state.push("thead_close", "thead", -1); + for (nextLine = startLine + 2; nextLine < endLine; nextLine++) { + if (state.sCount[nextLine] < state.blkIndent) { + break; + } + terminate = false; + for (i = 0, l = terminatorRules.length; i < l; i++) { + if (terminatorRules[i](state, nextLine, endLine, true)) { + terminate = true; + break; + } + } + if (terminate) { + break; + } + lineText = getLine(state, nextLine).trim(); + if (!lineText) { + break; + } + if (state.sCount[nextLine] - state.blkIndent >= 4) { + break; + } + columns = escapedSplit(lineText); + if (columns.length && columns[0] === "") columns.shift(); + if (columns.length && columns[columns.length - 1] === "") columns.pop(); + if (nextLine === startLine + 2) { + token = state.push("tbody_open", "tbody", 1); + token.map = tbodyLines = [ startLine + 2, 0 ]; + } + token = state.push("tr_open", "tr", 1); + token.map = [ nextLine, nextLine + 1 ]; + for (i = 0; i < columnCount; i++) { + token = state.push("td_open", "td", 1); + if (aligns[i]) { + token.attrs = [ [ "style", "text-align:" + aligns[i] ] ]; + } + token = state.push("inline", "", 0); + token.content = columns[i] ? columns[i].trim() : ""; + token.children = []; + token = state.push("td_close", "td", -1); + } + token = state.push("tr_close", "tr", -1); + } + if (tbodyLines) { + token = state.push("tbody_close", "tbody", -1); + tbodyLines[1] = nextLine; + } + token = state.push("table_close", "table", -1); + tableLines[1] = nextLine; + state.parentType = oldParentType; + state.line = nextLine; + return true; + }; + // Code block (4 spaces padded) + var code = function code(state, startLine, endLine /*, silent*/) { + var nextLine, last, token; + if (state.sCount[startLine] - state.blkIndent < 4) { + return false; + } + last = nextLine = startLine + 1; + while (nextLine < endLine) { + if (state.isEmpty(nextLine)) { + nextLine++; + continue; + } + if (state.sCount[nextLine] - state.blkIndent >= 4) { + nextLine++; + last = nextLine; + continue; + } + break; + } + state.line = last; + token = state.push("code_block", "code", 0); + token.content = state.getLines(startLine, last, 4 + state.blkIndent, false) + "\n"; + token.map = [ startLine, state.line ]; + return true; + }; + // fences (``` lang, ~~~ lang) + var fence = function fence(state, startLine, endLine, silent) { + var marker, len, params, nextLine, mem, token, markup, haveEndMarker = false, pos = state.bMarks[startLine] + state.tShift[startLine], max = state.eMarks[startLine]; + // if it's indented more than 3 spaces, it should be a code block + if (state.sCount[startLine] - state.blkIndent >= 4) { + return false; + } + if (pos + 3 > max) { + return false; + } + marker = state.src.charCodeAt(pos); + if (marker !== 126 /* ~ */ && marker !== 96 /* ` */) { + return false; + } + // scan marker length + mem = pos; + pos = state.skipChars(pos, marker); + len = pos - mem; + if (len < 3) { + return false; + } + markup = state.src.slice(mem, pos); + params = state.src.slice(pos, max); + if (marker === 96 /* ` */) { + if (params.indexOf(String.fromCharCode(marker)) >= 0) { + return false; + } + } + // Since start is found, we can report success here in validation mode + if (silent) { + return true; + } + // search end of block + nextLine = startLine; + for (;;) { + nextLine++; + if (nextLine >= endLine) { + // unclosed block should be autoclosed by end of document. + // also block seems to be autoclosed by end of parent + break; + } + pos = mem = state.bMarks[nextLine] + state.tShift[nextLine]; + max = state.eMarks[nextLine]; + if (pos < max && state.sCount[nextLine] < state.blkIndent) { + // non-empty line with negative indent should stop the list: + // - ``` + // test + break; + } + if (state.src.charCodeAt(pos) !== marker) { + continue; + } + if (state.sCount[nextLine] - state.blkIndent >= 4) { + // closing fence should be indented less than 4 spaces + continue; + } + pos = state.skipChars(pos, marker); + // closing code fence must be at least as long as the opening one + if (pos - mem < len) { + continue; + } + // make sure tail has spaces only + pos = state.skipSpaces(pos); + if (pos < max) { + continue; + } + haveEndMarker = true; + // found! + break; + } + // If a fence has heading spaces, they should be removed from its inner block + len = state.sCount[startLine]; + state.line = nextLine + (haveEndMarker ? 1 : 0); + token = state.push("fence", "code", 0); + token.info = params; + token.content = state.getLines(startLine + 1, nextLine, len, true); + token.markup = markup; + token.map = [ startLine, state.line ]; + return true; + }; + var isSpace$9 = utils.isSpace; + var blockquote = function blockquote(state, startLine, endLine, silent) { + var adjustTab, ch, i, initial, l, lastLineEmpty, lines, nextLine, offset, oldBMarks, oldBSCount, oldIndent, oldParentType, oldSCount, oldTShift, spaceAfterMarker, terminate, terminatorRules, token, isOutdented, oldLineMax = state.lineMax, pos = state.bMarks[startLine] + state.tShift[startLine], max = state.eMarks[startLine]; + // if it's indented more than 3 spaces, it should be a code block + if (state.sCount[startLine] - state.blkIndent >= 4) { + return false; + } + // check the block quote marker + if (state.src.charCodeAt(pos++) !== 62 /* > */) { + return false; + } + // we know that it's going to be a valid blockquote, + // so no point trying to find the end of it in silent mode + if (silent) { + return true; + } + // set offset past spaces and ">" + initial = offset = state.sCount[startLine] + 1; + // skip one optional space after '>' + if (state.src.charCodeAt(pos) === 32 /* space */) { + // ' > test ' + // ^ -- position start of line here: + pos++; + initial++; + offset++; + adjustTab = false; + spaceAfterMarker = true; + } else if (state.src.charCodeAt(pos) === 9 /* tab */) { + spaceAfterMarker = true; + if ((state.bsCount[startLine] + offset) % 4 === 3) { + // ' >\t test ' + // ^ -- position start of line here (tab has width===1) + pos++; + initial++; + offset++; + adjustTab = false; + } else { + // ' >\t test ' + // ^ -- position start of line here + shift bsCount slightly + // to make extra space appear + adjustTab = true; + } + } else { + spaceAfterMarker = false; + } + oldBMarks = [ state.bMarks[startLine] ]; + state.bMarks[startLine] = pos; + while (pos < max) { + ch = state.src.charCodeAt(pos); + if (isSpace$9(ch)) { + if (ch === 9) { + offset += 4 - (offset + state.bsCount[startLine] + (adjustTab ? 1 : 0)) % 4; + } else { + offset++; + } + } else { + break; + } + pos++; + } + oldBSCount = [ state.bsCount[startLine] ]; + state.bsCount[startLine] = state.sCount[startLine] + 1 + (spaceAfterMarker ? 1 : 0); + lastLineEmpty = pos >= max; + oldSCount = [ state.sCount[startLine] ]; + state.sCount[startLine] = offset - initial; + oldTShift = [ state.tShift[startLine] ]; + state.tShift[startLine] = pos - state.bMarks[startLine]; + terminatorRules = state.md.block.ruler.getRules("blockquote"); + oldParentType = state.parentType; + state.parentType = "blockquote"; + // Search the end of the block + + // Block ends with either: + // 1. an empty line outside: + // ``` + // > test + + // ``` + // 2. an empty line inside: + // ``` + // > + // test + // ``` + // 3. another tag: + // ``` + // > test + // - - - + // ``` + for (nextLine = startLine + 1; nextLine < endLine; nextLine++) { + // check if it's outdented, i.e. it's inside list item and indented + // less than said list item: + // ``` + // 1. anything + // > current blockquote + // 2. checking this line + // ``` + isOutdented = state.sCount[nextLine] < state.blkIndent; + pos = state.bMarks[nextLine] + state.tShift[nextLine]; + max = state.eMarks[nextLine]; + if (pos >= max) { + // Case 1: line is not inside the blockquote, and this line is empty. + break; + } + if (state.src.charCodeAt(pos++) === 62 /* > */ && !isOutdented) { + // This line is inside the blockquote. + // set offset past spaces and ">" + initial = offset = state.sCount[nextLine] + 1; + // skip one optional space after '>' + if (state.src.charCodeAt(pos) === 32 /* space */) { + // ' > test ' + // ^ -- position start of line here: + pos++; + initial++; + offset++; + adjustTab = false; + spaceAfterMarker = true; + } else if (state.src.charCodeAt(pos) === 9 /* tab */) { + spaceAfterMarker = true; + if ((state.bsCount[nextLine] + offset) % 4 === 3) { + // ' >\t test ' + // ^ -- position start of line here (tab has width===1) + pos++; + initial++; + offset++; + adjustTab = false; + } else { + // ' >\t test ' + // ^ -- position start of line here + shift bsCount slightly + // to make extra space appear + adjustTab = true; + } + } else { + spaceAfterMarker = false; + } + oldBMarks.push(state.bMarks[nextLine]); + state.bMarks[nextLine] = pos; + while (pos < max) { + ch = state.src.charCodeAt(pos); + if (isSpace$9(ch)) { + if (ch === 9) { + offset += 4 - (offset + state.bsCount[nextLine] + (adjustTab ? 1 : 0)) % 4; + } else { + offset++; + } + } else { + break; + } + pos++; + } + lastLineEmpty = pos >= max; + oldBSCount.push(state.bsCount[nextLine]); + state.bsCount[nextLine] = state.sCount[nextLine] + 1 + (spaceAfterMarker ? 1 : 0); + oldSCount.push(state.sCount[nextLine]); + state.sCount[nextLine] = offset - initial; + oldTShift.push(state.tShift[nextLine]); + state.tShift[nextLine] = pos - state.bMarks[nextLine]; + continue; + } + // Case 2: line is not inside the blockquote, and the last line was empty. + if (lastLineEmpty) { + break; + } + // Case 3: another tag found. + terminate = false; + for (i = 0, l = terminatorRules.length; i < l; i++) { + if (terminatorRules[i](state, nextLine, endLine, true)) { + terminate = true; + break; + } + } + if (terminate) { + // Quirk to enforce "hard termination mode" for paragraphs; + // normally if you call `tokenize(state, startLine, nextLine)`, + // paragraphs will look below nextLine for paragraph continuation, + // but if blockquote is terminated by another tag, they shouldn't + state.lineMax = nextLine; + if (state.blkIndent !== 0) { + // state.blkIndent was non-zero, we now set it to zero, + // so we need to re-calculate all offsets to appear as + // if indent wasn't changed + oldBMarks.push(state.bMarks[nextLine]); + oldBSCount.push(state.bsCount[nextLine]); + oldTShift.push(state.tShift[nextLine]); + oldSCount.push(state.sCount[nextLine]); + state.sCount[nextLine] -= state.blkIndent; + } + break; + } + oldBMarks.push(state.bMarks[nextLine]); + oldBSCount.push(state.bsCount[nextLine]); + oldTShift.push(state.tShift[nextLine]); + oldSCount.push(state.sCount[nextLine]); + // A negative indentation means that this is a paragraph continuation + + state.sCount[nextLine] = -1; + } + oldIndent = state.blkIndent; + state.blkIndent = 0; + token = state.push("blockquote_open", "blockquote", 1); + token.markup = ">"; + token.map = lines = [ startLine, 0 ]; + state.md.block.tokenize(state, startLine, nextLine); + token = state.push("blockquote_close", "blockquote", -1); + token.markup = ">"; + state.lineMax = oldLineMax; + state.parentType = oldParentType; + lines[1] = state.line; + // Restore original tShift; this might not be necessary since the parser + // has already been here, but just to make sure we can do that. + for (i = 0; i < oldTShift.length; i++) { + state.bMarks[i + startLine] = oldBMarks[i]; + state.tShift[i + startLine] = oldTShift[i]; + state.sCount[i + startLine] = oldSCount[i]; + state.bsCount[i + startLine] = oldBSCount[i]; + } + state.blkIndent = oldIndent; + return true; + }; + var isSpace$8 = utils.isSpace; + var hr = function hr(state, startLine, endLine, silent) { + var marker, cnt, ch, token, pos = state.bMarks[startLine] + state.tShift[startLine], max = state.eMarks[startLine]; + // if it's indented more than 3 spaces, it should be a code block + if (state.sCount[startLine] - state.blkIndent >= 4) { + return false; + } + marker = state.src.charCodeAt(pos++); + // Check hr marker + if (marker !== 42 /* * */ && marker !== 45 /* - */ && marker !== 95 /* _ */) { + return false; + } + // markers can be mixed with spaces, but there should be at least 3 of them + cnt = 1; + while (pos < max) { + ch = state.src.charCodeAt(pos++); + if (ch !== marker && !isSpace$8(ch)) { + return false; + } + if (ch === marker) { + cnt++; + } + } + if (cnt < 3) { + return false; + } + if (silent) { + return true; + } + state.line = startLine + 1; + token = state.push("hr", "hr", 0); + token.map = [ startLine, state.line ]; + token.markup = Array(cnt + 1).join(String.fromCharCode(marker)); + return true; + }; + var isSpace$7 = utils.isSpace; + // Search `[-+*][\n ]`, returns next pos after marker on success + // or -1 on fail. + function skipBulletListMarker(state, startLine) { + var marker, pos, max, ch; + pos = state.bMarks[startLine] + state.tShift[startLine]; + max = state.eMarks[startLine]; + marker = state.src.charCodeAt(pos++); + // Check bullet + if (marker !== 42 /* * */ && marker !== 45 /* - */ && marker !== 43 /* + */) { + return -1; + } + if (pos < max) { + ch = state.src.charCodeAt(pos); + if (!isSpace$7(ch)) { + // " -test " - is not a list item + return -1; + } + } + return pos; + } + // Search `\d+[.)][\n ]`, returns next pos after marker on success + // or -1 on fail. + function skipOrderedListMarker(state, startLine) { + var ch, start = state.bMarks[startLine] + state.tShift[startLine], pos = start, max = state.eMarks[startLine]; + // List marker should have at least 2 chars (digit + dot) + if (pos + 1 >= max) { + return -1; + } + ch = state.src.charCodeAt(pos++); + if (ch < 48 /* 0 */ || ch > 57 /* 9 */) { + return -1; + } + for (;;) { + // EOL -> fail + if (pos >= max) { + return -1; + } + ch = state.src.charCodeAt(pos++); + if (ch >= 48 /* 0 */ && ch <= 57 /* 9 */) { + // List marker should have no more than 9 digits + // (prevents integer overflow in browsers) + if (pos - start >= 10) { + return -1; + } + continue; + } + // found valid marker + if (ch === 41 /* ) */ || ch === 46 /* . */) { + break; + } + return -1; + } + if (pos < max) { + ch = state.src.charCodeAt(pos); + if (!isSpace$7(ch)) { + // " 1.test " - is not a list item + return -1; + } + } + return pos; + } + function markTightParagraphs(state, idx) { + var i, l, level = state.level + 2; + for (i = idx + 2, l = state.tokens.length - 2; i < l; i++) { + if (state.tokens[i].level === level && state.tokens[i].type === "paragraph_open") { + state.tokens[i + 2].hidden = true; + state.tokens[i].hidden = true; + i += 2; + } + } + } + var list = function list(state, startLine, endLine, silent) { + var ch, contentStart, i, indent, indentAfterMarker, initial, isOrdered, itemLines, l, listLines, listTokIdx, markerCharCode, markerValue, max, nextLine, offset, oldListIndent, oldParentType, oldSCount, oldTShift, oldTight, pos, posAfterMarker, prevEmptyEnd, start, terminate, terminatorRules, token, isTerminatingParagraph = false, tight = true; + // if it's indented more than 3 spaces, it should be a code block + if (state.sCount[startLine] - state.blkIndent >= 4) { + return false; + } + // Special case: + // - item 1 + // - item 2 + // - item 3 + // - item 4 + // - this one is a paragraph continuation + if (state.listIndent >= 0 && state.sCount[startLine] - state.listIndent >= 4 && state.sCount[startLine] < state.blkIndent) { + return false; + } + // limit conditions when list can interrupt + // a paragraph (validation mode only) + if (silent && state.parentType === "paragraph") { + // Next list item should still terminate previous list item; + // This code can fail if plugins use blkIndent as well as lists, + // but I hope the spec gets fixed long before that happens. + if (state.sCount[startLine] >= state.blkIndent) { + isTerminatingParagraph = true; + } + } + // Detect list type and position after marker + if ((posAfterMarker = skipOrderedListMarker(state, startLine)) >= 0) { + isOrdered = true; + start = state.bMarks[startLine] + state.tShift[startLine]; + markerValue = Number(state.src.slice(start, posAfterMarker - 1)); + // If we're starting a new ordered list right after + // a paragraph, it should start with 1. + if (isTerminatingParagraph && markerValue !== 1) return false; + } else if ((posAfterMarker = skipBulletListMarker(state, startLine)) >= 0) { + isOrdered = false; + } else { + return false; + } + // If we're starting a new unordered list right after + // a paragraph, first line should not be empty. + if (isTerminatingParagraph) { + if (state.skipSpaces(posAfterMarker) >= state.eMarks[startLine]) return false; + } + // We should terminate list on style change. Remember first one to compare. + markerCharCode = state.src.charCodeAt(posAfterMarker - 1); + // For validation mode we can terminate immediately + if (silent) { + return true; + } + // Start list + listTokIdx = state.tokens.length; + if (isOrdered) { + token = state.push("ordered_list_open", "ol", 1); + if (markerValue !== 1) { + token.attrs = [ [ "start", markerValue ] ]; + } + } else { + token = state.push("bullet_list_open", "ul", 1); + } + token.map = listLines = [ startLine, 0 ]; + token.markup = String.fromCharCode(markerCharCode); + + // Iterate list items + + nextLine = startLine; + prevEmptyEnd = false; + terminatorRules = state.md.block.ruler.getRules("list"); + oldParentType = state.parentType; + state.parentType = "list"; + while (nextLine < endLine) { + pos = posAfterMarker; + max = state.eMarks[nextLine]; + initial = offset = state.sCount[nextLine] + posAfterMarker - (state.bMarks[startLine] + state.tShift[startLine]); + while (pos < max) { + ch = state.src.charCodeAt(pos); + if (ch === 9) { + offset += 4 - (offset + state.bsCount[nextLine]) % 4; + } else if (ch === 32) { + offset++; + } else { + break; + } + pos++; + } + contentStart = pos; + if (contentStart >= max) { + // trimming space in "- \n 3" case, indent is 1 here + indentAfterMarker = 1; + } else { + indentAfterMarker = offset - initial; + } + // If we have more than 4 spaces, the indent is 1 + // (the rest is just indented code block) + if (indentAfterMarker > 4) { + indentAfterMarker = 1; + } + // " - test" + // ^^^^^ - calculating total length of this thing + indent = initial + indentAfterMarker; + // Run subparser & write tokens + token = state.push("list_item_open", "li", 1); + token.markup = String.fromCharCode(markerCharCode); + token.map = itemLines = [ startLine, 0 ]; + if (isOrdered) { + token.info = state.src.slice(start, posAfterMarker - 1); + } + // change current state, then restore it after parser subcall + oldTight = state.tight; + oldTShift = state.tShift[startLine]; + oldSCount = state.sCount[startLine]; + // - example list + // ^ listIndent position will be here + // ^ blkIndent position will be here + + oldListIndent = state.listIndent; + state.listIndent = state.blkIndent; + state.blkIndent = indent; + state.tight = true; + state.tShift[startLine] = contentStart - state.bMarks[startLine]; + state.sCount[startLine] = offset; + if (contentStart >= max && state.isEmpty(startLine + 1)) { + // workaround for this case + // (list item is empty, list terminates before "foo"): + // ~~~~~~~~ + // - + // foo + // ~~~~~~~~ + state.line = Math.min(state.line + 2, endLine); + } else { + state.md.block.tokenize(state, startLine, endLine, true); + } + // If any of list item is tight, mark list as tight + if (!state.tight || prevEmptyEnd) { + tight = false; + } + // Item become loose if finish with empty line, + // but we should filter last element, because it means list finish + prevEmptyEnd = state.line - startLine > 1 && state.isEmpty(state.line - 1); + state.blkIndent = state.listIndent; + state.listIndent = oldListIndent; + state.tShift[startLine] = oldTShift; + state.sCount[startLine] = oldSCount; + state.tight = oldTight; + token = state.push("list_item_close", "li", -1); + token.markup = String.fromCharCode(markerCharCode); + nextLine = startLine = state.line; + itemLines[1] = nextLine; + contentStart = state.bMarks[startLine]; + if (nextLine >= endLine) { + break; + } + + // Try to check if list is terminated or continued. + + if (state.sCount[nextLine] < state.blkIndent) { + break; + } + // if it's indented more than 3 spaces, it should be a code block + if (state.sCount[startLine] - state.blkIndent >= 4) { + break; + } + // fail if terminating block found + terminate = false; + for (i = 0, l = terminatorRules.length; i < l; i++) { + if (terminatorRules[i](state, nextLine, endLine, true)) { + terminate = true; + break; + } + } + if (terminate) { + break; + } + // fail if list has another type + if (isOrdered) { + posAfterMarker = skipOrderedListMarker(state, nextLine); + if (posAfterMarker < 0) { + break; + } + start = state.bMarks[nextLine] + state.tShift[nextLine]; + } else { + posAfterMarker = skipBulletListMarker(state, nextLine); + if (posAfterMarker < 0) { + break; + } + } + if (markerCharCode !== state.src.charCodeAt(posAfterMarker - 1)) { + break; + } + } + // Finalize list + if (isOrdered) { + token = state.push("ordered_list_close", "ol", -1); + } else { + token = state.push("bullet_list_close", "ul", -1); + } + token.markup = String.fromCharCode(markerCharCode); + listLines[1] = nextLine; + state.line = nextLine; + state.parentType = oldParentType; + // mark paragraphs tight if needed + if (tight) { + markTightParagraphs(state, listTokIdx); + } + return true; + }; + var normalizeReference$2 = utils.normalizeReference; + var isSpace$6 = utils.isSpace; + var reference = function reference(state, startLine, _endLine, silent) { + var ch, destEndPos, destEndLineNo, endLine, href, i, l, label, labelEnd, oldParentType, res, start, str, terminate, terminatorRules, title, lines = 0, pos = state.bMarks[startLine] + state.tShift[startLine], max = state.eMarks[startLine], nextLine = startLine + 1; + // if it's indented more than 3 spaces, it should be a code block + if (state.sCount[startLine] - state.blkIndent >= 4) { + return false; + } + if (state.src.charCodeAt(pos) !== 91 /* [ */) { + return false; + } + // Simple check to quickly interrupt scan on [link](url) at the start of line. + // Can be useful on practice: https://github.com/markdown-it/markdown-it/issues/54 + while (++pos < max) { + if (state.src.charCodeAt(pos) === 93 /* ] */ && state.src.charCodeAt(pos - 1) !== 92 /* \ */) { + if (pos + 1 === max) { + return false; + } + if (state.src.charCodeAt(pos + 1) !== 58 /* : */) { + return false; + } + break; + } + } + endLine = state.lineMax; + // jump line-by-line until empty one or EOF + terminatorRules = state.md.block.ruler.getRules("reference"); + oldParentType = state.parentType; + state.parentType = "reference"; + for (;nextLine < endLine && !state.isEmpty(nextLine); nextLine++) { + // this would be a code block normally, but after paragraph + // it's considered a lazy continuation regardless of what's there + if (state.sCount[nextLine] - state.blkIndent > 3) { + continue; + } + // quirk for blockquotes, this line should already be checked by that rule + if (state.sCount[nextLine] < 0) { + continue; + } + // Some tags can terminate paragraph without empty line. + terminate = false; + for (i = 0, l = terminatorRules.length; i < l; i++) { + if (terminatorRules[i](state, nextLine, endLine, true)) { + terminate = true; + break; + } + } + if (terminate) { + break; + } + } + str = state.getLines(startLine, nextLine, state.blkIndent, false).trim(); + max = str.length; + for (pos = 1; pos < max; pos++) { + ch = str.charCodeAt(pos); + if (ch === 91 /* [ */) { + return false; + } else if (ch === 93 /* ] */) { + labelEnd = pos; + break; + } else if (ch === 10 /* \n */) { + lines++; + } else if (ch === 92 /* \ */) { + pos++; + if (pos < max && str.charCodeAt(pos) === 10) { + lines++; + } + } + } + if (labelEnd < 0 || str.charCodeAt(labelEnd + 1) !== 58 /* : */) { + return false; + } + // [label]: destination 'title' + // ^^^ skip optional whitespace here + for (pos = labelEnd + 2; pos < max; pos++) { + ch = str.charCodeAt(pos); + if (ch === 10) { + lines++; + } else if (isSpace$6(ch)) ; else { + break; + } + } + // [label]: destination 'title' + // ^^^^^^^^^^^ parse this + res = state.md.helpers.parseLinkDestination(str, pos, max); + if (!res.ok) { + return false; + } + href = state.md.normalizeLink(res.str); + if (!state.md.validateLink(href)) { + return false; + } + pos = res.pos; + lines += res.lines; + // save cursor state, we could require to rollback later + destEndPos = pos; + destEndLineNo = lines; + // [label]: destination 'title' + // ^^^ skipping those spaces + start = pos; + for (;pos < max; pos++) { + ch = str.charCodeAt(pos); + if (ch === 10) { + lines++; + } else if (isSpace$6(ch)) ; else { + break; + } + } + // [label]: destination 'title' + // ^^^^^^^ parse this + res = state.md.helpers.parseLinkTitle(str, pos, max); + if (pos < max && start !== pos && res.ok) { + title = res.str; + pos = res.pos; + lines += res.lines; + } else { + title = ""; + pos = destEndPos; + lines = destEndLineNo; + } + // skip trailing spaces until the rest of the line + while (pos < max) { + ch = str.charCodeAt(pos); + if (!isSpace$6(ch)) { + break; + } + pos++; + } + if (pos < max && str.charCodeAt(pos) !== 10) { + if (title) { + // garbage at the end of the line after title, + // but it could still be a valid reference if we roll back + title = ""; + pos = destEndPos; + lines = destEndLineNo; + while (pos < max) { + ch = str.charCodeAt(pos); + if (!isSpace$6(ch)) { + break; + } + pos++; + } + } + } + if (pos < max && str.charCodeAt(pos) !== 10) { + // garbage at the end of the line + return false; + } + label = normalizeReference$2(str.slice(1, labelEnd)); + if (!label) { + // CommonMark 0.20 disallows empty labels + return false; + } + // Reference can not terminate anything. This check is for safety only. + /*istanbul ignore if*/ if (silent) { + return true; + } + if (typeof state.env.references === "undefined") { + state.env.references = {}; + } + if (typeof state.env.references[label] === "undefined") { + state.env.references[label] = { + title: title, + href: href + }; + } + state.parentType = oldParentType; + state.line = startLine + lines + 1; + return true; + }; + // List of valid html blocks names, accorting to commonmark spec + var html_blocks = [ "address", "article", "aside", "base", "basefont", "blockquote", "body", "caption", "center", "col", "colgroup", "dd", "details", "dialog", "dir", "div", "dl", "dt", "fieldset", "figcaption", "figure", "footer", "form", "frame", "frameset", "h1", "h2", "h3", "h4", "h5", "h6", "head", "header", "hr", "html", "iframe", "legend", "li", "link", "main", "menu", "menuitem", "nav", "noframes", "ol", "optgroup", "option", "p", "param", "section", "source", "summary", "table", "tbody", "td", "tfoot", "th", "thead", "title", "tr", "track", "ul" ]; + // Regexps to match html elements + var attr_name = "[a-zA-Z_:][a-zA-Z0-9:._-]*"; + var unquoted = "[^\"'=<>`\\x00-\\x20]+"; + var single_quoted = "'[^']*'"; + var double_quoted = '"[^"]*"'; + var attr_value = "(?:" + unquoted + "|" + single_quoted + "|" + double_quoted + ")"; + var attribute = "(?:\\s+" + attr_name + "(?:\\s*=\\s*" + attr_value + ")?)"; + var open_tag = "<[A-Za-z][A-Za-z0-9\\-]*" + attribute + "*\\s*\\/?>"; + var close_tag = "<\\/[A-Za-z][A-Za-z0-9\\-]*\\s*>"; + var comment = "\x3c!----\x3e|\x3c!--(?:-?[^>-])(?:-?[^-])*--\x3e"; + var processing = "<[?][\\s\\S]*?[?]>"; + var declaration = "]*>"; + var cdata = ""; + var HTML_TAG_RE$1 = new RegExp("^(?:" + open_tag + "|" + close_tag + "|" + comment + "|" + processing + "|" + declaration + "|" + cdata + ")"); + var HTML_OPEN_CLOSE_TAG_RE$1 = new RegExp("^(?:" + open_tag + "|" + close_tag + ")"); + var HTML_TAG_RE_1 = HTML_TAG_RE$1; + var HTML_OPEN_CLOSE_TAG_RE_1 = HTML_OPEN_CLOSE_TAG_RE$1; + var html_re = { + HTML_TAG_RE: HTML_TAG_RE_1, + HTML_OPEN_CLOSE_TAG_RE: HTML_OPEN_CLOSE_TAG_RE_1 + }; + var HTML_OPEN_CLOSE_TAG_RE = html_re.HTML_OPEN_CLOSE_TAG_RE; + // An array of opening and corresponding closing sequences for html tags, + // last argument defines whether it can terminate a paragraph or not + + var HTML_SEQUENCES = [ [ /^<(script|pre|style|textarea)(?=(\s|>|$))/i, /<\/(script|pre|style|textarea)>/i, true ], [ /^/, true ], [ /^<\?/, /\?>/, true ], [ /^/, true ], [ /^/, true ], [ new RegExp("^|$))", "i"), /^$/, true ], [ new RegExp(HTML_OPEN_CLOSE_TAG_RE.source + "\\s*$"), /^$/, false ] ]; + var html_block = function html_block(state, startLine, endLine, silent) { + var i, nextLine, token, lineText, pos = state.bMarks[startLine] + state.tShift[startLine], max = state.eMarks[startLine]; + // if it's indented more than 3 spaces, it should be a code block + if (state.sCount[startLine] - state.blkIndent >= 4) { + return false; + } + if (!state.md.options.html) { + return false; + } + if (state.src.charCodeAt(pos) !== 60 /* < */) { + return false; + } + lineText = state.src.slice(pos, max); + for (i = 0; i < HTML_SEQUENCES.length; i++) { + if (HTML_SEQUENCES[i][0].test(lineText)) { + break; + } + } + if (i === HTML_SEQUENCES.length) { + return false; + } + if (silent) { + // true if this sequence can be a terminator, false otherwise + return HTML_SEQUENCES[i][2]; + } + nextLine = startLine + 1; + // If we are here - we detected HTML block. + // Let's roll down till block end. + if (!HTML_SEQUENCES[i][1].test(lineText)) { + for (;nextLine < endLine; nextLine++) { + if (state.sCount[nextLine] < state.blkIndent) { + break; + } + pos = state.bMarks[nextLine] + state.tShift[nextLine]; + max = state.eMarks[nextLine]; + lineText = state.src.slice(pos, max); + if (HTML_SEQUENCES[i][1].test(lineText)) { + if (lineText.length !== 0) { + nextLine++; + } + break; + } + } + } + state.line = nextLine; + token = state.push("html_block", "", 0); + token.map = [ startLine, nextLine ]; + token.content = state.getLines(startLine, nextLine, state.blkIndent, true); + return true; + }; + var isSpace$5 = utils.isSpace; + var heading = function heading(state, startLine, endLine, silent) { + var ch, level, tmp, token, pos = state.bMarks[startLine] + state.tShift[startLine], max = state.eMarks[startLine]; + // if it's indented more than 3 spaces, it should be a code block + if (state.sCount[startLine] - state.blkIndent >= 4) { + return false; + } + ch = state.src.charCodeAt(pos); + if (ch !== 35 /* # */ || pos >= max) { + return false; + } + // count heading level + level = 1; + ch = state.src.charCodeAt(++pos); + while (ch === 35 /* # */ && pos < max && level <= 6) { + level++; + ch = state.src.charCodeAt(++pos); + } + if (level > 6 || pos < max && !isSpace$5(ch)) { + return false; + } + if (silent) { + return true; + } + // Let's cut tails like ' ### ' from the end of string + max = state.skipSpacesBack(max, pos); + tmp = state.skipCharsBack(max, 35, pos); + // # + if (tmp > pos && isSpace$5(state.src.charCodeAt(tmp - 1))) { + max = tmp; + } + state.line = startLine + 1; + token = state.push("heading_open", "h" + String(level), 1); + token.markup = "########".slice(0, level); + token.map = [ startLine, state.line ]; + token = state.push("inline", "", 0); + token.content = state.src.slice(pos, max).trim(); + token.map = [ startLine, state.line ]; + token.children = []; + token = state.push("heading_close", "h" + String(level), -1); + token.markup = "########".slice(0, level); + return true; + }; + // lheading (---, ===) + var lheading = function lheading(state, startLine, endLine /*, silent*/) { + var content, terminate, i, l, token, pos, max, level, marker, nextLine = startLine + 1, oldParentType, terminatorRules = state.md.block.ruler.getRules("paragraph"); + // if it's indented more than 3 spaces, it should be a code block + if (state.sCount[startLine] - state.blkIndent >= 4) { + return false; + } + oldParentType = state.parentType; + state.parentType = "paragraph"; + // use paragraph to match terminatorRules + // jump line-by-line until empty one or EOF + for (;nextLine < endLine && !state.isEmpty(nextLine); nextLine++) { + // this would be a code block normally, but after paragraph + // it's considered a lazy continuation regardless of what's there + if (state.sCount[nextLine] - state.blkIndent > 3) { + continue; + } + + // Check for underline in setext header + + if (state.sCount[nextLine] >= state.blkIndent) { + pos = state.bMarks[nextLine] + state.tShift[nextLine]; + max = state.eMarks[nextLine]; + if (pos < max) { + marker = state.src.charCodeAt(pos); + if (marker === 45 /* - */ || marker === 61 /* = */) { + pos = state.skipChars(pos, marker); + pos = state.skipSpaces(pos); + if (pos >= max) { + level = marker === 61 /* = */ ? 1 : 2; + break; + } + } + } + } + // quirk for blockquotes, this line should already be checked by that rule + if (state.sCount[nextLine] < 0) { + continue; + } + // Some tags can terminate paragraph without empty line. + terminate = false; + for (i = 0, l = terminatorRules.length; i < l; i++) { + if (terminatorRules[i](state, nextLine, endLine, true)) { + terminate = true; + break; + } + } + if (terminate) { + break; + } + } + if (!level) { + // Didn't find valid underline + return false; + } + content = state.getLines(startLine, nextLine, state.blkIndent, false).trim(); + state.line = nextLine + 1; + token = state.push("heading_open", "h" + String(level), 1); + token.markup = String.fromCharCode(marker); + token.map = [ startLine, state.line ]; + token = state.push("inline", "", 0); + token.content = content; + token.map = [ startLine, state.line - 1 ]; + token.children = []; + token = state.push("heading_close", "h" + String(level), -1); + token.markup = String.fromCharCode(marker); + state.parentType = oldParentType; + return true; + }; + // Paragraph + var paragraph = function paragraph(state, startLine /*, endLine*/) { + var content, terminate, i, l, token, oldParentType, nextLine = startLine + 1, terminatorRules = state.md.block.ruler.getRules("paragraph"), endLine = state.lineMax; + oldParentType = state.parentType; + state.parentType = "paragraph"; + // jump line-by-line until empty one or EOF + for (;nextLine < endLine && !state.isEmpty(nextLine); nextLine++) { + // this would be a code block normally, but after paragraph + // it's considered a lazy continuation regardless of what's there + if (state.sCount[nextLine] - state.blkIndent > 3) { + continue; + } + // quirk for blockquotes, this line should already be checked by that rule + if (state.sCount[nextLine] < 0) { + continue; + } + // Some tags can terminate paragraph without empty line. + terminate = false; + for (i = 0, l = terminatorRules.length; i < l; i++) { + if (terminatorRules[i](state, nextLine, endLine, true)) { + terminate = true; + break; + } + } + if (terminate) { + break; + } + } + content = state.getLines(startLine, nextLine, state.blkIndent, false).trim(); + state.line = nextLine; + token = state.push("paragraph_open", "p", 1); + token.map = [ startLine, state.line ]; + token = state.push("inline", "", 0); + token.content = content; + token.map = [ startLine, state.line ]; + token.children = []; + token = state.push("paragraph_close", "p", -1); + state.parentType = oldParentType; + return true; + }; + var isSpace$4 = utils.isSpace; + function StateBlock(src, md, env, tokens) { + var ch, s, start, pos, len, indent, offset, indent_found; + this.src = src; + // link to parser instance + this.md = md; + this.env = env; + + // Internal state vartiables + + this.tokens = tokens; + this.bMarks = []; + // line begin offsets for fast jumps + this.eMarks = []; + // line end offsets for fast jumps + this.tShift = []; + // offsets of the first non-space characters (tabs not expanded) + this.sCount = []; + // indents for each line (tabs expanded) + // An amount of virtual spaces (tabs expanded) between beginning + // of each line (bMarks) and real beginning of that line. + + // It exists only as a hack because blockquotes override bMarks + // losing information in the process. + + // It's used only when expanding tabs, you can think about it as + // an initial tab length, e.g. bsCount=21 applied to string `\t123` + // means first tab should be expanded to 4-21%4 === 3 spaces. + + this.bsCount = []; + // block parser variables + this.blkIndent = 0; + // required block content indent (for example, if we are + // inside a list, it would be positioned after list marker) + this.line = 0; + // line index in src + this.lineMax = 0; + // lines count + this.tight = false; + // loose/tight mode for lists + this.ddIndent = -1; + // indent of the current dd block (-1 if there isn't any) + this.listIndent = -1; + // indent of the current list block (-1 if there isn't any) + // can be 'blockquote', 'list', 'root', 'paragraph' or 'reference' + // used in lists to determine if they interrupt a paragraph + this.parentType = "root"; + this.level = 0; + // renderer + this.result = ""; + // Create caches + // Generate markers. + s = this.src; + indent_found = false; + for (start = pos = indent = offset = 0, len = s.length; pos < len; pos++) { + ch = s.charCodeAt(pos); + if (!indent_found) { + if (isSpace$4(ch)) { + indent++; + if (ch === 9) { + offset += 4 - offset % 4; + } else { + offset++; + } + continue; + } else { + indent_found = true; + } + } + if (ch === 10 || pos === len - 1) { + if (ch !== 10) { + pos++; + } + this.bMarks.push(start); + this.eMarks.push(pos); + this.tShift.push(indent); + this.sCount.push(offset); + this.bsCount.push(0); + indent_found = false; + indent = 0; + offset = 0; + start = pos + 1; + } + } + // Push fake entry to simplify cache bounds checks + this.bMarks.push(s.length); + this.eMarks.push(s.length); + this.tShift.push(0); + this.sCount.push(0); + this.bsCount.push(0); + this.lineMax = this.bMarks.length - 1; + // don't count last fake line + } + // Push new token to "stream". + + StateBlock.prototype.push = function(type, tag, nesting) { + var token$1 = new token(type, tag, nesting); + token$1.block = true; + if (nesting < 0) this.level--; + // closing tag + token$1.level = this.level; + if (nesting > 0) this.level++; + // opening tag + this.tokens.push(token$1); + return token$1; + }; + StateBlock.prototype.isEmpty = function isEmpty(line) { + return this.bMarks[line] + this.tShift[line] >= this.eMarks[line]; + }; + StateBlock.prototype.skipEmptyLines = function skipEmptyLines(from) { + for (var max = this.lineMax; from < max; from++) { + if (this.bMarks[from] + this.tShift[from] < this.eMarks[from]) { + break; + } + } + return from; + }; + // Skip spaces from given position. + StateBlock.prototype.skipSpaces = function skipSpaces(pos) { + var ch; + for (var max = this.src.length; pos < max; pos++) { + ch = this.src.charCodeAt(pos); + if (!isSpace$4(ch)) { + break; + } + } + return pos; + }; + // Skip spaces from given position in reverse. + StateBlock.prototype.skipSpacesBack = function skipSpacesBack(pos, min) { + if (pos <= min) { + return pos; + } + while (pos > min) { + if (!isSpace$4(this.src.charCodeAt(--pos))) { + return pos + 1; + } + } + return pos; + }; + // Skip char codes from given position + StateBlock.prototype.skipChars = function skipChars(pos, code) { + for (var max = this.src.length; pos < max; pos++) { + if (this.src.charCodeAt(pos) !== code) { + break; + } + } + return pos; + }; + // Skip char codes reverse from given position - 1 + StateBlock.prototype.skipCharsBack = function skipCharsBack(pos, code, min) { + if (pos <= min) { + return pos; + } + while (pos > min) { + if (code !== this.src.charCodeAt(--pos)) { + return pos + 1; + } + } + return pos; + }; + // cut lines range from source. + StateBlock.prototype.getLines = function getLines(begin, end, indent, keepLastLF) { + var i, lineIndent, ch, first, last, queue, lineStart, line = begin; + if (begin >= end) { + return ""; + } + queue = new Array(end - begin); + for (i = 0; line < end; line++, i++) { + lineIndent = 0; + lineStart = first = this.bMarks[line]; + if (line + 1 < end || keepLastLF) { + // No need for bounds check because we have fake entry on tail. + last = this.eMarks[line] + 1; + } else { + last = this.eMarks[line]; + } + while (first < last && lineIndent < indent) { + ch = this.src.charCodeAt(first); + if (isSpace$4(ch)) { + if (ch === 9) { + lineIndent += 4 - (lineIndent + this.bsCount[line]) % 4; + } else { + lineIndent++; + } + } else if (first - lineStart < this.tShift[line]) { + // patched tShift masked characters to look like spaces (blockquotes, list markers) + lineIndent++; + } else { + break; + } + first++; + } + if (lineIndent > indent) { + // partially expanding tabs in code blocks, e.g '\t\tfoobar' + // with indent=2 becomes ' \tfoobar' + queue[i] = new Array(lineIndent - indent + 1).join(" ") + this.src.slice(first, last); + } else { + queue[i] = this.src.slice(first, last); + } + } + return queue.join(""); + }; + // re-export Token class to use in block rules + StateBlock.prototype.Token = token; + var state_block = StateBlock; + var _rules$1 = [ + // First 2 params - rule name & source. Secondary array - list of rules, + // which can be terminated by this one. + [ "table", table, [ "paragraph", "reference" ] ], [ "code", code ], [ "fence", fence, [ "paragraph", "reference", "blockquote", "list" ] ], [ "blockquote", blockquote, [ "paragraph", "reference", "blockquote", "list" ] ], [ "hr", hr, [ "paragraph", "reference", "blockquote", "list" ] ], [ "list", list, [ "paragraph", "reference", "blockquote" ] ], [ "reference", reference ], [ "html_block", html_block, [ "paragraph", "reference", "blockquote" ] ], [ "heading", heading, [ "paragraph", "reference", "blockquote" ] ], [ "lheading", lheading ], [ "paragraph", paragraph ] ]; + /** + * new ParserBlock() + **/ function ParserBlock() { + /** + * ParserBlock#ruler -> Ruler + * + * [[Ruler]] instance. Keep configuration of block rules. + **/ + this.ruler = new ruler; + for (var i = 0; i < _rules$1.length; i++) { + this.ruler.push(_rules$1[i][0], _rules$1[i][1], { + alt: (_rules$1[i][2] || []).slice() + }); + } + } + // Generate tokens for input range + + ParserBlock.prototype.tokenize = function(state, startLine, endLine) { + var ok, i, rules = this.ruler.getRules(""), len = rules.length, line = startLine, hasEmptyLines = false, maxNesting = state.md.options.maxNesting; + while (line < endLine) { + state.line = line = state.skipEmptyLines(line); + if (line >= endLine) { + break; + } + // Termination condition for nested calls. + // Nested calls currently used for blockquotes & lists + if (state.sCount[line] < state.blkIndent) { + break; + } + // If nesting level exceeded - skip tail to the end. That's not ordinary + // situation and we should not care about content. + if (state.level >= maxNesting) { + state.line = endLine; + break; + } + // Try all possible rules. + // On success, rule should: + + // - update `state.line` + // - update `state.tokens` + // - return true + for (i = 0; i < len; i++) { + ok = rules[i](state, line, endLine, false); + if (ok) { + break; + } + } + // set state.tight if we had an empty line before current tag + // i.e. latest empty line should not count + state.tight = !hasEmptyLines; + // paragraph might "eat" one newline after it in nested lists + if (state.isEmpty(state.line - 1)) { + hasEmptyLines = true; + } + line = state.line; + if (line < endLine && state.isEmpty(line)) { + hasEmptyLines = true; + line++; + state.line = line; + } + } + }; + /** + * ParserBlock.parse(str, md, env, outTokens) + * + * Process input string and push block tokens into `outTokens` + **/ ParserBlock.prototype.parse = function(src, md, env, outTokens) { + var state; + if (!src) { + return; + } + state = new this.State(src, md, env, outTokens); + this.tokenize(state, state.line, state.lineMax); + }; + ParserBlock.prototype.State = state_block; + var parser_block = ParserBlock; + // Skip text characters for text token, place those to pending buffer + // Rule to skip pure text + // '{}$%@~+=:' reserved for extentions + // !, ", #, $, %, &, ', (, ), *, +, ,, -, ., /, :, ;, <, =, >, ?, @, [, \, ], ^, _, `, {, |, }, or ~ + // !!!! Don't confuse with "Markdown ASCII Punctuation" chars + // http://spec.commonmark.org/0.15/#ascii-punctuation-character + function isTerminatorChar(ch) { + switch (ch) { + case 10 /* \n */ : + case 33 /* ! */ : + case 35 /* # */ : + case 36 /* $ */ : + case 37 /* % */ : + case 38 /* & */ : + case 42 /* * */ : + case 43 /* + */ : + case 45 /* - */ : + case 58 /* : */ : + case 60 /* < */ : + case 61 /* = */ : + case 62 /* > */ : + case 64 /* @ */ : + case 91 /* [ */ : + case 92 /* \ */ : + case 93 /* ] */ : + case 94 /* ^ */ : + case 95 /* _ */ : + case 96 /* ` */ : + case 123 /* { */ : + case 125 /* } */ : + case 126 /* ~ */ : + return true; + + default: + return false; + } + } + var text = function text(state, silent) { + var pos = state.pos; + while (pos < state.posMax && !isTerminatorChar(state.src.charCodeAt(pos))) { + pos++; + } + if (pos === state.pos) { + return false; + } + if (!silent) { + state.pending += state.src.slice(state.pos, pos); + } + state.pos = pos; + return true; + }; + // Process links like https://example.org/ + // RFC3986: scheme = ALPHA *( ALPHA / DIGIT / "+" / "-" / "." ) + var SCHEME_RE = /(?:^|[^a-z0-9.+-])([a-z][a-z0-9.+-]*)$/i; + var linkify = function linkify(state, silent) { + var pos, max, match, proto, link, url, fullUrl, token; + if (!state.md.options.linkify) return false; + if (state.linkLevel > 0) return false; + pos = state.pos; + max = state.posMax; + if (pos + 3 > max) return false; + if (state.src.charCodeAt(pos) !== 58 /* : */) return false; + if (state.src.charCodeAt(pos + 1) !== 47 /* / */) return false; + if (state.src.charCodeAt(pos + 2) !== 47 /* / */) return false; + match = state.pending.match(SCHEME_RE); + if (!match) return false; + proto = match[1]; + link = state.md.linkify.matchAtStart(state.src.slice(pos - proto.length)); + if (!link) return false; + url = link.url; + // disallow '*' at the end of the link (conflicts with emphasis) + url = url.replace(/\*+$/, ""); + fullUrl = state.md.normalizeLink(url); + if (!state.md.validateLink(fullUrl)) return false; + if (!silent) { + state.pending = state.pending.slice(0, -proto.length); + token = state.push("link_open", "a", 1); + token.attrs = [ [ "href", fullUrl ] ]; + token.markup = "linkify"; + token.info = "auto"; + token = state.push("text", "", 0); + token.content = state.md.normalizeLinkText(url); + token = state.push("link_close", "a", -1); + token.markup = "linkify"; + token.info = "auto"; + } + state.pos += url.length - proto.length; + return true; + }; + var isSpace$3 = utils.isSpace; + var newline = function newline(state, silent) { + var pmax, max, ws, pos = state.pos; + if (state.src.charCodeAt(pos) !== 10 /* \n */) { + return false; + } + pmax = state.pending.length - 1; + max = state.posMax; + // ' \n' -> hardbreak + // Lookup in pending chars is bad practice! Don't copy to other rules! + // Pending string is stored in concat mode, indexed lookups will cause + // convertion to flat mode. + if (!silent) { + if (pmax >= 0 && state.pending.charCodeAt(pmax) === 32) { + if (pmax >= 1 && state.pending.charCodeAt(pmax - 1) === 32) { + // Find whitespaces tail of pending chars. + ws = pmax - 1; + while (ws >= 1 && state.pending.charCodeAt(ws - 1) === 32) ws--; + state.pending = state.pending.slice(0, ws); + state.push("hardbreak", "br", 0); + } else { + state.pending = state.pending.slice(0, -1); + state.push("softbreak", "br", 0); + } + } else { + state.push("softbreak", "br", 0); + } + } + pos++; + // skip heading spaces for next line + while (pos < max && isSpace$3(state.src.charCodeAt(pos))) { + pos++; + } + state.pos = pos; + return true; + }; + var isSpace$2 = utils.isSpace; + var ESCAPED = []; + for (var i = 0; i < 256; i++) { + ESCAPED.push(0); + } + "\\!\"#$%&'()*+,./:;<=>?@[]^_`{|}~-".split("").forEach((function(ch) { + ESCAPED[ch.charCodeAt(0)] = 1; + })); + var _escape = function escape(state, silent) { + var ch1, ch2, origStr, escapedStr, token, pos = state.pos, max = state.posMax; + if (state.src.charCodeAt(pos) !== 92 /* \ */) return false; + pos++; + // '\' at the end of the inline block + if (pos >= max) return false; + ch1 = state.src.charCodeAt(pos); + if (ch1 === 10) { + if (!silent) { + state.push("hardbreak", "br", 0); + } + pos++; + // skip leading whitespaces from next line + while (pos < max) { + ch1 = state.src.charCodeAt(pos); + if (!isSpace$2(ch1)) break; + pos++; + } + state.pos = pos; + return true; + } + escapedStr = state.src[pos]; + if (ch1 >= 55296 && ch1 <= 56319 && pos + 1 < max) { + ch2 = state.src.charCodeAt(pos + 1); + if (ch2 >= 56320 && ch2 <= 57343) { + escapedStr += state.src[pos + 1]; + pos++; + } + } + origStr = "\\" + escapedStr; + if (!silent) { + token = state.push("text_special", "", 0); + if (ch1 < 256 && ESCAPED[ch1] !== 0) { + token.content = escapedStr; + } else { + token.content = origStr; + } + token.markup = origStr; + token.info = "escape"; + } + state.pos = pos + 1; + return true; + }; + // Parse backticks + var backticks = function backtick(state, silent) { + var start, max, marker, token, matchStart, matchEnd, openerLength, closerLength, pos = state.pos, ch = state.src.charCodeAt(pos); + if (ch !== 96 /* ` */) { + return false; + } + start = pos; + pos++; + max = state.posMax; + // scan marker length + while (pos < max && state.src.charCodeAt(pos) === 96 /* ` */) { + pos++; + } + marker = state.src.slice(start, pos); + openerLength = marker.length; + if (state.backticksScanned && (state.backticks[openerLength] || 0) <= start) { + if (!silent) state.pending += marker; + state.pos += openerLength; + return true; + } + matchStart = matchEnd = pos; + // Nothing found in the cache, scan until the end of the line (or until marker is found) + while ((matchStart = state.src.indexOf("`", matchEnd)) !== -1) { + matchEnd = matchStart + 1; + // scan marker length + while (matchEnd < max && state.src.charCodeAt(matchEnd) === 96 /* ` */) { + matchEnd++; + } + closerLength = matchEnd - matchStart; + if (closerLength === openerLength) { + // Found matching closer length. + if (!silent) { + token = state.push("code_inline", "code", 0); + token.markup = marker; + token.content = state.src.slice(pos, matchStart).replace(/\n/g, " ").replace(/^ (.+) $/, "$1"); + } + state.pos = matchEnd; + return true; + } + // Some different length found, put it in cache as upper limit of where closer can be found + state.backticks[closerLength] = matchStart; + } + // Scanned through the end, didn't find anything + state.backticksScanned = true; + if (!silent) state.pending += marker; + state.pos += openerLength; + return true; + }; + // ~~strike through~~ + // Insert each marker as a separate text token, and add it to delimiter list + + var tokenize$1 = function strikethrough(state, silent) { + var i, scanned, token, len, ch, start = state.pos, marker = state.src.charCodeAt(start); + if (silent) { + return false; + } + if (marker !== 126 /* ~ */) { + return false; + } + scanned = state.scanDelims(state.pos, true); + len = scanned.length; + ch = String.fromCharCode(marker); + if (len < 2) { + return false; + } + if (len % 2) { + token = state.push("text", "", 0); + token.content = ch; + len--; + } + for (i = 0; i < len; i += 2) { + token = state.push("text", "", 0); + token.content = ch + ch; + state.delimiters.push({ + marker: marker, + length: 0, + // disable "rule of 3" length checks meant for emphasis + token: state.tokens.length - 1, + end: -1, + open: scanned.can_open, + close: scanned.can_close + }); + } + state.pos += scanned.length; + return true; + }; + function postProcess$1(state, delimiters) { + var i, j, startDelim, endDelim, token, loneMarkers = [], max = delimiters.length; + for (i = 0; i < max; i++) { + startDelim = delimiters[i]; + if (startDelim.marker !== 126 /* ~ */) { + continue; + } + if (startDelim.end === -1) { + continue; + } + endDelim = delimiters[startDelim.end]; + token = state.tokens[startDelim.token]; + token.type = "s_open"; + token.tag = "s"; + token.nesting = 1; + token.markup = "~~"; + token.content = ""; + token = state.tokens[endDelim.token]; + token.type = "s_close"; + token.tag = "s"; + token.nesting = -1; + token.markup = "~~"; + token.content = ""; + if (state.tokens[endDelim.token - 1].type === "text" && state.tokens[endDelim.token - 1].content === "~") { + loneMarkers.push(endDelim.token - 1); + } + } + // If a marker sequence has an odd number of characters, it's splitted + // like this: `~~~~~` -> `~` + `~~` + `~~`, leaving one marker at the + // start of the sequence. + + // So, we have to move all those markers after subsequent s_close tags. + + while (loneMarkers.length) { + i = loneMarkers.pop(); + j = i + 1; + while (j < state.tokens.length && state.tokens[j].type === "s_close") { + j++; + } + j--; + if (i !== j) { + token = state.tokens[j]; + state.tokens[j] = state.tokens[i]; + state.tokens[i] = token; + } + } + } + // Walk through delimiter list and replace text tokens with tags + + var postProcess_1$1 = function strikethrough(state) { + var curr, tokens_meta = state.tokens_meta, max = state.tokens_meta.length; + postProcess$1(state, state.delimiters); + for (curr = 0; curr < max; curr++) { + if (tokens_meta[curr] && tokens_meta[curr].delimiters) { + postProcess$1(state, tokens_meta[curr].delimiters); + } + } + }; + var strikethrough = { + tokenize: tokenize$1, + postProcess: postProcess_1$1 + }; + // Process *this* and _that_ + // Insert each marker as a separate text token, and add it to delimiter list + + var tokenize = function emphasis(state, silent) { + var i, scanned, token, start = state.pos, marker = state.src.charCodeAt(start); + if (silent) { + return false; + } + if (marker !== 95 /* _ */ && marker !== 42 /* * */) { + return false; + } + scanned = state.scanDelims(state.pos, marker === 42); + for (i = 0; i < scanned.length; i++) { + token = state.push("text", "", 0); + token.content = String.fromCharCode(marker); + state.delimiters.push({ + // Char code of the starting marker (number). + marker: marker, + // Total length of these series of delimiters. + length: scanned.length, + // A position of the token this delimiter corresponds to. + token: state.tokens.length - 1, + // If this delimiter is matched as a valid opener, `end` will be + // equal to its position, otherwise it's `-1`. + end: -1, + // Boolean flags that determine if this delimiter could open or close + // an emphasis. + open: scanned.can_open, + close: scanned.can_close + }); + } + state.pos += scanned.length; + return true; + }; + function postProcess(state, delimiters) { + var i, startDelim, endDelim, token, ch, isStrong, max = delimiters.length; + for (i = max - 1; i >= 0; i--) { + startDelim = delimiters[i]; + if (startDelim.marker !== 95 /* _ */ && startDelim.marker !== 42 /* * */) { + continue; + } + // Process only opening markers + if (startDelim.end === -1) { + continue; + } + endDelim = delimiters[startDelim.end]; + // If the previous delimiter has the same marker and is adjacent to this one, + // merge those into one strong delimiter. + + // `whatever` -> `whatever` + + isStrong = i > 0 && delimiters[i - 1].end === startDelim.end + 1 && + // check that first two markers match and adjacent + delimiters[i - 1].marker === startDelim.marker && delimiters[i - 1].token === startDelim.token - 1 && + // check that last two markers are adjacent (we can safely assume they match) + delimiters[startDelim.end + 1].token === endDelim.token + 1; + ch = String.fromCharCode(startDelim.marker); + token = state.tokens[startDelim.token]; + token.type = isStrong ? "strong_open" : "em_open"; + token.tag = isStrong ? "strong" : "em"; + token.nesting = 1; + token.markup = isStrong ? ch + ch : ch; + token.content = ""; + token = state.tokens[endDelim.token]; + token.type = isStrong ? "strong_close" : "em_close"; + token.tag = isStrong ? "strong" : "em"; + token.nesting = -1; + token.markup = isStrong ? ch + ch : ch; + token.content = ""; + if (isStrong) { + state.tokens[delimiters[i - 1].token].content = ""; + state.tokens[delimiters[startDelim.end + 1].token].content = ""; + i--; + } + } + } + // Walk through delimiter list and replace text tokens with tags + + var postProcess_1 = function emphasis(state) { + var curr, tokens_meta = state.tokens_meta, max = state.tokens_meta.length; + postProcess(state, state.delimiters); + for (curr = 0; curr < max; curr++) { + if (tokens_meta[curr] && tokens_meta[curr].delimiters) { + postProcess(state, tokens_meta[curr].delimiters); + } + } + }; + var emphasis = { + tokenize: tokenize, + postProcess: postProcess_1 + }; + var normalizeReference$1 = utils.normalizeReference; + var isSpace$1 = utils.isSpace; + var link = function link(state, silent) { + var attrs, code, label, labelEnd, labelStart, pos, res, ref, token, href = "", title = "", oldPos = state.pos, max = state.posMax, start = state.pos, parseReference = true; + if (state.src.charCodeAt(state.pos) !== 91 /* [ */) { + return false; + } + labelStart = state.pos + 1; + labelEnd = state.md.helpers.parseLinkLabel(state, state.pos, true); + // parser failed to find ']', so it's not a valid link + if (labelEnd < 0) { + return false; + } + pos = labelEnd + 1; + if (pos < max && state.src.charCodeAt(pos) === 40 /* ( */) { + // Inline link + // might have found a valid shortcut link, disable reference parsing + parseReference = false; + // [link]( "title" ) + // ^^ skipping these spaces + pos++; + for (;pos < max; pos++) { + code = state.src.charCodeAt(pos); + if (!isSpace$1(code) && code !== 10) { + break; + } + } + if (pos >= max) { + return false; + } + // [link]( "title" ) + // ^^^^^^ parsing link destination + start = pos; + res = state.md.helpers.parseLinkDestination(state.src, pos, state.posMax); + if (res.ok) { + href = state.md.normalizeLink(res.str); + if (state.md.validateLink(href)) { + pos = res.pos; + } else { + href = ""; + } + // [link]( "title" ) + // ^^ skipping these spaces + start = pos; + for (;pos < max; pos++) { + code = state.src.charCodeAt(pos); + if (!isSpace$1(code) && code !== 10) { + break; + } + } + // [link]( "title" ) + // ^^^^^^^ parsing link title + res = state.md.helpers.parseLinkTitle(state.src, pos, state.posMax); + if (pos < max && start !== pos && res.ok) { + title = res.str; + pos = res.pos; + // [link]( "title" ) + // ^^ skipping these spaces + for (;pos < max; pos++) { + code = state.src.charCodeAt(pos); + if (!isSpace$1(code) && code !== 10) { + break; + } + } + } + } + if (pos >= max || state.src.charCodeAt(pos) !== 41 /* ) */) { + // parsing a valid shortcut link failed, fallback to reference + parseReference = true; + } + pos++; + } + if (parseReference) { + // Link reference + if (typeof state.env.references === "undefined") { + return false; + } + if (pos < max && state.src.charCodeAt(pos) === 91 /* [ */) { + start = pos + 1; + pos = state.md.helpers.parseLinkLabel(state, pos); + if (pos >= 0) { + label = state.src.slice(start, pos++); + } else { + pos = labelEnd + 1; + } + } else { + pos = labelEnd + 1; + } + // covers label === '' and label === undefined + // (collapsed reference link and shortcut reference link respectively) + if (!label) { + label = state.src.slice(labelStart, labelEnd); + } + ref = state.env.references[normalizeReference$1(label)]; + if (!ref) { + state.pos = oldPos; + return false; + } + href = ref.href; + title = ref.title; + } + + // We found the end of the link, and know for a fact it's a valid link; + // so all that's left to do is to call tokenizer. + + if (!silent) { + state.pos = labelStart; + state.posMax = labelEnd; + token = state.push("link_open", "a", 1); + token.attrs = attrs = [ [ "href", href ] ]; + if (title) { + attrs.push([ "title", title ]); + } + state.linkLevel++; + state.md.inline.tokenize(state); + state.linkLevel--; + token = state.push("link_close", "a", -1); + } + state.pos = pos; + state.posMax = max; + return true; + }; + var normalizeReference = utils.normalizeReference; + var isSpace = utils.isSpace; + var image = function image(state, silent) { + var attrs, code, content, label, labelEnd, labelStart, pos, ref, res, title, token, tokens, start, href = "", oldPos = state.pos, max = state.posMax; + if (state.src.charCodeAt(state.pos) !== 33 /* ! */) { + return false; + } + if (state.src.charCodeAt(state.pos + 1) !== 91 /* [ */) { + return false; + } + labelStart = state.pos + 2; + labelEnd = state.md.helpers.parseLinkLabel(state, state.pos + 1, false); + // parser failed to find ']', so it's not a valid link + if (labelEnd < 0) { + return false; + } + pos = labelEnd + 1; + if (pos < max && state.src.charCodeAt(pos) === 40 /* ( */) { + // Inline link + // [link]( "title" ) + // ^^ skipping these spaces + pos++; + for (;pos < max; pos++) { + code = state.src.charCodeAt(pos); + if (!isSpace(code) && code !== 10) { + break; + } + } + if (pos >= max) { + return false; + } + // [link]( "title" ) + // ^^^^^^ parsing link destination + start = pos; + res = state.md.helpers.parseLinkDestination(state.src, pos, state.posMax); + if (res.ok) { + href = state.md.normalizeLink(res.str); + if (state.md.validateLink(href)) { + pos = res.pos; + } else { + href = ""; + } + } + // [link]( "title" ) + // ^^ skipping these spaces + start = pos; + for (;pos < max; pos++) { + code = state.src.charCodeAt(pos); + if (!isSpace(code) && code !== 10) { + break; + } + } + // [link]( "title" ) + // ^^^^^^^ parsing link title + res = state.md.helpers.parseLinkTitle(state.src, pos, state.posMax); + if (pos < max && start !== pos && res.ok) { + title = res.str; + pos = res.pos; + // [link]( "title" ) + // ^^ skipping these spaces + for (;pos < max; pos++) { + code = state.src.charCodeAt(pos); + if (!isSpace(code) && code !== 10) { + break; + } + } + } else { + title = ""; + } + if (pos >= max || state.src.charCodeAt(pos) !== 41 /* ) */) { + state.pos = oldPos; + return false; + } + pos++; + } else { + // Link reference + if (typeof state.env.references === "undefined") { + return false; + } + if (pos < max && state.src.charCodeAt(pos) === 91 /* [ */) { + start = pos + 1; + pos = state.md.helpers.parseLinkLabel(state, pos); + if (pos >= 0) { + label = state.src.slice(start, pos++); + } else { + pos = labelEnd + 1; + } + } else { + pos = labelEnd + 1; + } + // covers label === '' and label === undefined + // (collapsed reference link and shortcut reference link respectively) + if (!label) { + label = state.src.slice(labelStart, labelEnd); + } + ref = state.env.references[normalizeReference(label)]; + if (!ref) { + state.pos = oldPos; + return false; + } + href = ref.href; + title = ref.title; + } + + // We found the end of the link, and know for a fact it's a valid link; + // so all that's left to do is to call tokenizer. + + if (!silent) { + content = state.src.slice(labelStart, labelEnd); + state.md.inline.parse(content, state.md, state.env, tokens = []); + token = state.push("image", "img", 0); + token.attrs = attrs = [ [ "src", href ], [ "alt", "" ] ]; + token.children = tokens; + token.content = content; + if (title) { + attrs.push([ "title", title ]); + } + } + state.pos = pos; + state.posMax = max; + return true; + }; + // Process autolinks '' + /*eslint max-len:0*/ var EMAIL_RE = /^([a-zA-Z0-9.!#$%&'*+\/=?^_`{|}~-]+@[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)*)$/; + var AUTOLINK_RE = /^([a-zA-Z][a-zA-Z0-9+.\-]{1,31}):([^<>\x00-\x20]*)$/; + var autolink = function autolink(state, silent) { + var url, fullUrl, token, ch, start, max, pos = state.pos; + if (state.src.charCodeAt(pos) !== 60 /* < */) { + return false; + } + start = state.pos; + max = state.posMax; + for (;;) { + if (++pos >= max) return false; + ch = state.src.charCodeAt(pos); + if (ch === 60 /* < */) return false; + if (ch === 62 /* > */) break; + } + url = state.src.slice(start + 1, pos); + if (AUTOLINK_RE.test(url)) { + fullUrl = state.md.normalizeLink(url); + if (!state.md.validateLink(fullUrl)) { + return false; + } + if (!silent) { + token = state.push("link_open", "a", 1); + token.attrs = [ [ "href", fullUrl ] ]; + token.markup = "autolink"; + token.info = "auto"; + token = state.push("text", "", 0); + token.content = state.md.normalizeLinkText(url); + token = state.push("link_close", "a", -1); + token.markup = "autolink"; + token.info = "auto"; + } + state.pos += url.length + 2; + return true; + } + if (EMAIL_RE.test(url)) { + fullUrl = state.md.normalizeLink("mailto:" + url); + if (!state.md.validateLink(fullUrl)) { + return false; + } + if (!silent) { + token = state.push("link_open", "a", 1); + token.attrs = [ [ "href", fullUrl ] ]; + token.markup = "autolink"; + token.info = "auto"; + token = state.push("text", "", 0); + token.content = state.md.normalizeLinkText(url); + token = state.push("link_close", "a", -1); + token.markup = "autolink"; + token.info = "auto"; + } + state.pos += url.length + 2; + return true; + } + return false; + }; + var HTML_TAG_RE = html_re.HTML_TAG_RE; + function isLinkOpen(str) { + return /^\s]/i.test(str); + } + function isLinkClose(str) { + return /^<\/a\s*>/i.test(str); + } + function isLetter(ch) { + /*eslint no-bitwise:0*/ + var lc = ch | 32; + // to lower case + return lc >= 97 /* a */ && lc <= 122 /* z */; + } + var html_inline = function html_inline(state, silent) { + var ch, match, max, token, pos = state.pos; + if (!state.md.options.html) { + return false; + } + // Check start + max = state.posMax; + if (state.src.charCodeAt(pos) !== 60 /* < */ || pos + 2 >= max) { + return false; + } + // Quick fail on second char + ch = state.src.charCodeAt(pos + 1); + if (ch !== 33 /* ! */ && ch !== 63 /* ? */ && ch !== 47 /* / */ && !isLetter(ch)) { + return false; + } + match = state.src.slice(pos).match(HTML_TAG_RE); + if (!match) { + return false; + } + if (!silent) { + token = state.push("html_inline", "", 0); + token.content = state.src.slice(pos, pos + match[0].length); + if (isLinkOpen(token.content)) state.linkLevel++; + if (isLinkClose(token.content)) state.linkLevel--; + } + state.pos += match[0].length; + return true; + }; + var has = utils.has; + var isValidEntityCode = utils.isValidEntityCode; + var fromCodePoint = utils.fromCodePoint; + var DIGITAL_RE = /^&#((?:x[a-f0-9]{1,6}|[0-9]{1,7}));/i; + var NAMED_RE = /^&([a-z][a-z0-9]{1,31});/i; + var entity = function entity(state, silent) { + var ch, code, match, token, pos = state.pos, max = state.posMax; + if (state.src.charCodeAt(pos) !== 38 /* & */) return false; + if (pos + 1 >= max) return false; + ch = state.src.charCodeAt(pos + 1); + if (ch === 35 /* # */) { + match = state.src.slice(pos).match(DIGITAL_RE); + if (match) { + if (!silent) { + code = match[1][0].toLowerCase() === "x" ? parseInt(match[1].slice(1), 16) : parseInt(match[1], 10); + token = state.push("text_special", "", 0); + token.content = isValidEntityCode(code) ? fromCodePoint(code) : fromCodePoint(65533); + token.markup = match[0]; + token.info = "entity"; + } + state.pos += match[0].length; + return true; + } + } else { + match = state.src.slice(pos).match(NAMED_RE); + if (match) { + if (has(entities, match[1])) { + if (!silent) { + token = state.push("text_special", "", 0); + token.content = entities[match[1]]; + token.markup = match[0]; + token.info = "entity"; + } + state.pos += match[0].length; + return true; + } + } + } + return false; + }; + // For each opening emphasis-like marker find a matching closing one + function processDelimiters(state, delimiters) { + var closerIdx, openerIdx, closer, opener, minOpenerIdx, newMinOpenerIdx, isOddMatch, lastJump, openersBottom = {}, max = delimiters.length; + if (!max) return; + // headerIdx is the first delimiter of the current (where closer is) delimiter run + var headerIdx = 0; + var lastTokenIdx = -2; + // needs any value lower than -1 + var jumps = []; + for (closerIdx = 0; closerIdx < max; closerIdx++) { + closer = delimiters[closerIdx]; + jumps.push(0); + // markers belong to same delimiter run if: + // - they have adjacent tokens + // - AND markers are the same + + if (delimiters[headerIdx].marker !== closer.marker || lastTokenIdx !== closer.token - 1) { + headerIdx = closerIdx; + } + lastTokenIdx = closer.token; + // Length is only used for emphasis-specific "rule of 3", + // if it's not defined (in strikethrough or 3rd party plugins), + // we can default it to 0 to disable those checks. + + closer.length = closer.length || 0; + if (!closer.close) continue; + // Previously calculated lower bounds (previous fails) + // for each marker, each delimiter length modulo 3, + // and for whether this closer can be an opener; + // https://github.com/commonmark/cmark/commit/34250e12ccebdc6372b8b49c44fab57c72443460 + if (!openersBottom.hasOwnProperty(closer.marker)) { + openersBottom[closer.marker] = [ -1, -1, -1, -1, -1, -1 ]; + } + minOpenerIdx = openersBottom[closer.marker][(closer.open ? 3 : 0) + closer.length % 3]; + openerIdx = headerIdx - jumps[headerIdx] - 1; + newMinOpenerIdx = openerIdx; + for (;openerIdx > minOpenerIdx; openerIdx -= jumps[openerIdx] + 1) { + opener = delimiters[openerIdx]; + if (opener.marker !== closer.marker) continue; + if (opener.open && opener.end < 0) { + isOddMatch = false; + // from spec: + + // If one of the delimiters can both open and close emphasis, then the + // sum of the lengths of the delimiter runs containing the opening and + // closing delimiters must not be a multiple of 3 unless both lengths + // are multiples of 3. + + if (opener.close || closer.open) { + if ((opener.length + closer.length) % 3 === 0) { + if (opener.length % 3 !== 0 || closer.length % 3 !== 0) { + isOddMatch = true; + } + } + } + if (!isOddMatch) { + // If previous delimiter cannot be an opener, we can safely skip + // the entire sequence in future checks. This is required to make + // sure algorithm has linear complexity (see *_*_*_*_*_... case). + lastJump = openerIdx > 0 && !delimiters[openerIdx - 1].open ? jumps[openerIdx - 1] + 1 : 0; + jumps[closerIdx] = closerIdx - openerIdx + lastJump; + jumps[openerIdx] = lastJump; + closer.open = false; + opener.end = closerIdx; + opener.close = false; + newMinOpenerIdx = -1; + // treat next token as start of run, + // it optimizes skips in **<...>**a**<...>** pathological case + lastTokenIdx = -2; + break; + } + } + } + if (newMinOpenerIdx !== -1) { + // If match for this delimiter run failed, we want to set lower bound for + // future lookups. This is required to make sure algorithm has linear + // complexity. + // See details here: + // https://github.com/commonmark/cmark/issues/178#issuecomment-270417442 + openersBottom[closer.marker][(closer.open ? 3 : 0) + (closer.length || 0) % 3] = newMinOpenerIdx; + } + } + } + var balance_pairs = function link_pairs(state) { + var curr, tokens_meta = state.tokens_meta, max = state.tokens_meta.length; + processDelimiters(state, state.delimiters); + for (curr = 0; curr < max; curr++) { + if (tokens_meta[curr] && tokens_meta[curr].delimiters) { + processDelimiters(state, tokens_meta[curr].delimiters); + } + } + }; + // Clean up tokens after emphasis and strikethrough postprocessing: + var fragments_join = function fragments_join(state) { + var curr, last, level = 0, tokens = state.tokens, max = state.tokens.length; + for (curr = last = 0; curr < max; curr++) { + // re-calculate levels after emphasis/strikethrough turns some text nodes + // into opening/closing tags + if (tokens[curr].nesting < 0) level--; + // closing tag + tokens[curr].level = level; + if (tokens[curr].nesting > 0) level++; + // opening tag + if (tokens[curr].type === "text" && curr + 1 < max && tokens[curr + 1].type === "text") { + // collapse two adjacent text nodes + tokens[curr + 1].content = tokens[curr].content + tokens[curr + 1].content; + } else { + if (curr !== last) { + tokens[last] = tokens[curr]; + } + last++; + } + } + if (curr !== last) { + tokens.length = last; + } + }; + var isWhiteSpace = utils.isWhiteSpace; + var isPunctChar = utils.isPunctChar; + var isMdAsciiPunct = utils.isMdAsciiPunct; + function StateInline(src, md, env, outTokens) { + this.src = src; + this.env = env; + this.md = md; + this.tokens = outTokens; + this.tokens_meta = Array(outTokens.length); + this.pos = 0; + this.posMax = this.src.length; + this.level = 0; + this.pending = ""; + this.pendingLevel = 0; + // Stores { start: end } pairs. Useful for backtrack + // optimization of pairs parse (emphasis, strikes). + this.cache = {}; + // List of emphasis-like delimiters for current tag + this.delimiters = []; + // Stack of delimiter lists for upper level tags + this._prev_delimiters = []; + // backtick length => last seen position + this.backticks = {}; + this.backticksScanned = false; + // Counter used to disable inline linkify-it execution + // inside and markdown links + this.linkLevel = 0; + } + // Flush pending text + + StateInline.prototype.pushPending = function() { + var token$1 = new token("text", "", 0); + token$1.content = this.pending; + token$1.level = this.pendingLevel; + this.tokens.push(token$1); + this.pending = ""; + return token$1; + }; + // Push new token to "stream". + // If pending text exists - flush it as text token + + StateInline.prototype.push = function(type, tag, nesting) { + if (this.pending) { + this.pushPending(); + } + var token$1 = new token(type, tag, nesting); + var token_meta = null; + if (nesting < 0) { + // closing tag + this.level--; + this.delimiters = this._prev_delimiters.pop(); + } + token$1.level = this.level; + if (nesting > 0) { + // opening tag + this.level++; + this._prev_delimiters.push(this.delimiters); + this.delimiters = []; + token_meta = { + delimiters: this.delimiters + }; + } + this.pendingLevel = this.level; + this.tokens.push(token$1); + this.tokens_meta.push(token_meta); + return token$1; + }; + // Scan a sequence of emphasis-like markers, and determine whether + // it can start an emphasis sequence or end an emphasis sequence. + + // - start - position to scan from (it should point at a valid marker); + // - canSplitWord - determine if these markers can be found inside a word + + StateInline.prototype.scanDelims = function(start, canSplitWord) { + var pos = start, lastChar, nextChar, count, can_open, can_close, isLastWhiteSpace, isLastPunctChar, isNextWhiteSpace, isNextPunctChar, left_flanking = true, right_flanking = true, max = this.posMax, marker = this.src.charCodeAt(start); + // treat beginning of the line as a whitespace + lastChar = start > 0 ? this.src.charCodeAt(start - 1) : 32; + while (pos < max && this.src.charCodeAt(pos) === marker) { + pos++; + } + count = pos - start; + // treat end of the line as a whitespace + nextChar = pos < max ? this.src.charCodeAt(pos) : 32; + isLastPunctChar = isMdAsciiPunct(lastChar) || isPunctChar(String.fromCharCode(lastChar)); + isNextPunctChar = isMdAsciiPunct(nextChar) || isPunctChar(String.fromCharCode(nextChar)); + isLastWhiteSpace = isWhiteSpace(lastChar); + isNextWhiteSpace = isWhiteSpace(nextChar); + if (isNextWhiteSpace) { + left_flanking = false; + } else if (isNextPunctChar) { + if (!(isLastWhiteSpace || isLastPunctChar)) { + left_flanking = false; + } + } + if (isLastWhiteSpace) { + right_flanking = false; + } else if (isLastPunctChar) { + if (!(isNextWhiteSpace || isNextPunctChar)) { + right_flanking = false; + } + } + if (!canSplitWord) { + can_open = left_flanking && (!right_flanking || isLastPunctChar); + can_close = right_flanking && (!left_flanking || isNextPunctChar); + } else { + can_open = left_flanking; + can_close = right_flanking; + } + return { + can_open: can_open, + can_close: can_close, + length: count + }; + }; + // re-export Token class to use in block rules + StateInline.prototype.Token = token; + var state_inline = StateInline; + //////////////////////////////////////////////////////////////////////////////// + // Parser rules + var _rules = [ [ "text", text ], [ "linkify", linkify ], [ "newline", newline ], [ "escape", _escape ], [ "backticks", backticks ], [ "strikethrough", strikethrough.tokenize ], [ "emphasis", emphasis.tokenize ], [ "link", link ], [ "image", image ], [ "autolink", autolink ], [ "html_inline", html_inline ], [ "entity", entity ] ]; + // `rule2` ruleset was created specifically for emphasis/strikethrough + // post-processing and may be changed in the future. + + // Don't use this for anything except pairs (plugins working with `balance_pairs`). + + var _rules2 = [ [ "balance_pairs", balance_pairs ], [ "strikethrough", strikethrough.postProcess ], [ "emphasis", emphasis.postProcess ], + // rules for pairs separate '**' into its own text tokens, which may be left unused, + // rule below merges unused segments back with the rest of the text + [ "fragments_join", fragments_join ] ]; + /** + * new ParserInline() + **/ function ParserInline() { + var i; + /** + * ParserInline#ruler -> Ruler + * + * [[Ruler]] instance. Keep configuration of inline rules. + **/ this.ruler = new ruler; + for (i = 0; i < _rules.length; i++) { + this.ruler.push(_rules[i][0], _rules[i][1]); + } + /** + * ParserInline#ruler2 -> Ruler + * + * [[Ruler]] instance. Second ruler used for post-processing + * (e.g. in emphasis-like rules). + **/ this.ruler2 = new ruler; + for (i = 0; i < _rules2.length; i++) { + this.ruler2.push(_rules2[i][0], _rules2[i][1]); + } + } + // Skip single token by running all rules in validation mode; + // returns `true` if any rule reported success + + ParserInline.prototype.skipToken = function(state) { + var ok, i, pos = state.pos, rules = this.ruler.getRules(""), len = rules.length, maxNesting = state.md.options.maxNesting, cache = state.cache; + if (typeof cache[pos] !== "undefined") { + state.pos = cache[pos]; + return; + } + if (state.level < maxNesting) { + for (i = 0; i < len; i++) { + // Increment state.level and decrement it later to limit recursion. + // It's harmless to do here, because no tokens are created. But ideally, + // we'd need a separate private state variable for this purpose. + state.level++; + ok = rules[i](state, true); + state.level--; + if (ok) { + break; + } + } + } else { + // Too much nesting, just skip until the end of the paragraph. + // NOTE: this will cause links to behave incorrectly in the following case, + // when an amount of `[` is exactly equal to `maxNesting + 1`: + // [[[[[[[[[[[[[[[[[[[[[foo]() + // TODO: remove this workaround when CM standard will allow nested links + // (we can replace it by preventing links from being parsed in + // validation mode) + state.pos = state.posMax; + } + if (!ok) { + state.pos++; + } + cache[pos] = state.pos; + }; + // Generate tokens for input range + + ParserInline.prototype.tokenize = function(state) { + var ok, i, rules = this.ruler.getRules(""), len = rules.length, end = state.posMax, maxNesting = state.md.options.maxNesting; + while (state.pos < end) { + // Try all possible rules. + // On success, rule should: + // - update `state.pos` + // - update `state.tokens` + // - return true + if (state.level < maxNesting) { + for (i = 0; i < len; i++) { + ok = rules[i](state, false); + if (ok) { + break; + } + } + } + if (ok) { + if (state.pos >= end) { + break; + } + continue; + } + state.pending += state.src[state.pos++]; + } + if (state.pending) { + state.pushPending(); + } + }; + /** + * ParserInline.parse(str, md, env, outTokens) + * + * Process input string and push inline tokens into `outTokens` + **/ ParserInline.prototype.parse = function(str, md, env, outTokens) { + var i, rules, len; + var state = new this.State(str, md, env, outTokens); + this.tokenize(state); + rules = this.ruler2.getRules(""); + len = rules.length; + for (i = 0; i < len; i++) { + rules[i](state); + } + }; + ParserInline.prototype.State = state_inline; + var parser_inline = ParserInline; + var re = function(opts) { + var re = {}; + opts = opts || {}; + // Use direct extract instead of `regenerate` to reduse browserified size + re.src_Any = regex$3.source; + re.src_Cc = regex$2.source; + re.src_Z = regex.source; + re.src_P = regex$4.source; + // \p{\Z\P\Cc\CF} (white spaces + control + format + punctuation) + re.src_ZPCc = [ re.src_Z, re.src_P, re.src_Cc ].join("|"); + // \p{\Z\Cc} (white spaces + control) + re.src_ZCc = [ re.src_Z, re.src_Cc ].join("|"); + // Experimental. List of chars, completely prohibited in links + // because can separate it from other part of text + var text_separators = "[><\uff5c]"; + // All possible word characters (everything without punctuation, spaces & controls) + // Defined via punctuation & spaces to save space + // Should be something like \p{\L\N\S\M} (\w but without `_`) + re.src_pseudo_letter = "(?:(?!" + text_separators + "|" + re.src_ZPCc + ")" + re.src_Any + ")"; + // The same as abothe but without [0-9] + // var src_pseudo_letter_non_d = '(?:(?![0-9]|' + src_ZPCc + ')' + src_Any + ')'; + //////////////////////////////////////////////////////////////////////////////// + re.src_ip4 = "(?:(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.){3}(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)"; + // Prohibit any of "@/[]()" in user/pass to avoid wrong domain fetch. + re.src_auth = "(?:(?:(?!" + re.src_ZCc + "|[@/\\[\\]()]).)+@)?"; + re.src_port = "(?::(?:6(?:[0-4]\\d{3}|5(?:[0-4]\\d{2}|5(?:[0-2]\\d|3[0-5])))|[1-5]?\\d{1,4}))?"; + re.src_host_terminator = "(?=$|" + text_separators + "|" + re.src_ZPCc + ")" + "(?!" + (opts["---"] ? "-(?!--)|" : "-|") + "_|:\\d|\\.-|\\.(?!$|" + re.src_ZPCc + "))"; + re.src_path = "(?:" + "[/?#]" + "(?:" + "(?!" + re.src_ZCc + "|" + text_separators + "|[()[\\]{}.,\"'?!\\-;]).|" + "\\[(?:(?!" + re.src_ZCc + "|\\]).)*\\]|" + "\\((?:(?!" + re.src_ZCc + "|[)]).)*\\)|" + "\\{(?:(?!" + re.src_ZCc + "|[}]).)*\\}|" + '\\"(?:(?!' + re.src_ZCc + '|["]).)+\\"|' + "\\'(?:(?!" + re.src_ZCc + "|[']).)+\\'|" + "\\'(?=" + re.src_pseudo_letter + "|[-])|" + // allow `I'm_king` if no pair found + "\\.{2,}[a-zA-Z0-9%/&]|" + // google has many dots in "google search" links (#66, #81). + // github has ... in commit range links, + // Restrict to + // - english + // - percent-encoded + // - parts of file path + // - params separator + // until more examples found. + "\\.(?!" + re.src_ZCc + "|[.]|$)|" + (opts["---"] ? "\\-(?!--(?:[^-]|$))(?:-*)|" : "\\-+|") + ",(?!" + re.src_ZCc + "|$)|" + // allow `,,,` in paths + ";(?!" + re.src_ZCc + "|$)|" + // allow `;` if not followed by space-like char + "\\!+(?!" + re.src_ZCc + "|[!]|$)|" + // allow `!!!` in paths, but not at the end + "\\?(?!" + re.src_ZCc + "|[?]|$)" + ")+" + "|\\/" + ")?"; + // Allow anything in markdown spec, forbid quote (") at the first position + // because emails enclosed in quotes are far more common + re.src_email_name = '[\\-;:&=\\+\\$,\\.a-zA-Z0-9_][\\-;:&=\\+\\$,\\"\\.a-zA-Z0-9_]*'; + re.src_xn = "xn--[a-z0-9\\-]{1,59}"; + // More to read about domain names + // http://serverfault.com/questions/638260/ + re.src_domain_root = + // Allow letters & digits (http://test1) + "(?:" + re.src_xn + "|" + re.src_pseudo_letter + "{1,63}" + ")"; + re.src_domain = "(?:" + re.src_xn + "|" + "(?:" + re.src_pseudo_letter + ")" + "|" + "(?:" + re.src_pseudo_letter + "(?:-|" + re.src_pseudo_letter + "){0,61}" + re.src_pseudo_letter + ")" + ")"; + re.src_host = "(?:" + + // Don't need IP check, because digits are already allowed in normal domain names + // src_ip4 + + // '|' + + "(?:(?:(?:" + re.src_domain + ")\\.)*" + re.src_domain /*_root*/ + ")" + ")"; + re.tpl_host_fuzzy = "(?:" + re.src_ip4 + "|" + "(?:(?:(?:" + re.src_domain + ")\\.)+(?:%TLDS%))" + ")"; + re.tpl_host_no_ip_fuzzy = "(?:(?:(?:" + re.src_domain + ")\\.)+(?:%TLDS%))"; + re.src_host_strict = re.src_host + re.src_host_terminator; + re.tpl_host_fuzzy_strict = re.tpl_host_fuzzy + re.src_host_terminator; + re.src_host_port_strict = re.src_host + re.src_port + re.src_host_terminator; + re.tpl_host_port_fuzzy_strict = re.tpl_host_fuzzy + re.src_port + re.src_host_terminator; + re.tpl_host_port_no_ip_fuzzy_strict = re.tpl_host_no_ip_fuzzy + re.src_port + re.src_host_terminator; + //////////////////////////////////////////////////////////////////////////////// + // Main rules + // Rude test fuzzy links by host, for quick deny + re.tpl_host_fuzzy_test = "localhost|www\\.|\\.\\d{1,3}\\.|(?:\\.(?:%TLDS%)(?:" + re.src_ZPCc + "|>|$))"; + re.tpl_email_fuzzy = "(^|" + text_separators + '|"|\\(|' + re.src_ZCc + ")" + "(" + re.src_email_name + "@" + re.tpl_host_fuzzy_strict + ")"; + re.tpl_link_fuzzy = + // Fuzzy link can't be prepended with .:/\- and non punctuation. + // but can start with > (markdown blockquote) + "(^|(?![.:/\\-_@])(?:[$+<=>^`|\uff5c]|" + re.src_ZPCc + "))" + "((?![$+<=>^`|\uff5c])" + re.tpl_host_port_fuzzy_strict + re.src_path + ")"; + re.tpl_link_no_ip_fuzzy = + // Fuzzy link can't be prepended with .:/\- and non punctuation. + // but can start with > (markdown blockquote) + "(^|(?![.:/\\-_@])(?:[$+<=>^`|\uff5c]|" + re.src_ZPCc + "))" + "((?![$+<=>^`|\uff5c])" + re.tpl_host_port_no_ip_fuzzy_strict + re.src_path + ")"; + return re; + }; + //////////////////////////////////////////////////////////////////////////////// + // Helpers + // Merge objects + + function assign(obj /*from1, from2, from3, ...*/) { + var sources = Array.prototype.slice.call(arguments, 1); + sources.forEach((function(source) { + if (!source) { + return; + } + Object.keys(source).forEach((function(key) { + obj[key] = source[key]; + })); + })); + return obj; + } + function _class(obj) { + return Object.prototype.toString.call(obj); + } + function isString(obj) { + return _class(obj) === "[object String]"; + } + function isObject(obj) { + return _class(obj) === "[object Object]"; + } + function isRegExp(obj) { + return _class(obj) === "[object RegExp]"; + } + function isFunction(obj) { + return _class(obj) === "[object Function]"; + } + function escapeRE(str) { + return str.replace(/[.?*+^$[\]\\(){}|-]/g, "\\$&"); + } + //////////////////////////////////////////////////////////////////////////////// + var defaultOptions = { + fuzzyLink: true, + fuzzyEmail: true, + fuzzyIP: false + }; + function isOptionsObj(obj) { + return Object.keys(obj || {}).reduce((function(acc, k) { + return acc || defaultOptions.hasOwnProperty(k); + }), false); + } + var defaultSchemas = { + "http:": { + validate: function(text, pos, self) { + var tail = text.slice(pos); + if (!self.re.http) { + // compile lazily, because "host"-containing variables can change on tlds update. + self.re.http = new RegExp("^\\/\\/" + self.re.src_auth + self.re.src_host_port_strict + self.re.src_path, "i"); + } + if (self.re.http.test(tail)) { + return tail.match(self.re.http)[0].length; + } + return 0; + } + }, + "https:": "http:", + "ftp:": "http:", + "//": { + validate: function(text, pos, self) { + var tail = text.slice(pos); + if (!self.re.no_http) { + // compile lazily, because "host"-containing variables can change on tlds update. + self.re.no_http = new RegExp("^" + self.re.src_auth + + // Don't allow single-level domains, because of false positives like '//test' + // with code comments + "(?:localhost|(?:(?:" + self.re.src_domain + ")\\.)+" + self.re.src_domain_root + ")" + self.re.src_port + self.re.src_host_terminator + self.re.src_path, "i"); + } + if (self.re.no_http.test(tail)) { + // should not be `://` & `///`, that protects from errors in protocol name + if (pos >= 3 && text[pos - 3] === ":") { + return 0; + } + if (pos >= 3 && text[pos - 3] === "/") { + return 0; + } + return tail.match(self.re.no_http)[0].length; + } + return 0; + } + }, + "mailto:": { + validate: function(text, pos, self) { + var tail = text.slice(pos); + if (!self.re.mailto) { + self.re.mailto = new RegExp("^" + self.re.src_email_name + "@" + self.re.src_host_strict, "i"); + } + if (self.re.mailto.test(tail)) { + return tail.match(self.re.mailto)[0].length; + } + return 0; + } + } + }; + /*eslint-disable max-len*/ + // RE pattern for 2-character tlds (autogenerated by ./support/tlds_2char_gen.js) + var tlds_2ch_src_re = "a[cdefgilmnoqrstuwxz]|b[abdefghijmnorstvwyz]|c[acdfghiklmnoruvwxyz]|d[ejkmoz]|e[cegrstu]|f[ijkmor]|g[abdefghilmnpqrstuwy]|h[kmnrtu]|i[delmnoqrst]|j[emop]|k[eghimnprwyz]|l[abcikrstuvy]|m[acdeghklmnopqrstuvwxyz]|n[acefgilopruz]|om|p[aefghklmnrstwy]|qa|r[eosuw]|s[abcdeghijklmnortuvxyz]|t[cdfghjklmnortvwz]|u[agksyz]|v[aceginu]|w[fs]|y[et]|z[amw]"; + // DON'T try to make PRs with changes. Extend TLDs with LinkifyIt.tlds() instead + var tlds_default = "biz|com|edu|gov|net|org|pro|web|xxx|aero|asia|coop|info|museum|name|shop|\u0440\u0444".split("|"); + /*eslint-enable max-len*/ + //////////////////////////////////////////////////////////////////////////////// + function resetScanCache(self) { + self.__index__ = -1; + self.__text_cache__ = ""; + } + function createValidator(re) { + return function(text, pos) { + var tail = text.slice(pos); + if (re.test(tail)) { + return tail.match(re)[0].length; + } + return 0; + }; + } + function createNormalizer() { + return function(match, self) { + self.normalize(match); + }; + } + // Schemas compiler. Build regexps. + + function compile(self) { + // Load & clone RE patterns. + var re$1 = self.re = re(self.__opts__); + // Define dynamic patterns + var tlds = self.__tlds__.slice(); + self.onCompile(); + if (!self.__tlds_replaced__) { + tlds.push(tlds_2ch_src_re); + } + tlds.push(re$1.src_xn); + re$1.src_tlds = tlds.join("|"); + function untpl(tpl) { + return tpl.replace("%TLDS%", re$1.src_tlds); + } + re$1.email_fuzzy = RegExp(untpl(re$1.tpl_email_fuzzy), "i"); + re$1.link_fuzzy = RegExp(untpl(re$1.tpl_link_fuzzy), "i"); + re$1.link_no_ip_fuzzy = RegExp(untpl(re$1.tpl_link_no_ip_fuzzy), "i"); + re$1.host_fuzzy_test = RegExp(untpl(re$1.tpl_host_fuzzy_test), "i"); + + // Compile each schema + + var aliases = []; + self.__compiled__ = {}; + // Reset compiled data + function schemaError(name, val) { + throw new Error('(LinkifyIt) Invalid schema "' + name + '": ' + val); + } + Object.keys(self.__schemas__).forEach((function(name) { + var val = self.__schemas__[name]; + // skip disabled methods + if (val === null) { + return; + } + var compiled = { + validate: null, + link: null + }; + self.__compiled__[name] = compiled; + if (isObject(val)) { + if (isRegExp(val.validate)) { + compiled.validate = createValidator(val.validate); + } else if (isFunction(val.validate)) { + compiled.validate = val.validate; + } else { + schemaError(name, val); + } + if (isFunction(val.normalize)) { + compiled.normalize = val.normalize; + } else if (!val.normalize) { + compiled.normalize = createNormalizer(); + } else { + schemaError(name, val); + } + return; + } + if (isString(val)) { + aliases.push(name); + return; + } + schemaError(name, val); + })); + + // Compile postponed aliases + + aliases.forEach((function(alias) { + if (!self.__compiled__[self.__schemas__[alias]]) { + // Silently fail on missed schemas to avoid errons on disable. + // schemaError(alias, self.__schemas__[alias]); + return; + } + self.__compiled__[alias].validate = self.__compiled__[self.__schemas__[alias]].validate; + self.__compiled__[alias].normalize = self.__compiled__[self.__schemas__[alias]].normalize; + })); + + // Fake record for guessed links + + self.__compiled__[""] = { + validate: null, + normalize: createNormalizer() + }; + + // Build schema condition + + var slist = Object.keys(self.__compiled__).filter((function(name) { + // Filter disabled & fake schemas + return name.length > 0 && self.__compiled__[name]; + })).map(escapeRE).join("|"); + // (?!_) cause 1.5x slowdown + self.re.schema_test = RegExp("(^|(?!_)(?:[><\uff5c]|" + re$1.src_ZPCc + "))(" + slist + ")", "i"); + self.re.schema_search = RegExp("(^|(?!_)(?:[><\uff5c]|" + re$1.src_ZPCc + "))(" + slist + ")", "ig"); + self.re.schema_at_start = RegExp("^" + self.re.schema_search.source, "i"); + self.re.pretest = RegExp("(" + self.re.schema_test.source + ")|(" + self.re.host_fuzzy_test.source + ")|@", "i"); + + // Cleanup + + resetScanCache(self); + } + /** + * class Match + * + * Match result. Single element of array, returned by [[LinkifyIt#match]] + **/ function Match(self, shift) { + var start = self.__index__, end = self.__last_index__, text = self.__text_cache__.slice(start, end); + /** + * Match#schema -> String + * + * Prefix (protocol) for matched string. + **/ this.schema = self.__schema__.toLowerCase(); + /** + * Match#index -> Number + * + * First position of matched string. + **/ this.index = start + shift; + /** + * Match#lastIndex -> Number + * + * Next position after matched string. + **/ this.lastIndex = end + shift; + /** + * Match#raw -> String + * + * Matched string. + **/ this.raw = text; + /** + * Match#text -> String + * + * Notmalized text of matched string. + **/ this.text = text; + /** + * Match#url -> String + * + * Normalized url of matched string. + **/ this.url = text; + } + function createMatch(self, shift) { + var match = new Match(self, shift); + self.__compiled__[match.schema].normalize(match, self); + return match; + } + /** + * class LinkifyIt + **/ + /** + * new LinkifyIt(schemas, options) + * - schemas (Object): Optional. Additional schemas to validate (prefix/validator) + * - options (Object): { fuzzyLink|fuzzyEmail|fuzzyIP: true|false } + * + * Creates new linkifier instance with optional additional schemas. + * Can be called without `new` keyword for convenience. + * + * By default understands: + * + * - `http(s)://...` , `ftp://...`, `mailto:...` & `//...` links + * - "fuzzy" links and emails (example.com, foo@bar.com). + * + * `schemas` is an object, where each key/value describes protocol/rule: + * + * - __key__ - link prefix (usually, protocol name with `:` at the end, `skype:` + * for example). `linkify-it` makes shure that prefix is not preceeded with + * alphanumeric char and symbols. Only whitespaces and punctuation allowed. + * - __value__ - rule to check tail after link prefix + * - _String_ - just alias to existing rule + * - _Object_ + * - _validate_ - validator function (should return matched length on success), + * or `RegExp`. + * - _normalize_ - optional function to normalize text & url of matched result + * (for example, for @twitter mentions). + * + * `options`: + * + * - __fuzzyLink__ - recognige URL-s without `http(s):` prefix. Default `true`. + * - __fuzzyIP__ - allow IPs in fuzzy links above. Can conflict with some texts + * like version numbers. Default `false`. + * - __fuzzyEmail__ - recognize emails without `mailto:` prefix. + * + **/ function LinkifyIt(schemas, options) { + if (!(this instanceof LinkifyIt)) { + return new LinkifyIt(schemas, options); + } + if (!options) { + if (isOptionsObj(schemas)) { + options = schemas; + schemas = {}; + } + } + this.__opts__ = assign({}, defaultOptions, options); + // Cache last tested result. Used to skip repeating steps on next `match` call. + this.__index__ = -1; + this.__last_index__ = -1; + // Next scan position + this.__schema__ = ""; + this.__text_cache__ = ""; + this.__schemas__ = assign({}, defaultSchemas, schemas); + this.__compiled__ = {}; + this.__tlds__ = tlds_default; + this.__tlds_replaced__ = false; + this.re = {}; + compile(this); + } + /** chainable + * LinkifyIt#add(schema, definition) + * - schema (String): rule name (fixed pattern prefix) + * - definition (String|RegExp|Object): schema definition + * + * Add new rule definition. See constructor description for details. + **/ LinkifyIt.prototype.add = function add(schema, definition) { + this.__schemas__[schema] = definition; + compile(this); + return this; + }; + /** chainable + * LinkifyIt#set(options) + * - options (Object): { fuzzyLink|fuzzyEmail|fuzzyIP: true|false } + * + * Set recognition options for links without schema. + **/ LinkifyIt.prototype.set = function set(options) { + this.__opts__ = assign(this.__opts__, options); + return this; + }; + /** + * LinkifyIt#test(text) -> Boolean + * + * Searches linkifiable pattern and returns `true` on success or `false` on fail. + **/ LinkifyIt.prototype.test = function test(text) { + // Reset scan cache + this.__text_cache__ = text; + this.__index__ = -1; + if (!text.length) { + return false; + } + var m, ml, me, len, shift, next, re, tld_pos, at_pos; + // try to scan for link with schema - that's the most simple rule + if (this.re.schema_test.test(text)) { + re = this.re.schema_search; + re.lastIndex = 0; + while ((m = re.exec(text)) !== null) { + len = this.testSchemaAt(text, m[2], re.lastIndex); + if (len) { + this.__schema__ = m[2]; + this.__index__ = m.index + m[1].length; + this.__last_index__ = m.index + m[0].length + len; + break; + } + } + } + if (this.__opts__.fuzzyLink && this.__compiled__["http:"]) { + // guess schemaless links + tld_pos = text.search(this.re.host_fuzzy_test); + if (tld_pos >= 0) { + // if tld is located after found link - no need to check fuzzy pattern + if (this.__index__ < 0 || tld_pos < this.__index__) { + if ((ml = text.match(this.__opts__.fuzzyIP ? this.re.link_fuzzy : this.re.link_no_ip_fuzzy)) !== null) { + shift = ml.index + ml[1].length; + if (this.__index__ < 0 || shift < this.__index__) { + this.__schema__ = ""; + this.__index__ = shift; + this.__last_index__ = ml.index + ml[0].length; + } + } + } + } + } + if (this.__opts__.fuzzyEmail && this.__compiled__["mailto:"]) { + // guess schemaless emails + at_pos = text.indexOf("@"); + if (at_pos >= 0) { + // We can't skip this check, because this cases are possible: + // 192.168.1.1@gmail.com, my.in@example.com + if ((me = text.match(this.re.email_fuzzy)) !== null) { + shift = me.index + me[1].length; + next = me.index + me[0].length; + if (this.__index__ < 0 || shift < this.__index__ || shift === this.__index__ && next > this.__last_index__) { + this.__schema__ = "mailto:"; + this.__index__ = shift; + this.__last_index__ = next; + } + } + } + } + return this.__index__ >= 0; + }; + /** + * LinkifyIt#pretest(text) -> Boolean + * + * Very quick check, that can give false positives. Returns true if link MAY BE + * can exists. Can be used for speed optimization, when you need to check that + * link NOT exists. + **/ LinkifyIt.prototype.pretest = function pretest(text) { + return this.re.pretest.test(text); + }; + /** + * LinkifyIt#testSchemaAt(text, name, position) -> Number + * - text (String): text to scan + * - name (String): rule (schema) name + * - position (Number): text offset to check from + * + * Similar to [[LinkifyIt#test]] but checks only specific protocol tail exactly + * at given position. Returns length of found pattern (0 on fail). + **/ LinkifyIt.prototype.testSchemaAt = function testSchemaAt(text, schema, pos) { + // If not supported schema check requested - terminate + if (!this.__compiled__[schema.toLowerCase()]) { + return 0; + } + return this.__compiled__[schema.toLowerCase()].validate(text, pos, this); + }; + /** + * LinkifyIt#match(text) -> Array|null + * + * Returns array of found link descriptions or `null` on fail. We strongly + * recommend to use [[LinkifyIt#test]] first, for best speed. + * + * ##### Result match description + * + * - __schema__ - link schema, can be empty for fuzzy links, or `//` for + * protocol-neutral links. + * - __index__ - offset of matched text + * - __lastIndex__ - index of next char after mathch end + * - __raw__ - matched text + * - __text__ - normalized text + * - __url__ - link, generated from matched text + **/ LinkifyIt.prototype.match = function match(text) { + var shift = 0, result = []; + // Try to take previous element from cache, if .test() called before + if (this.__index__ >= 0 && this.__text_cache__ === text) { + result.push(createMatch(this, shift)); + shift = this.__last_index__; + } + // Cut head if cache was used + var tail = shift ? text.slice(shift) : text; + // Scan string until end reached + while (this.test(tail)) { + result.push(createMatch(this, shift)); + tail = tail.slice(this.__last_index__); + shift += this.__last_index__; + } + if (result.length) { + return result; + } + return null; + }; + /** + * LinkifyIt#matchAtStart(text) -> Match|null + * + * Returns fully-formed (not fuzzy) link if it starts at the beginning + * of the string, and null otherwise. + **/ LinkifyIt.prototype.matchAtStart = function matchAtStart(text) { + // Reset scan cache + this.__text_cache__ = text; + this.__index__ = -1; + if (!text.length) return null; + var m = this.re.schema_at_start.exec(text); + if (!m) return null; + var len = this.testSchemaAt(text, m[2], m[0].length); + if (!len) return null; + this.__schema__ = m[2]; + this.__index__ = m.index + m[1].length; + this.__last_index__ = m.index + m[0].length + len; + return createMatch(this, 0); + }; + /** chainable + * LinkifyIt#tlds(list [, keepOld]) -> this + * - list (Array): list of tlds + * - keepOld (Boolean): merge with current list if `true` (`false` by default) + * + * Load (or merge) new tlds list. Those are user for fuzzy links (without prefix) + * to avoid false positives. By default this algorythm used: + * + * - hostname with any 2-letter root zones are ok. + * - biz|com|edu|gov|net|org|pro|web|xxx|aero|asia|coop|info|museum|name|shop|рф + * are ok. + * - encoded (`xn--...`) root zones are ok. + * + * If list is replaced, then exact match for 2-chars root zones will be checked. + **/ LinkifyIt.prototype.tlds = function tlds(list, keepOld) { + list = Array.isArray(list) ? list : [ list ]; + if (!keepOld) { + this.__tlds__ = list.slice(); + this.__tlds_replaced__ = true; + compile(this); + return this; + } + this.__tlds__ = this.__tlds__.concat(list).sort().filter((function(el, idx, arr) { + return el !== arr[idx - 1]; + })).reverse(); + compile(this); + return this; + }; + /** + * LinkifyIt#normalize(match) + * + * Default normalizer (if schema does not define it's own). + **/ LinkifyIt.prototype.normalize = function normalize(match) { + // Do minimal possible changes by default. Need to collect feedback prior + // to move forward https://github.com/markdown-it/linkify-it/issues/1 + if (!match.schema) { + match.url = "http://" + match.url; + } + if (match.schema === "mailto:" && !/^mailto:/i.test(match.url)) { + match.url = "mailto:" + match.url; + } + }; + /** + * LinkifyIt#onCompile() + * + * Override to modify basic RegExp-s. + **/ LinkifyIt.prototype.onCompile = function onCompile() {}; + var linkifyIt = LinkifyIt; + /*! https://mths.be/punycode v1.4.1 by @mathias */ + /** Highest positive signed 32-bit float value */ var maxInt = 2147483647; + // aka. 0x7FFFFFFF or 2^31-1 + /** Bootstring parameters */ var base = 36; + var tMin = 1; + var tMax = 26; + var skew = 38; + var damp = 700; + var initialBias = 72; + var initialN = 128; + // 0x80 + var delimiter = "-"; + // '\x2D' + /** Regular expressions */ var regexPunycode = /^xn--/; + var regexNonASCII = /[^\x20-\x7E]/; + // unprintable ASCII chars + non-ASCII chars + var regexSeparators = /[\x2E\u3002\uFF0E\uFF61]/g; + // RFC 3490 separators + /** Error messages */ var errors = { + overflow: "Overflow: input needs wider integers to process", + "not-basic": "Illegal input >= 0x80 (not a basic code point)", + "invalid-input": "Invalid input" + }; + /** Convenience shortcuts */ var baseMinusTMin = base - tMin; + var floor = Math.floor; + var stringFromCharCode = String.fromCharCode; + /*--------------------------------------------------------------------------*/ + /** + * A generic error utility function. + * @private + * @param {String} type The error type. + * @returns {Error} Throws a `RangeError` with the applicable error message. + */ function error(type) { + throw new RangeError(errors[type]); + } + /** + * A generic `Array#map` utility function. + * @private + * @param {Array} array The array to iterate over. + * @param {Function} callback The function that gets called for every array + * item. + * @returns {Array} A new array of values returned by the callback function. + */ function map(array, fn) { + var length = array.length; + var result = []; + while (length--) { + result[length] = fn(array[length]); + } + return result; + } + /** + * A simple `Array#map`-like wrapper to work with domain name strings or email + * addresses. + * @private + * @param {String} domain The domain name or email address. + * @param {Function} callback The function that gets called for every + * character. + * @returns {Array} A new string of characters returned by the callback + * function. + */ function mapDomain(string, fn) { + var parts = string.split("@"); + var result = ""; + if (parts.length > 1) { + // In email addresses, only the domain name should be punycoded. Leave + // the local part (i.e. everything up to `@`) intact. + result = parts[0] + "@"; + string = parts[1]; + } + // Avoid `split(regex)` for IE8 compatibility. See #17. + string = string.replace(regexSeparators, "."); + var labels = string.split("."); + var encoded = map(labels, fn).join("."); + return result + encoded; + } + /** + * Creates an array containing the numeric code points of each Unicode + * character in the string. While JavaScript uses UCS-2 internally, + * this function will convert a pair of surrogate halves (each of which + * UCS-2 exposes as separate characters) into a single code point, + * matching UTF-16. + * @see `punycode.ucs2.encode` + * @see + * @memberOf punycode.ucs2 + * @name decode + * @param {String} string The Unicode input string (UCS-2). + * @returns {Array} The new array of code points. + */ function ucs2decode(string) { + var output = [], counter = 0, length = string.length, value, extra; + while (counter < length) { + value = string.charCodeAt(counter++); + if (value >= 55296 && value <= 56319 && counter < length) { + // high surrogate, and there is a next character + extra = string.charCodeAt(counter++); + if ((extra & 64512) == 56320) { + // low surrogate + output.push(((value & 1023) << 10) + (extra & 1023) + 65536); + } else { + // unmatched surrogate; only append this code unit, in case the next + // code unit is the high surrogate of a surrogate pair + output.push(value); + counter--; + } + } else { + output.push(value); + } + } + return output; + } + /** + * Creates a string based on an array of numeric code points. + * @see `punycode.ucs2.decode` + * @memberOf punycode.ucs2 + * @name encode + * @param {Array} codePoints The array of numeric code points. + * @returns {String} The new Unicode string (UCS-2). + */ function ucs2encode(array) { + return map(array, (function(value) { + var output = ""; + if (value > 65535) { + value -= 65536; + output += stringFromCharCode(value >>> 10 & 1023 | 55296); + value = 56320 | value & 1023; + } + output += stringFromCharCode(value); + return output; + })).join(""); + } + /** + * Converts a basic code point into a digit/integer. + * @see `digitToBasic()` + * @private + * @param {Number} codePoint The basic numeric code point value. + * @returns {Number} The numeric value of a basic code point (for use in + * representing integers) in the range `0` to `base - 1`, or `base` if + * the code point does not represent a value. + */ function basicToDigit(codePoint) { + if (codePoint - 48 < 10) { + return codePoint - 22; + } + if (codePoint - 65 < 26) { + return codePoint - 65; + } + if (codePoint - 97 < 26) { + return codePoint - 97; + } + return base; + } + /** + * Converts a digit/integer into a basic code point. + * @see `basicToDigit()` + * @private + * @param {Number} digit The numeric value of a basic code point. + * @returns {Number} The basic code point whose value (when used for + * representing integers) is `digit`, which needs to be in the range + * `0` to `base - 1`. If `flag` is non-zero, the uppercase form is + * used; else, the lowercase form is used. The behavior is undefined + * if `flag` is non-zero and `digit` has no uppercase form. + */ function digitToBasic(digit, flag) { + // 0..25 map to ASCII a..z or A..Z + // 26..35 map to ASCII 0..9 + return digit + 22 + 75 * (digit < 26) - ((flag != 0) << 5); + } + /** + * Bias adaptation function as per section 3.4 of RFC 3492. + * https://tools.ietf.org/html/rfc3492#section-3.4 + * @private + */ function adapt(delta, numPoints, firstTime) { + var k = 0; + delta = firstTime ? floor(delta / damp) : delta >> 1; + delta += floor(delta / numPoints); + for (;delta > baseMinusTMin * tMax >> 1; k += base) { + delta = floor(delta / baseMinusTMin); + } + return floor(k + (baseMinusTMin + 1) * delta / (delta + skew)); + } + /** + * Converts a Punycode string of ASCII-only symbols to a string of Unicode + * symbols. + * @memberOf punycode + * @param {String} input The Punycode string of ASCII-only symbols. + * @returns {String} The resulting string of Unicode symbols. + */ function decode(input) { + // Don't use UCS-2 + var output = [], inputLength = input.length, out, i = 0, n = initialN, bias = initialBias, basic, j, index, oldi, w, k, digit, t, + /** Cached calculation results */ + baseMinusT; + // Handle the basic code points: let `basic` be the number of input code + // points before the last delimiter, or `0` if there is none, then copy + // the first basic code points to the output. + basic = input.lastIndexOf(delimiter); + if (basic < 0) { + basic = 0; + } + for (j = 0; j < basic; ++j) { + // if it's not a basic code point + if (input.charCodeAt(j) >= 128) { + error("not-basic"); + } + output.push(input.charCodeAt(j)); + } + // Main decoding loop: start just after the last delimiter if any basic code + // points were copied; start at the beginning otherwise. + for (index = basic > 0 ? basic + 1 : 0; index < inputLength; ) { + // `index` is the index of the next character to be consumed. + // Decode a generalized variable-length integer into `delta`, + // which gets added to `i`. The overflow checking is easier + // if we increase `i` as we go, then subtract off its starting + // value at the end to obtain `delta`. + for (oldi = i, w = 1, k = base; ;k += base) { + if (index >= inputLength) { + error("invalid-input"); + } + digit = basicToDigit(input.charCodeAt(index++)); + if (digit >= base || digit > floor((maxInt - i) / w)) { + error("overflow"); + } + i += digit * w; + t = k <= bias ? tMin : k >= bias + tMax ? tMax : k - bias; + if (digit < t) { + break; + } + baseMinusT = base - t; + if (w > floor(maxInt / baseMinusT)) { + error("overflow"); + } + w *= baseMinusT; + } + out = output.length + 1; + bias = adapt(i - oldi, out, oldi == 0); + // `i` was supposed to wrap around from `out` to `0`, + // incrementing `n` each time, so we'll fix that now: + if (floor(i / out) > maxInt - n) { + error("overflow"); + } + n += floor(i / out); + i %= out; + // Insert `n` at position `i` of the output + output.splice(i++, 0, n); + } + return ucs2encode(output); + } + /** + * Converts a string of Unicode symbols (e.g. a domain name label) to a + * Punycode string of ASCII-only symbols. + * @memberOf punycode + * @param {String} input The string of Unicode symbols. + * @returns {String} The resulting Punycode string of ASCII-only symbols. + */ function encode(input) { + var n, delta, handledCPCount, basicLength, bias, j, m, q, k, t, currentValue, output = [], + /** `inputLength` will hold the number of code points in `input`. */ + inputLength, + /** Cached calculation results */ + handledCPCountPlusOne, baseMinusT, qMinusT; + // Convert the input in UCS-2 to Unicode + input = ucs2decode(input); + // Cache the length + inputLength = input.length; + // Initialize the state + n = initialN; + delta = 0; + bias = initialBias; + // Handle the basic code points + for (j = 0; j < inputLength; ++j) { + currentValue = input[j]; + if (currentValue < 128) { + output.push(stringFromCharCode(currentValue)); + } + } + handledCPCount = basicLength = output.length; + // `handledCPCount` is the number of code points that have been handled; + // `basicLength` is the number of basic code points. + // Finish the basic string - if it is not empty - with a delimiter + if (basicLength) { + output.push(delimiter); + } + // Main encoding loop: + while (handledCPCount < inputLength) { + // All non-basic code points < n have been handled already. Find the next + // larger one: + for (m = maxInt, j = 0; j < inputLength; ++j) { + currentValue = input[j]; + if (currentValue >= n && currentValue < m) { + m = currentValue; + } + } + // Increase `delta` enough to advance the decoder's state to , + // but guard against overflow + handledCPCountPlusOne = handledCPCount + 1; + if (m - n > floor((maxInt - delta) / handledCPCountPlusOne)) { + error("overflow"); + } + delta += (m - n) * handledCPCountPlusOne; + n = m; + for (j = 0; j < inputLength; ++j) { + currentValue = input[j]; + if (currentValue < n && ++delta > maxInt) { + error("overflow"); + } + if (currentValue == n) { + // Represent delta as a generalized variable-length integer + for (q = delta, k = base; ;k += base) { + t = k <= bias ? tMin : k >= bias + tMax ? tMax : k - bias; + if (q < t) { + break; + } + qMinusT = q - t; + baseMinusT = base - t; + output.push(stringFromCharCode(digitToBasic(t + qMinusT % baseMinusT, 0))); + q = floor(qMinusT / baseMinusT); + } + output.push(stringFromCharCode(digitToBasic(q, 0))); + bias = adapt(delta, handledCPCountPlusOne, handledCPCount == basicLength); + delta = 0; + ++handledCPCount; + } + } + ++delta; + ++n; + } + return output.join(""); + } + /** + * Converts a Punycode string representing a domain name or an email address + * to Unicode. Only the Punycoded parts of the input will be converted, i.e. + * it doesn't matter if you call it on a string that has already been + * converted to Unicode. + * @memberOf punycode + * @param {String} input The Punycoded domain name or email address to + * convert to Unicode. + * @returns {String} The Unicode representation of the given Punycode + * string. + */ function toUnicode(input) { + return mapDomain(input, (function(string) { + return regexPunycode.test(string) ? decode(string.slice(4).toLowerCase()) : string; + })); + } + /** + * Converts a Unicode string representing a domain name or an email address to + * Punycode. Only the non-ASCII parts of the domain name will be converted, + * i.e. it doesn't matter if you call it with a domain that's already in + * ASCII. + * @memberOf punycode + * @param {String} input The domain name or email address to convert, as a + * Unicode string. + * @returns {String} The Punycode representation of the given domain name or + * email address. + */ function toASCII(input) { + return mapDomain(input, (function(string) { + return regexNonASCII.test(string) ? "xn--" + encode(string) : string; + })); + } + var version = "1.4.1"; + /** + * An object of methods to convert from JavaScript's internal character + * representation (UCS-2) to Unicode code points, and back. + * @see + * @memberOf punycode + * @type Object + */ var ucs2 = { + decode: ucs2decode, + encode: ucs2encode + }; + var punycode$1 = { + version: version, + ucs2: ucs2, + toASCII: toASCII, + toUnicode: toUnicode, + encode: encode, + decode: decode + }; + var punycode$2 = Object.freeze({ + __proto__: null, + decode: decode, + encode: encode, + toUnicode: toUnicode, + toASCII: toASCII, + version: version, + ucs2: ucs2, + default: punycode$1 + }); + // markdown-it default options + var _default = { + options: { + html: false, + // Enable HTML tags in source + xhtmlOut: false, + // Use '/' to close single tags (
) + breaks: false, + // Convert '\n' in paragraphs into
+ langPrefix: "language-", + // CSS language prefix for fenced blocks + linkify: false, + // autoconvert URL-like texts to links + // Enable some language-neutral replacements + quotes beautification + typographer: false, + // Double + single quotes replacement pairs, when typographer enabled, + // and smartquotes on. Could be either a String or an Array. + // For example, you can use '«»„“' for Russian, '„“‚‘' for German, + // and ['«\xA0', '\xA0»', '‹\xA0', '\xA0›'] for French (including nbsp). + quotes: "\u201c\u201d\u2018\u2019", + /* “”‘’ */ + // Highlighter function. Should return escaped HTML, + // or '' if the source string is not changed and should be escaped externaly. + // If result starts with ) + breaks: false, + // Convert '\n' in paragraphs into
+ langPrefix: "language-", + // CSS language prefix for fenced blocks + linkify: false, + // autoconvert URL-like texts to links + // Enable some language-neutral replacements + quotes beautification + typographer: false, + // Double + single quotes replacement pairs, when typographer enabled, + // and smartquotes on. Could be either a String or an Array. + // For example, you can use '«»„“' for Russian, '„“‚‘' for German, + // and ['«\xA0', '\xA0»', '‹\xA0', '\xA0›'] for French (including nbsp). + quotes: "\u201c\u201d\u2018\u2019", + /* “”‘’ */ + // Highlighter function. Should return escaped HTML, + // or '' if the source string is not changed and should be escaped externaly. + // If result starts with ) + breaks: false, + // Convert '\n' in paragraphs into
+ langPrefix: "language-", + // CSS language prefix for fenced blocks + linkify: false, + // autoconvert URL-like texts to links + // Enable some language-neutral replacements + quotes beautification + typographer: false, + // Double + single quotes replacement pairs, when typographer enabled, + // and smartquotes on. Could be either a String or an Array. + // For example, you can use '«»„“' for Russian, '„“‚‘' for German, + // and ['«\xA0', '\xA0»', '‹\xA0', '\xA0›'] for French (including nbsp). + quotes: "\u201c\u201d\u2018\u2019", + /* “”‘’ */ + // Highlighter function. Should return escaped HTML, + // or '' if the source string is not changed and should be escaped externaly. + // If result starts with = 0) { + try { + parsed.hostname = punycode.toASCII(parsed.hostname); + } catch (er) {} + } + } + return mdurl.encode(mdurl.format(parsed)); + } + function normalizeLinkText(url) { + var parsed = mdurl.parse(url, true); + if (parsed.hostname) { + // Encode hostnames in urls like: + // `http://host/`, `https://host/`, `mailto:user@host`, `//host/` + // We don't encode unknown schemas, because it's likely that we encode + // something we shouldn't (e.g. `skype:name` treated as `skype:host`) + if (!parsed.protocol || RECODE_HOSTNAME_FOR.indexOf(parsed.protocol) >= 0) { + try { + parsed.hostname = punycode.toUnicode(parsed.hostname); + } catch (er) {} + } + } + // add '%' to exclude list because of https://github.com/markdown-it/markdown-it/issues/720 + return mdurl.decode(mdurl.format(parsed), mdurl.decode.defaultChars + "%"); + } + /** + * class MarkdownIt + * + * Main parser/renderer class. + * + * ##### Usage + * + * ```javascript + * // node.js, "classic" way: + * var MarkdownIt = require('markdown-it'), + * md = new MarkdownIt(); + * var result = md.render('# markdown-it rulezz!'); + * + * // node.js, the same, but with sugar: + * var md = require('markdown-it')(); + * var result = md.render('# markdown-it rulezz!'); + * + * // browser without AMD, added to "window" on script load + * // Note, there are no dash. + * var md = window.markdownit(); + * var result = md.render('# markdown-it rulezz!'); + * ``` + * + * Single line rendering, without paragraph wrap: + * + * ```javascript + * var md = require('markdown-it')(); + * var result = md.renderInline('__markdown-it__ rulezz!'); + * ``` + **/ + /** + * new MarkdownIt([presetName, options]) + * - presetName (String): optional, `commonmark` / `zero` + * - options (Object) + * + * Creates parser instanse with given config. Can be called without `new`. + * + * ##### presetName + * + * MarkdownIt provides named presets as a convenience to quickly + * enable/disable active syntax rules and options for common use cases. + * + * - ["commonmark"](https://github.com/markdown-it/markdown-it/blob/master/lib/presets/commonmark.js) - + * configures parser to strict [CommonMark](http://commonmark.org/) mode. + * - [default](https://github.com/markdown-it/markdown-it/blob/master/lib/presets/default.js) - + * similar to GFM, used when no preset name given. Enables all available rules, + * but still without html, typographer & autolinker. + * - ["zero"](https://github.com/markdown-it/markdown-it/blob/master/lib/presets/zero.js) - + * all rules disabled. Useful to quickly setup your config via `.enable()`. + * For example, when you need only `bold` and `italic` markup and nothing else. + * + * ##### options: + * + * - __html__ - `false`. Set `true` to enable HTML tags in source. Be careful! + * That's not safe! You may need external sanitizer to protect output from XSS. + * It's better to extend features via plugins, instead of enabling HTML. + * - __xhtmlOut__ - `false`. Set `true` to add '/' when closing single tags + * (`
`). This is needed only for full CommonMark compatibility. In real + * world you will need HTML output. + * - __breaks__ - `false`. Set `true` to convert `\n` in paragraphs into `
`. + * - __langPrefix__ - `language-`. CSS language class prefix for fenced blocks. + * Can be useful for external highlighters. + * - __linkify__ - `false`. Set `true` to autoconvert URL-like text to links. + * - __typographer__ - `false`. Set `true` to enable [some language-neutral + * replacement](https://github.com/markdown-it/markdown-it/blob/master/lib/rules_core/replacements.js) + + * quotes beautification (smartquotes). + * - __quotes__ - `“”‘’`, String or Array. Double + single quotes replacement + * pairs, when typographer enabled and smartquotes on. For example, you can + * use `'«»„“'` for Russian, `'„“‚‘'` for German, and + * `['«\xA0', '\xA0»', '‹\xA0', '\xA0›']` for French (including nbsp). + * - __highlight__ - `null`. Highlighter function for fenced code blocks. + * Highlighter `function (str, lang)` should return escaped HTML. It can also + * return empty string if the source was not changed and should be escaped + * externaly. If result starts with `): + * + * ```javascript + * var hljs = require('highlight.js') // https://highlightjs.org/ + * + * // Actual default values + * var md = require('markdown-it')({ + * highlight: function (str, lang) { + * if (lang && hljs.getLanguage(lang)) { + * try { + * return '
' +
+	 *                hljs.highlight(str, { language: lang, ignoreIllegals: true }).value +
+	 *                '
'; + * } catch (__) {} + * } + * + * return '
' + md.utils.escapeHtml(str) + '
'; + * } + * }); + * ``` + * + **/ function MarkdownIt(presetName, options) { + if (!(this instanceof MarkdownIt)) { + return new MarkdownIt(presetName, options); + } + if (!options) { + if (!utils.isString(presetName)) { + options = presetName || {}; + presetName = "default"; + } + } + /** + * MarkdownIt#inline -> ParserInline + * + * Instance of [[ParserInline]]. You may need it to add new rules when + * writing plugins. For simple rules control use [[MarkdownIt.disable]] and + * [[MarkdownIt.enable]]. + **/ this.inline = new parser_inline; + /** + * MarkdownIt#block -> ParserBlock + * + * Instance of [[ParserBlock]]. You may need it to add new rules when + * writing plugins. For simple rules control use [[MarkdownIt.disable]] and + * [[MarkdownIt.enable]]. + **/ this.block = new parser_block; + /** + * MarkdownIt#core -> Core + * + * Instance of [[Core]] chain executor. You may need it to add new rules when + * writing plugins. For simple rules control use [[MarkdownIt.disable]] and + * [[MarkdownIt.enable]]. + **/ this.core = new parser_core; + /** + * MarkdownIt#renderer -> Renderer + * + * Instance of [[Renderer]]. Use it to modify output look. Or to add rendering + * rules for new token types, generated by plugins. + * + * ##### Example + * + * ```javascript + * var md = require('markdown-it')(); + * + * function myToken(tokens, idx, options, env, self) { + * //... + * return result; + * }; + * + * md.renderer.rules['my_token'] = myToken + * ``` + * + * See [[Renderer]] docs and [source code](https://github.com/markdown-it/markdown-it/blob/master/lib/renderer.js). + **/ this.renderer = new renderer; + /** + * MarkdownIt#linkify -> LinkifyIt + * + * [linkify-it](https://github.com/markdown-it/linkify-it) instance. + * Used by [linkify](https://github.com/markdown-it/markdown-it/blob/master/lib/rules_core/linkify.js) + * rule. + **/ this.linkify = new linkifyIt; + /** + * MarkdownIt#validateLink(url) -> Boolean + * + * Link validation function. CommonMark allows too much in links. By default + * we disable `javascript:`, `vbscript:`, `file:` schemas, and almost all `data:...` schemas + * except some embedded image types. + * + * You can change this behaviour: + * + * ```javascript + * var md = require('markdown-it')(); + * // enable everything + * md.validateLink = function () { return true; } + * ``` + **/ this.validateLink = validateLink; + /** + * MarkdownIt#normalizeLink(url) -> String + * + * Function used to encode link url to a machine-readable format, + * which includes url-encoding, punycode, etc. + **/ this.normalizeLink = normalizeLink; + /** + * MarkdownIt#normalizeLinkText(url) -> String + * + * Function used to decode link url to a human-readable format` + **/ this.normalizeLinkText = normalizeLinkText; + // Expose utils & helpers for easy acces from plugins + /** + * MarkdownIt#utils -> utils + * + * Assorted utility functions, useful to write plugins. See details + * [here](https://github.com/markdown-it/markdown-it/blob/master/lib/common/utils.js). + **/ this.utils = utils; + /** + * MarkdownIt#helpers -> helpers + * + * Link components parser functions, useful to write plugins. See details + * [here](https://github.com/markdown-it/markdown-it/blob/master/lib/helpers). + **/ this.helpers = utils.assign({}, helpers); + this.options = {}; + this.configure(presetName); + if (options) { + this.set(options); + } + } + /** chainable + * MarkdownIt.set(options) + * + * Set parser options (in the same format as in constructor). Probably, you + * will never need it, but you can change options after constructor call. + * + * ##### Example + * + * ```javascript + * var md = require('markdown-it')() + * .set({ html: true, breaks: true }) + * .set({ typographer, true }); + * ``` + * + * __Note:__ To achieve the best possible performance, don't modify a + * `markdown-it` instance options on the fly. If you need multiple configurations + * it's best to create multiple instances and initialize each with separate + * config. + **/ MarkdownIt.prototype.set = function(options) { + utils.assign(this.options, options); + return this; + }; + /** chainable, internal + * MarkdownIt.configure(presets) + * + * Batch load of all options and compenent settings. This is internal method, + * and you probably will not need it. But if you will - see available presets + * and data structure [here](https://github.com/markdown-it/markdown-it/tree/master/lib/presets) + * + * We strongly recommend to use presets instead of direct config loads. That + * will give better compatibility with next versions. + **/ MarkdownIt.prototype.configure = function(presets) { + var self = this, presetName; + if (utils.isString(presets)) { + presetName = presets; + presets = config[presetName]; + if (!presets) { + throw new Error('Wrong `markdown-it` preset "' + presetName + '", check name'); + } + } + if (!presets) { + throw new Error("Wrong `markdown-it` preset, can't be empty"); + } + if (presets.options) { + self.set(presets.options); + } + if (presets.components) { + Object.keys(presets.components).forEach((function(name) { + if (presets.components[name].rules) { + self[name].ruler.enableOnly(presets.components[name].rules); + } + if (presets.components[name].rules2) { + self[name].ruler2.enableOnly(presets.components[name].rules2); + } + })); + } + return this; + }; + /** chainable + * MarkdownIt.enable(list, ignoreInvalid) + * - list (String|Array): rule name or list of rule names to enable + * - ignoreInvalid (Boolean): set `true` to ignore errors when rule not found. + * + * Enable list or rules. It will automatically find appropriate components, + * containing rules with given names. If rule not found, and `ignoreInvalid` + * not set - throws exception. + * + * ##### Example + * + * ```javascript + * var md = require('markdown-it')() + * .enable(['sub', 'sup']) + * .disable('smartquotes'); + * ``` + **/ MarkdownIt.prototype.enable = function(list, ignoreInvalid) { + var result = []; + if (!Array.isArray(list)) { + list = [ list ]; + } + [ "core", "block", "inline" ].forEach((function(chain) { + result = result.concat(this[chain].ruler.enable(list, true)); + }), this); + result = result.concat(this.inline.ruler2.enable(list, true)); + var missed = list.filter((function(name) { + return result.indexOf(name) < 0; + })); + if (missed.length && !ignoreInvalid) { + throw new Error("MarkdownIt. Failed to enable unknown rule(s): " + missed); + } + return this; + }; + /** chainable + * MarkdownIt.disable(list, ignoreInvalid) + * - list (String|Array): rule name or list of rule names to disable. + * - ignoreInvalid (Boolean): set `true` to ignore errors when rule not found. + * + * The same as [[MarkdownIt.enable]], but turn specified rules off. + **/ MarkdownIt.prototype.disable = function(list, ignoreInvalid) { + var result = []; + if (!Array.isArray(list)) { + list = [ list ]; + } + [ "core", "block", "inline" ].forEach((function(chain) { + result = result.concat(this[chain].ruler.disable(list, true)); + }), this); + result = result.concat(this.inline.ruler2.disable(list, true)); + var missed = list.filter((function(name) { + return result.indexOf(name) < 0; + })); + if (missed.length && !ignoreInvalid) { + throw new Error("MarkdownIt. Failed to disable unknown rule(s): " + missed); + } + return this; + }; + /** chainable + * MarkdownIt.use(plugin, params) + * + * Load specified plugin with given params into current parser instance. + * It's just a sugar to call `plugin(md, params)` with curring. + * + * ##### Example + * + * ```javascript + * var iterator = require('markdown-it-for-inline'); + * var md = require('markdown-it')() + * .use(iterator, 'foo_replace', 'text', function (tokens, idx) { + * tokens[idx].content = tokens[idx].content.replace(/foo/g, 'bar'); + * }); + * ``` + **/ MarkdownIt.prototype.use = function(plugin /*, params, ... */) { + var args = [ this ].concat(Array.prototype.slice.call(arguments, 1)); + plugin.apply(plugin, args); + return this; + }; + /** internal + * MarkdownIt.parse(src, env) -> Array + * - src (String): source string + * - env (Object): environment sandbox + * + * Parse input string and return list of block tokens (special token type + * "inline" will contain list of inline tokens). You should not call this + * method directly, until you write custom renderer (for example, to produce + * AST). + * + * `env` is used to pass data between "distributed" rules and return additional + * metadata like reference info, needed for the renderer. It also can be used to + * inject data in specific cases. Usually, you will be ok to pass `{}`, + * and then pass updated object to renderer. + **/ MarkdownIt.prototype.parse = function(src, env) { + if (typeof src !== "string") { + throw new Error("Input data should be a String"); + } + var state = new this.core.State(src, this, env); + this.core.process(state); + return state.tokens; + }; + /** + * MarkdownIt.render(src [, env]) -> String + * - src (String): source string + * - env (Object): environment sandbox + * + * Render markdown string into html. It does all magic for you :). + * + * `env` can be used to inject additional metadata (`{}` by default). + * But you will not need it with high probability. See also comment + * in [[MarkdownIt.parse]]. + **/ MarkdownIt.prototype.render = function(src, env) { + env = env || {}; + return this.renderer.render(this.parse(src, env), this.options, env); + }; + /** internal + * MarkdownIt.parseInline(src, env) -> Array + * - src (String): source string + * - env (Object): environment sandbox + * + * The same as [[MarkdownIt.parse]] but skip all block rules. It returns the + * block tokens list with the single `inline` element, containing parsed inline + * tokens in `children` property. Also updates `env` object. + **/ MarkdownIt.prototype.parseInline = function(src, env) { + var state = new this.core.State(src, this, env); + state.inlineMode = true; + this.core.process(state); + return state.tokens; + }; + /** + * MarkdownIt.renderInline(src [, env]) -> String + * - src (String): source string + * - env (Object): environment sandbox + * + * Similar to [[MarkdownIt.render]] but for single paragraph content. Result + * will NOT be wrapped into `

` tags. + **/ MarkdownIt.prototype.renderInline = function(src, env) { + env = env || {}; + return this.renderer.render(this.parseInline(src, env), this.options, env); + }; + var lib = MarkdownIt; + var markdownIt = lib; + return markdownIt; +})); diff --git a/src/interface/desktop/assets/org.min.js b/src/interface/desktop/assets/org.min.js new file mode 100644 index 00000000..a919b280 --- /dev/null +++ b/src/interface/desktop/assets/org.min.js @@ -0,0 +1,1823 @@ +// Generated by export.rb at Sat Feb 21 07:44:29 UTC 2015 +/* + Copyright (c) 2014 Masafumi Oyamada + + Permission is hereby granted, free of charge, to any person obtaining a copy + of this software and associated documentation files (the "Software"), to deal + in the Software without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be included in + all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + THE SOFTWARE. +*/ + +var Org = (function () { + var exports = {}; + + // ------------------------------------------------------------ + // Syntax + // ------------------------------------------------------------ + + var Syntax = { + rules: {}, + + define: function (name, syntax) { + this.rules[name] = syntax; + var methodName = "is" + name.substring(0, 1).toUpperCase() + name.substring(1); + this[methodName] = function (line) { + return this.rules[name].exec(line); + }; + } + }; + + Syntax.define("header", /^(\*+)\s+(.*)$/); // m[1] => level, m[2] => content + Syntax.define("preformatted", /^(\s*):(?: (.*)$|$)/); // m[1] => indentation, m[2] => content + Syntax.define("unorderedListElement", /^(\s*)(?:-|\+|\s+\*)\s+(.*)$/); // m[1] => indentation, m[2] => content + Syntax.define("orderedListElement", /^(\s*)(\d+)(?:\.|\))\s+(.*)$/); // m[1] => indentation, m[2] => number, m[3] => content + Syntax.define("tableSeparator", /^(\s*)\|((?:\+|-)*?)\|?$/); // m[1] => indentation, m[2] => content + Syntax.define("tableRow", /^(\s*)\|(.*?)\|?$/); // m[1] => indentation, m[2] => content + Syntax.define("blank", /^$/); + Syntax.define("horizontalRule", /^(\s*)-{5,}$/); // + Syntax.define("directive", /^(\s*)#\+(?:(begin|end)_)?(.*)$/i); // m[1] => indentation, m[2] => type, m[3] => content + Syntax.define("comment", /^(\s*)#(.*)$/); + Syntax.define("line", /^(\s*)(.*)$/); + + const propertyDrawer = /:PROPERTIES:(.*):END:/g; + + // ------------------------------------------------------------ + // Token + // ------------------------------------------------------------ + + function Token() { + } + + Token.prototype = { + isListElement: function () { + return this.type === Lexer.tokens.orderedListElement || + this.type === Lexer.tokens.unorderedListElement; + }, + + isTableElement: function () { + return this.type === Lexer.tokens.tableSeparator || + this.type === Lexer.tokens.tableRow; + } + }; + + // ------------------------------------------------------------ + // Lexer + // ------------------------------------------------------------ + + function Lexer(stream) { + this.stream = stream; + this.tokenStack = []; + } + + Lexer.prototype = { + tokenize: function (line) { + var token = new Token(); + token.fromLineNumber = this.stream.lineNumber; + + if (Syntax.isHeader(line)) { + token.type = Lexer.tokens.header; + token.indentation = 0; + token.content = RegExp.$2; + // specific + token.level = 2; + } else if (Syntax.isPreformatted(line)) { + token.type = Lexer.tokens.preformatted; + token.indentation = RegExp.$1.length; + token.content = RegExp.$2; + } else if (Syntax.isUnorderedListElement(line)) { + token.type = Lexer.tokens.unorderedListElement; + token.indentation = RegExp.$1.length; + token.content = RegExp.$2; + } else if (Syntax.isOrderedListElement(line)) { + token.type = Lexer.tokens.orderedListElement; + token.indentation = RegExp.$1.length; + token.content = RegExp.$3; + // specific + token.number = RegExp.$2; + } else if (Syntax.isTableSeparator(line)) { + token.type = Lexer.tokens.tableSeparator; + token.indentation = RegExp.$1.length; + token.content = RegExp.$2; + } else if (Syntax.isTableRow(line)) { + token.type = Lexer.tokens.tableRow; + token.indentation = RegExp.$1.length; + token.content = RegExp.$2; + } else if (Syntax.isBlank(line)) { + token.type = Lexer.tokens.blank; + token.indentation = 0; + token.content = null; + } else if (Syntax.isHorizontalRule(line)) { + token.type = Lexer.tokens.horizontalRule; + token.indentation = RegExp.$1.length; + token.content = null; + } else if (Syntax.isDirective(line)) { + token.type = Lexer.tokens.directive; + token.indentation = RegExp.$1.length; + token.content = RegExp.$3; + // decide directive type (begin, end or oneshot) + var directiveTypeString = RegExp.$2; + if (/^begin/i.test(directiveTypeString)) + token.beginDirective = true; + else if (/^end/i.test(directiveTypeString)) + token.endDirective = true; + else + token.oneshotDirective = true; + } else if (Syntax.isComment(line)) { + token.type = Lexer.tokens.comment; + token.indentation = RegExp.$1.length; + token.content = RegExp.$2; + } else if (Syntax.isLine(line)) { + token.type = Lexer.tokens.line; + token.indentation = RegExp.$1.length; + token.content = RegExp.$2; + } else { + throw new Error("SyntaxError: Unknown line: " + line); + } + + return token; + }, + + pushToken: function (token) { + this.tokenStack.push(token); + }, + + pushDummyTokenByType: function (type) { + var token = new Token(); + token.type = type; + this.tokenStack.push(token); + }, + + peekStackedToken: function () { + return this.tokenStack.length > 0 ? + this.tokenStack[this.tokenStack.length - 1] : null; + }, + + getStackedToken: function () { + return this.tokenStack.length > 0 ? + this.tokenStack.pop() : null; + }, + + peekNextToken: function () { + return this.peekStackedToken() || + this.tokenize(this.stream.peekNextLine()); + }, + + getNextToken: function () { + return this.getStackedToken() || + this.tokenize(this.stream.getNextLine()); + }, + + hasNext: function () { + return this.stream.hasNext(); + }, + + getLineNumber: function () { + return this.stream.lineNumber; + } + }; + + Lexer.tokens = {}; + [ + "header", + "orderedListElement", + "unorderedListElement", + "tableRow", + "tableSeparator", + "preformatted", + "line", + "horizontalRule", + "blank", + "directive", + "comment" + ].forEach(function (tokenName, i) { + Lexer.tokens[tokenName] = i; + }); + + // ------------------------------------------------------------ + // Exports + // ------------------------------------------------------------ + + if (typeof exports !== "undefined") + exports.Lexer = Lexer; + + function PrototypeNode(type, children) { + this.type = type; + this.children = []; + + if (children) { + for (var i = 0, len = children.length; i < len; ++i) { + this.appendChild(children[i]); + } + } + } + PrototypeNode.prototype = { + previousSibling: null, + parent: null, + get firstChild() { + return this.children.length < 1 ? + null : this.children[0]; + }, + get lastChild() { + return this.children.length < 1 ? + null : this.children[this.children.length - 1]; + }, + appendChild: function (newChild) { + var previousSibling = this.children.length < 1 ? + null : this.lastChild; + this.children.push(newChild); + newChild.previousSibling = previousSibling; + newChild.parent = this; + }, + toString: function () { + var string = "<" + this.type + ">"; + + if (typeof this.value !== "undefined") { + string += " " + this.value; + } else if (this.children) { + string += "\n" + this.children.map(function (child, idx) { + return "#" + idx + " " + child.toString(); + }).join("\n").split("\n").map(function (line) { + return " " + line; + }).join("\n"); + } + + return string; + } + }; + + var Node = { + types: {}, + + define: function (name, postProcess) { + this.types[name] = name; + + var methodName = "create" + name.substring(0, 1).toUpperCase() + name.substring(1); + var postProcessGiven = typeof postProcess === "function"; + + this[methodName] = function (children, options) { + var node = new PrototypeNode(name, children); + + if (postProcessGiven) + postProcess(node, options || {}); + + return node; + }; + } + }; + + Node.define("text", function (node, options) { + node.value = options.value; + }); + Node.define("header", function (node, options) { + node.level = options.level; + }); + Node.define("orderedList"); + Node.define("unorderedList"); + Node.define("definitionList"); + Node.define("listElement"); + Node.define("paragraph"); + Node.define("preformatted"); + Node.define("table"); + Node.define("tableRow"); + Node.define("tableCell"); + Node.define("horizontalRule"); + Node.define("directive"); + + // Inline + Node.define("inlineContainer"); + + Node.define("bold"); + Node.define("italic"); + Node.define("underline"); + Node.define("code"); + Node.define("verbatim"); + Node.define("dashed"); + Node.define("link", function (node, options) { + node.src = options.src; + }); + + if (typeof exports !== "undefined") + exports.Node = Node; + + function Stream(sequence) { + this.sequences = sequence.split(/\r?\n/); + this.totalLines = this.sequences.length; + this.lineNumber = 0; + } + + Stream.prototype.peekNextLine = function () { + return this.hasNext() ? this.sequences[this.lineNumber] : null; + }; + + Stream.prototype.getNextLine = function () { + return this.hasNext() ? this.sequences[this.lineNumber++] : null; + }; + + Stream.prototype.hasNext = function () { + return this.lineNumber < this.totalLines; + }; + + if (typeof exports !== "undefined") { + exports.Stream = Stream; + } + + // var Stream = require("./stream.js").Stream; + // var Lexer = require("./lexer.js").Lexer; + // var Node = require("./node.js").Node; + + function Parser() { + this.inlineParser = new InlineParser(); + } + + Parser.parseStream = function (stream, options) { + var parser = new Parser(); + parser.initStatus(stream, options); + parser.parseNodes(); + return parser.nodes; + }; + + Parser.prototype = { + initStatus: function (stream, options) { + if (typeof stream === "string") + stream = new Stream(stream); + this.lexer = new Lexer(stream); + this.nodes = []; + this.options = { + toc: true, + num: true, + "^": "{}", + multilineCell: false + }; + // Override option values + if (options && typeof options === "object") { + for (var key in options) { + this.options[key] = options[key]; + } + } + this.document = { + options: this.options, + directiveValues: {}, + convert: function (ConverterClass, exportOptions) { + var converter = new ConverterClass(this, exportOptions); + return converter.result; + } + }; + }, + + parse: function (stream, options) { + this.initStatus(stream, options); + this.parseDocument(); + this.document.nodes = this.nodes; + return this.document; + }, + + createErrorReport: function (message) { + return new Error(message + " at line " + this.lexer.getLineNumber()); + }, + + skipBlank: function () { + var blankToken = null; + while (this.lexer.peekNextToken().type === Lexer.tokens.blank) + blankToken = this.lexer.getNextToken(); + return blankToken; + }, + + setNodeOriginFromToken: function (node, token) { + node.fromLineNumber = token.fromLineNumber; + return node; + }, + + appendNode: function (newNode) { + var previousSibling = this.nodes.length > 0 ? this.nodes[this.nodes.length - 1] : null; + this.nodes.push(newNode); + newNode.previousSibling = previousSibling; + }, + + // ------------------------------------------------------------ + // ::= * + // ------------------------------------------------------------ + + parseDocument: function () { + this.parseTitle(); + this.parseNodes(); + }, + + parseNodes: function () { + while (this.lexer.hasNext()) { + var element = this.parseElement(); + if (element) this.appendNode(element); + } + }, + + parseTitle: function () { + this.skipBlank(); + + if (this.lexer.hasNext() && + this.lexer.peekNextToken().type === Lexer.tokens.line) + this.document.title = this.createTextNode(this.lexer.getNextToken().content); + else + this.document.title = null; + + this.lexer.pushDummyTokenByType(Lexer.tokens.blank); + }, + + // ------------------------------------------------------------ + // ::= (

| + // | | + // | )* + // ------------------------------------------------------------ + + parseElement: function () { + var element = null; + + switch (this.lexer.peekNextToken().type) { + case Lexer.tokens.header: + element = this.parseHeader(); + break; + case Lexer.tokens.preformatted: + element = this.parsePreformatted(); + break; + case Lexer.tokens.orderedListElement: + case Lexer.tokens.unorderedListElement: + element = this.parseList(); + break; + case Lexer.tokens.line: + element = this.parseText(); + break; + case Lexer.tokens.tableRow: + case Lexer.tokens.tableSeparator: + element = this.parseTable(); + break; + case Lexer.tokens.blank: + this.skipBlank(); + if (this.lexer.hasNext()) { + if (this.lexer.peekNextToken().type === Lexer.tokens.line) + element = this.parseParagraph(); + else + element = this.parseElement(); + } + break; + case Lexer.tokens.horizontalRule: + this.lexer.getNextToken(); + element = Node.createHorizontalRule(); + break; + case Lexer.tokens.directive: + element = this.parseDirective(); + break; + case Lexer.tokens.comment: + // Skip + this.lexer.getNextToken(); + break; + default: + throw this.createErrorReport("Unhandled token: " + this.lexer.peekNextToken().type); + } + + return element; + }, + + parseElementBesidesDirectiveEnd: function () { + try { + // Temporary, override the definition of `parseElement` + this.parseElement = this.parseElementBesidesDirectiveEndBody; + return this.parseElement(); + } finally { + this.parseElement = this.originalParseElement; + } + }, + + parseElementBesidesDirectiveEndBody: function () { + if (this.lexer.peekNextToken().type === Lexer.tokens.directive && + this.lexer.peekNextToken().endDirective) { + return null; + } + + return this.originalParseElement(); + }, + + // ------------------------------------------------------------ + //
+ // + // : preformatted + // : block + // ------------------------------------------------------------ + + parseHeader: function () { + var headerToken = this.lexer.getNextToken(); + var header = Node.createHeader([ + this.createTextNode(headerToken.content) // TODO: Parse inline markups + ], { level: headerToken.level }); + this.setNodeOriginFromToken(header, headerToken); + + return header; + }, + + // ------------------------------------------------------------ + // + // + // : preformatted + // : block + // ------------------------------------------------------------ + + parsePreformatted: function () { + var preformattedFirstToken = this.lexer.peekNextToken(); + var preformatted = Node.createPreformatted([]); + this.setNodeOriginFromToken(preformatted, preformattedFirstToken); + + var textContents = []; + + while (this.lexer.hasNext()) { + var token = this.lexer.peekNextToken(); + if (token.type !== Lexer.tokens.preformatted || + token.indentation < preformattedFirstToken.indentation) + break; + this.lexer.getNextToken(); + textContents.push(token.content); + } + + preformatted.appendChild(this.createTextNode(textContents.join("\n"), true /* no emphasis */)); + + return preformatted; + }, + + // ------------------------------------------------------------ + // + // + // - foo + // 1. bar + // 2. baz + // ------------------------------------------------------------ + + // XXX: not consider codes (e.g., =Foo::Bar=) + definitionPattern: /^(.*?) :: *(.*)$/, + + parseList: function () { + var rootToken = this.lexer.peekNextToken(); + var list; + var isDefinitionList = false; + + if (this.definitionPattern.test(rootToken.content)) { + list = Node.createDefinitionList([]); + isDefinitionList = true; + } else { + list = rootToken.type === Lexer.tokens.unorderedListElement ? + Node.createUnorderedList([]) : Node.createOrderedList([]); + } + this.setNodeOriginFromToken(list, rootToken); + + while (this.lexer.hasNext()) { + var nextToken = this.lexer.peekNextToken(); + if (!nextToken.isListElement() || nextToken.indentation !== rootToken.indentation) + break; + list.appendChild(this.parseListElement(rootToken.indentation, isDefinitionList)); + } + + return list; + }, + + unknownDefinitionTerm: "???", + + parseListElement: function (rootIndentation, isDefinitionList) { + var listElementToken = this.lexer.getNextToken(); + var listElement = Node.createListElement([]); + this.setNodeOriginFromToken(listElement, listElementToken); + + listElement.isDefinitionList = isDefinitionList; + + if (isDefinitionList) { + var match = this.definitionPattern.exec(listElementToken.content); + listElement.term = [ + this.createTextNode(match && match[1] ? match[1] : this.unknownDefinitionTerm) + ]; + listElement.appendChild(this.createTextNode(match ? match[2] : listElementToken.content)); + } else { + listElement.appendChild(this.createTextNode(listElementToken.content)); + } + + while (this.lexer.hasNext()) { + var blankToken = this.skipBlank(); + if (!this.lexer.hasNext()) + break; + + var notBlankNextToken = this.lexer.peekNextToken(); + if (blankToken && !notBlankNextToken.isListElement()) + this.lexer.pushToken(blankToken); // Recover blank token only when next line is not listElement. + // End of the list if hit less indented line or end of directive + if (notBlankNextToken.indentation <= rootIndentation || + (notBlankNextToken.type === Lexer.tokens.directive && notBlankNextToken.endDirective)) + break; + + var element = this.parseElement(); // recursive + if (element) + listElement.appendChild(element); + } + + return listElement; + }, + + // ------------------------------------------------------------ + //
::= + + // ------------------------------------------------------------ + + parseTable: function () { + var nextToken = this.lexer.peekNextToken(); + var table = Node.createTable([]); + this.setNodeOriginFromToken(table, nextToken); + var sawSeparator = false; + + var allowMultilineCell = nextToken.type === Lexer.tokens.tableSeparator && this.options.multilineCell; + + while (this.lexer.hasNext() && + (nextToken = this.lexer.peekNextToken()).isTableElement()) { + if (nextToken.type === Lexer.tokens.tableRow) { + var tableRow = this.parseTableRow(allowMultilineCell); + table.appendChild(tableRow); + } else { + // Lexer.tokens.tableSeparator + sawSeparator = true; + this.lexer.getNextToken(); + } + } + + if (sawSeparator && table.children.length) { + table.children[0].children.forEach(function (cell) { + cell.isHeader = true; + }); + } + + return table; + }, + + // ------------------------------------------------------------ + // ::= + + // ------------------------------------------------------------ + + parseTableRow: function (allowMultilineCell) { + var tableRowTokens = []; + + while (this.lexer.peekNextToken().type === Lexer.tokens.tableRow) { + tableRowTokens.push(this.lexer.getNextToken()); + if (!allowMultilineCell) { + break; + } + } + + if (!tableRowTokens.length) { + throw this.createErrorReport("Expected table row"); + } + + var firstTableRowToken = tableRowTokens.shift(); + var tableCellTexts = firstTableRowToken.content.split("|"); + + tableRowTokens.forEach(function (rowToken) { + rowToken.content.split("|").forEach(function (cellText, cellIdx) { + tableCellTexts[cellIdx] = (tableCellTexts[cellIdx] || "") + "\n" + cellText; + }); + }); + + // TODO: Prepare two pathes: (1) + var tableCells = tableCellTexts.map( + // TODO: consider '|' escape? + function (text) { + return Node.createTableCell(Parser.parseStream(text)); + }, this); + + return this.setNodeOriginFromToken(Node.createTableRow(tableCells), firstTableRowToken); + }, + + // ------------------------------------------------------------ + // ::= "#+.*" + // ------------------------------------------------------------ + + parseDirective: function () { + var directiveToken = this.lexer.getNextToken(); + var directiveNode = this.createDirectiveNodeFromToken(directiveToken); + + if (directiveToken.endDirective) + throw this.createErrorReport("Unmatched 'end' directive for " + directiveNode.directiveName); + + if (directiveToken.oneshotDirective) { + this.interpretDirective(directiveNode); + return directiveNode; + } + + if (!directiveToken.beginDirective) + throw this.createErrorReport("Invalid directive " + directiveNode.directiveName); + + // Parse begin ~ end + directiveNode.children = []; + if (this.isVerbatimDirective(directiveNode)) + return this.parseDirectiveBlockVerbatim(directiveNode); + else + return this.parseDirectiveBlock(directiveNode); + }, + + createDirectiveNodeFromToken: function (directiveToken) { + var matched = /^[ ]*([^ ]*)[ ]*(.*)[ ]*$/.exec(directiveToken.content); + + var directiveNode = Node.createDirective(null); + this.setNodeOriginFromToken(directiveNode, directiveToken); + directiveNode.directiveName = matched[1].toLowerCase(); + directiveNode.directiveArguments = this.parseDirectiveArguments(matched[2]); + directiveNode.directiveOptions = this.parseDirectiveOptions(matched[2]); + directiveNode.directiveRawValue = matched[2]; + + return directiveNode; + }, + + isVerbatimDirective: function (directiveNode) { + var directiveName = directiveNode.directiveName; + return directiveName === "src" || directiveName === "example" || directiveName === "html"; + }, + + parseDirectiveBlock: function (directiveNode, verbatim) { + this.lexer.pushDummyTokenByType(Lexer.tokens.blank); + + while (this.lexer.hasNext()) { + var nextToken = this.lexer.peekNextToken(); + if (nextToken.type === Lexer.tokens.directive && + nextToken.endDirective && + this.createDirectiveNodeFromToken(nextToken).directiveName === directiveNode.directiveName) { + // Close directive + this.lexer.getNextToken(); + return directiveNode; + } + var element = this.parseElementBesidesDirectiveEnd(); + if (element) + directiveNode.appendChild(element); + } + + throw this.createErrorReport("Unclosed directive " + directiveNode.directiveName); + }, + + parseDirectiveBlockVerbatim: function (directiveNode) { + var textContent = []; + + while (this.lexer.hasNext()) { + var nextToken = this.lexer.peekNextToken(); + if (nextToken.type === Lexer.tokens.directive && + nextToken.endDirective && + this.createDirectiveNodeFromToken(nextToken).directiveName === directiveNode.directiveName) { + this.lexer.getNextToken(); + directiveNode.appendChild(this.createTextNode(textContent.join("\n"), true)); + return directiveNode; + } + textContent.push(this.lexer.stream.getNextLine()); + } + + throw this.createErrorReport("Unclosed directive " + directiveNode.directiveName); + }, + + parseDirectiveArguments: function (parameters) { + return parameters.split(/[ ]+/).filter(function (param) { + return param.length && param[0] !== "-"; + }); + }, + + parseDirectiveOptions: function (parameters) { + return parameters.split(/[ ]+/).filter(function (param) { + return param.length && param[0] === "-"; + }); + }, + + interpretDirective: function (directiveNode) { + // http://orgmode.org/manual/Export-options.html + switch (directiveNode.directiveName) { + case "options:": + this.interpretOptionDirective(directiveNode); + break; + case "title:": + this.document.title = directiveNode.directiveRawValue; + break; + case "author:": + this.document.author = directiveNode.directiveRawValue; + break; + case "email:": + this.document.email = directiveNode.directiveRawValue; + break; + default: + this.document.directiveValues[directiveNode.directiveName] = directiveNode.directiveRawValue; + break; + } + }, + + interpretOptionDirective: function (optionDirectiveNode) { + optionDirectiveNode.directiveArguments.forEach(function (pairString) { + var pair = pairString.split(":"); + this.options[pair[0]] = this.convertLispyValue(pair[1]); + }, this); + }, + + convertLispyValue: function (lispyValue) { + switch (lispyValue) { + case "t": + return true; + case "nil": + return false; + default: + if (/^[0-9]+$/.test(lispyValue)) + return parseInt(lispyValue); + return lispyValue; + } + }, + + // ------------------------------------------------------------ + // ::= * + // ------------------------------------------------------------ + + parseParagraph: function () { + var paragraphFisrtToken = this.lexer.peekNextToken(); + var paragraph = Node.createParagraph([]); + this.setNodeOriginFromToken(paragraph, paragraphFisrtToken); + + var textContents = []; + + while (this.lexer.hasNext()) { + var nextToken = this.lexer.peekNextToken(); + if (nextToken.type !== Lexer.tokens.line + || nextToken.indentation < paragraphFisrtToken.indentation) + break; + this.lexer.getNextToken(); + textContents.push(nextToken.content); + } + + paragraph.appendChild(this.createTextNode(textContents.join("\n"))); + + return paragraph; + }, + + parseText: function (noEmphasis) { + var lineToken = this.lexer.getNextToken(); + return this.createTextNode(lineToken.content, noEmphasis); + }, + + // ------------------------------------------------------------ + // (DOM Like) + // ------------------------------------------------------------ + + createTextNode: function (text, noEmphasis) { + return noEmphasis ? Node.createText(null, { value: text }) + : this.inlineParser.parseEmphasis(text); + } + }; + Parser.prototype.originalParseElement = Parser.prototype.parseElement; + + // ------------------------------------------------------------ + // Parser for Inline Elements + // + // @refs org-emphasis-regexp-components + // ------------------------------------------------------------ + + function InlineParser() { + this.preEmphasis = " \t\\('\""; + this.postEmphasis = "- \t.,:!?;'\"\\)"; + this.borderForbidden = " \t\r\n,\"'"; + this.bodyRegexp = "[\\s\\S]*?"; + this.markers = "*/_=~+"; + + this.emphasisPattern = this.buildEmphasisPattern(); + this.linkPattern = /\[\[([^\]]*)\](?:\[([^\]]*)\])?\]/g; // \1 => link, \2 => text + } + + InlineParser.prototype = { + parseEmphasis: function (text) { + var emphasisPattern = this.emphasisPattern; + emphasisPattern.lastIndex = 0; + + var result = [], + match, + previousLast = 0, + savedLastIndex; + + while ((match = emphasisPattern.exec(text))) { + var whole = match[0]; + var pre = match[1]; + var marker = match[2]; + var body = match[3]; + var post = match[4]; + + { + // parse links + var matchBegin = emphasisPattern.lastIndex - whole.length; + var beforeContent = text.substring(previousLast, matchBegin + pre.length); + savedLastIndex = emphasisPattern.lastIndex; + result.push(this.parseLink(beforeContent)); + emphasisPattern.lastIndex = savedLastIndex; + } + + var bodyNode = [Node.createText(null, { value: body })]; + var bodyContainer = this.emphasizeElementByMarker(bodyNode, marker); + result.push(bodyContainer); + + previousLast = emphasisPattern.lastIndex - post.length; + } + + if (emphasisPattern.lastIndex === 0 || + emphasisPattern.lastIndex !== text.length - 1) + result.push(this.parseLink(text.substring(previousLast))); + + if (result.length === 1) { + // Avoid duplicated inline container wrapping + return result[0]; + } else { + return Node.createInlineContainer(result); + } + }, + + depth: 0, + parseLink: function (text) { + var linkPattern = this.linkPattern; + linkPattern.lastIndex = 0; + + var match, + result = [], + previousLast = 0, + savedLastIndex; + + while ((match = linkPattern.exec(text))) { + var whole = match[0]; + var src = match[1]; + var title = match[2]; + + // parse before content + var matchBegin = linkPattern.lastIndex - whole.length; + var beforeContent = text.substring(previousLast, matchBegin); + result.push(Node.createText(null, { value: beforeContent })); + + // parse link + var link = Node.createLink([]); + link.src = src; + if (title) { + savedLastIndex = linkPattern.lastIndex; + link.appendChild(this.parseEmphasis(title)); + linkPattern.lastIndex = savedLastIndex; + } else { + link.appendChild(Node.createText(null, { value: src })); + } + result.push(link); + + previousLast = linkPattern.lastIndex; + } + + if (linkPattern.lastIndex === 0 || + linkPattern.lastIndex !== text.length - 1) + result.push(Node.createText(null, { value: text.substring(previousLast) })); + + return Node.createInlineContainer(result); + }, + + emphasizeElementByMarker: function (element, marker) { + switch (marker) { + case "*": + return Node.createBold(element); + case "/": + return Node.createItalic(element); + case "_": + return Node.createUnderline(element); + case "=": + case "~": + return Node.createCode(element); + case "+": + return Node.createDashed(element); + } + }, + + buildEmphasisPattern: function () { + return new RegExp( + "([" + this.preEmphasis + "]|^|\r?\n)" + // \1 => pre + "([" + this.markers + "])" + // \2 => marker + "([^" + this.borderForbidden + "]|" + // \3 => body + "[^" + this.borderForbidden + "]" + + this.bodyRegexp + + "[^" + this.borderForbidden + "])" + + "\\2" + + "([" + this.postEmphasis +"]|$|\r?\n)", // \4 => post + // flags + "g" + ); + } + }; + + if (typeof exports !== "undefined") { + exports.Parser = Parser; + exports.InlineParser = InlineParser; + } + + // var Node = require("../node.js").Node; + + function Converter() { + } + + Converter.prototype = { + exportOptions: { + headerOffset: 1, + exportFromLineNumber: false, + suppressSubScriptHandling: false, + suppressAutoLink: false, + // HTML + translateSymbolArrow: false, + suppressCheckboxHandling: false, + // { "directive:": function (node, childText, auxData) {} } + customDirectiveHandler: null, + // e.g., "org-js-" + htmlClassPrefix: null, + htmlIdPrefix: null + }, + + untitled: "Untitled", + result: null, + + // TODO: Manage TODO lists + + initialize: function (orgDocument, exportOptions) { + this.orgDocument = orgDocument; + this.documentOptions = orgDocument.options || {}; + this.exportOptions = exportOptions || {}; + + this.headers = []; + this.headerOffset = + typeof this.exportOptions.headerOffset === "number" ? this.exportOptions.headerOffset : 1; + this.sectionNumbers = [0]; + }, + + createTocItem: function (headerNode, parentTocs) { + var childTocs = []; + childTocs.parent = parentTocs; + var tocItem = { headerNode: headerNode, childTocs: childTocs }; + return tocItem; + }, + + computeToc: function (exportTocLevel) { + if (typeof exportTocLevel !== "number") + exportTocLevel = Infinity; + + var toc = []; + toc.parent = null; + + var previousLevel = 1; + var currentTocs = toc; // first + + for (var i = 0; i < this.headers.length; ++i) { + var headerNode = this.headers[i]; + + if (headerNode.level > exportTocLevel) + continue; + + var levelDiff = headerNode.level - previousLevel; + if (levelDiff > 0) { + for (var j = 0; j < levelDiff; ++j) { + if (currentTocs.length === 0) { + // Create a dummy tocItem + var dummyHeader = Node.createHeader([], { + level: previousLevel + j + }); + dummyHeader.sectionNumberText = ""; + currentTocs.push(this.createTocItem(dummyHeader, currentTocs)); + } + currentTocs = currentTocs[currentTocs.length - 1].childTocs; + } + } else if (levelDiff < 0) { + levelDiff = -levelDiff; + for (var k = 0; k < levelDiff; ++k) { + currentTocs = currentTocs.parent; + } + } + + currentTocs.push(this.createTocItem(headerNode, currentTocs)); + + previousLevel = headerNode.level; + } + + return toc; + }, + + convertNode: function (node, recordHeader, insideCodeElement) { + if (!insideCodeElement) { + if (node.type === Node.types.directive) { + if (node.directiveName === "example" || + node.directiveName === "src") { + insideCodeElement = true; + } + } else if (node.type === Node.types.preformatted) { + insideCodeElement = true; + } + } + + if (typeof node === "string") { + node = Node.createText(null, { value: node }); + } + + var childText = node.children ? this.convertNodesInternal(node.children, recordHeader, insideCodeElement) : ""; + var text; + + var auxData = this.computeAuxDataForNode(node); + + switch (node.type) { + case Node.types.header: + // Compute section number + var sectionNumberText = null; + if (recordHeader) { + var thisHeaderLevel = node.level; + var previousHeaderLevel = this.sectionNumbers.length; + if (thisHeaderLevel > previousHeaderLevel) { + // Fill missing section number + var levelDiff = thisHeaderLevel - previousHeaderLevel; + for (var j = 0; j < levelDiff; ++j) { + this.sectionNumbers[thisHeaderLevel - 1 - j] = 0; // Extend + } + } else if (thisHeaderLevel < previousHeaderLevel) { + this.sectionNumbers.length = thisHeaderLevel; // Collapse + } + this.sectionNumbers[thisHeaderLevel - 1]++; + sectionNumberText = this.sectionNumbers.join("."); + node.sectionNumberText = sectionNumberText; // Can be used in ToC + } + + text = this.convertHeader(node, childText, auxData, sectionNumberText); + + if (recordHeader) + this.headers.push(node); + break; + case Node.types.orderedList: + text = this.convertOrderedList(node, childText, auxData); + break; + case Node.types.unorderedList: + text = this.convertUnorderedList(node, childText, auxData); + break; + case Node.types.definitionList: + text = this.convertDefinitionList(node, childText, auxData); + break; + case Node.types.listElement: + if (node.isDefinitionList) { + var termText = this.convertNodesInternal(node.term, recordHeader, insideCodeElement); + text = this.convertDefinitionItem(node, childText, auxData, + termText, childText); + } else { + text = this.convertListItem(node, childText, auxData); + } + break; + case Node.types.paragraph: + text = this.convertParagraph(node, childText, auxData); + break; + case Node.types.preformatted: + text = this.convertPreformatted(node, childText, auxData); + break; + case Node.types.table: + text = this.convertTable(node, childText, auxData); + break; + case Node.types.tableRow: + text = this.convertTableRow(node, childText, auxData); + break; + case Node.types.tableCell: + if (node.isHeader) + text = this.convertTableHeader(node, childText, auxData); + else + text = this.convertTableCell(node, childText, auxData); + break; + case Node.types.horizontalRule: + text = this.convertHorizontalRule(node, childText, auxData); + break; + // ============================================================ // + // Inline + // ============================================================ // + case Node.types.inlineContainer: + text = this.convertInlineContainer(node, childText, auxData); + break; + case Node.types.bold: + text = this.convertBold(node, childText, auxData); + break; + case Node.types.italic: + text = this.convertItalic(node, childText, auxData); + break; + case Node.types.underline: + text = this.convertUnderline(node, childText, auxData); + break; + case Node.types.code: + text = this.convertCode(node, childText, auxData); + break; + case Node.types.dashed: + text = this.convertDashed(node, childText, auxData); + break; + case Node.types.link: + text = this.convertLink(node, childText, auxData); + break; + case Node.types.directive: + switch (node.directiveName) { + case "quote": + text = this.convertQuote(node, childText, auxData); + break; + case "example": + text = this.convertExample(node, childText, auxData); + break; + case "src": + text = this.convertSrc(node, childText, auxData); + break; + case "html": + case "html:": + text = this.convertHTML(node, childText, auxData); + break; + default: + if (this.exportOptions.customDirectiveHandler && + this.exportOptions.customDirectiveHandler[node.directiveName]) { + text = this.exportOptions.customDirectiveHandler[node.directiveName]( + node, childText, auxData + ); + } else { + text = childText; + } + } + break; + case Node.types.text: + text = this.convertText(node.value, insideCodeElement); + break; + default: + throw Error("Unknown node type: " + node.type); + } + + if (typeof this.postProcess === "function") { + text = this.postProcess(node, text, insideCodeElement); + } + + return text; + }, + + convertText: function (text, insideCodeElement) { + var escapedText = this.escapeSpecialChars(text, insideCodeElement); + + if (!this.exportOptions.suppressSubScriptHandling && !insideCodeElement) { + escapedText = this.makeSubscripts(escapedText, insideCodeElement); + } + if (!this.exportOptions.suppressAutoLink) { + escapedText = this.linkURL(escapedText); + } + + return escapedText; + }, + + // By default, ignore html + convertHTML: function (node, childText, auxData) { + return childText; + }, + + convertNodesInternal: function (nodes, recordHeader, insideCodeElement) { + var nodesTexts = []; + for (var i = 0; i < nodes.length; ++i) { + var node = nodes[i]; + var nodeText = this.convertNode(node, recordHeader, insideCodeElement); + nodesTexts.push(nodeText); + } + + // Return entries without their property drawers + return this.combineNodesTexts(nodesTexts).replace(propertyDrawer, ''); + }, + + convertHeaderBlock: function (headerBlock, recordHeader) { + throw Error("convertHeaderBlock is not implemented"); + }, + + convertHeaderTree: function (headerTree, recordHeader) { + return this.convertHeaderBlock(headerTree, recordHeader); + }, + + convertNodesToHeaderTree: function (nodes, nextBlockBegin, blockHeader) { + var childBlocks = []; + var childNodes = []; + + if (typeof nextBlockBegin === "undefined") { + nextBlockBegin = 0; + } + if (typeof blockHeader === "undefined") { + blockHeader = null; + } + + for (var i = nextBlockBegin; i < nodes.length;) { + var node = nodes[i]; + + var isHeader = node.type === Node.types.header; + + if (!isHeader) { + childNodes.push(node); + i = i + 1; + continue; + } + + // Header + if (blockHeader && node.level <= blockHeader.level) { + // Finish Block + break; + } else { + // blockHeader.level < node.level + // Begin child block + var childBlock = this.convertNodesToHeaderTree(nodes, i + 1, node); + childBlocks.push(childBlock); + i = childBlock.nextIndex; + } + } + + // Finish block + return { + header: blockHeader, + childNodes: childNodes, + nextIndex: i, + childBlocks: childBlocks + }; + }, + + convertNodes: function (nodes, recordHeader, insideCodeElement) { + return this.convertNodesInternal(nodes, recordHeader, insideCodeElement); + }, + + combineNodesTexts: function (nodesTexts) { + return nodesTexts.join(""); + }, + + getNodeTextContent: function (node) { + if (node.type === Node.types.text) + return this.escapeSpecialChars(node.value); + else + return node.children ? node.children.map(this.getNodeTextContent, this).join("") : ""; + }, + + // @Override + escapeSpecialChars: function (text) { + throw Error("Implement escapeSpecialChars"); + }, + + // http://daringfireball.net/2010/07/improved_regex_for_matching_urls + urlPattern: /\b(?:https?:\/\/|www\d{0,3}[.]|[a-z0-9.\-]+[.][a-z]{2,4}\/)(?:[^\s()<>]+|\(([^\s()<>]+|(\([^\s()<>]+\)))*\))+(?:\(([^\s()<>]+|(\([^\s()<>]+\)))*\)|[^\s`!()\[\]{};:'".,<>?«»“”‘’])/ig, + + // @Override + linkURL: function (text) { + var self = this; + return text.replace(this.urlPattern, function (matched) { + if (matched.indexOf("://") < 0) + matched = "http://" + matched; + return self.makeLink(matched); + }); + }, + + makeLink: function (url) { + throw Error("Implement makeLink"); + }, + + makeSubscripts: function (text) { + if (this.documentOptions["^"] === "{}") + return text.replace(/\b([^_ \t]*)_{([^}]*)}/g, + this.makeSubscript); + else if (this.documentOptions["^"]) + return text.replace(/\b([^_ \t]*)_([^_]*)\b/g, + this.makeSubscript); + else + return text; + }, + + makeSubscript: function (match, body, subscript) { + throw Error("Implement makeSubscript"); + }, + + stripParametersFromURL: function (url) { + return url.replace(/\?.*$/, ""); + }, + + imageExtensionPattern: new RegExp("(" + [ + "bmp", "png", "jpeg", "jpg", "gif", "tiff", + "tif", "xbm", "xpm", "pbm", "pgm", "ppm", "svg" + ].join("|") + ")$", "i") + }; + + if (typeof exports !== "undefined") + exports.Converter = Converter; + + // var Converter = require("./converter.js").Converter; + // var Node = require("../node.js").Node; + + function ConverterHTML(orgDocument, exportOptions) { + this.initialize(orgDocument, exportOptions); + this.result = this.convert(); + } + + ConverterHTML.prototype = { + __proto__: Converter.prototype, + + convert: function () { + var contentHTML = this.convertNodes(this.orgDocument.nodes, false /* record headers */); + + return { + contentHTML: contentHTML, + toString: function () { + return contentHTML; + } + }; + }, + + tocToHTML: function (toc) { + function tocToHTMLFunction(tocList) { + var html = ""; + for (var i = 0; i < tocList.length; ++i) { + var tocItem = tocList[i]; + var sectionNumberText = tocItem.headerNode.sectionNumberText; + var sectionNumber = this.documentOptions.num ? + this.inlineTag("span", sectionNumberText, { + "class": "section-number" + }) : ""; + var header = this.getNodeTextContent(tocItem.headerNode); + var headerLink = this.inlineTag("a", sectionNumber + header, { + href: "#header-" + sectionNumberText.replace(/\./g, "-") + }); + var subList = tocItem.childTocs.length ? tocToHTMLFunction.call(this, tocItem.childTocs) : ""; + html += this.tag("li", headerLink + subList); + } + return this.tag("ul", html); + } + + return tocToHTMLFunction.call(this, toc); + }, + + computeAuxDataForNode: function (node) { + while (node.parent && + node.parent.type === Node.types.inlineContainer) { + node = node.parent; + } + var attributesNode = node.previousSibling; + var attributesText = ""; + while (attributesNode && + attributesNode.type === Node.types.directive && + attributesNode.directiveName === "attr_html:") { + attributesText += attributesNode.directiveRawValue + " "; + attributesNode = attributesNode.previousSibling; + } + return attributesText; + }, + + // Method to construct org-js generated class + orgClassName: function (className) { + return this.exportOptions.htmlClassPrefix ? + this.exportOptions.htmlClassPrefix + className + : className; + }, + + // Method to construct org-js generated id + orgId: function (id) { + return this.exportOptions.htmlIdPrefix ? + this.exportOptions.htmlIdPrefix + id + : id; + }, + + // ---------------------------------------------------- + // Node conversion + // ---------------------------------------------------- + + convertHeader: function (node, childText, auxData, sectionNumberText) { + var headerAttributes = {}; + + // Parse task status + taskStatusRegex = /^\s*([A-Z]+) / + taskStatusMatch = childText.match(taskStatusRegex); + taskStatus = taskStatusMatch && taskStatusMatch[1]; + childText = childText.replace(taskStatusRegex, ""); + + if (taskStatus) { + childText = this.inlineTag("span", taskStatus, { + "class": "task-status " + taskStatus + }) + childText; + } + + // Parse task tags + taskTagsRegex = /:([\w:]+):/g + taskTagsMatch = childText.match(taskTagsRegex); + taskTags = taskTagsMatch && taskTagsMatch[0].split(":").slice(1,-1); + childText = childText.replace(taskTagsRegex, ""); + + if (taskTags) { + taskTags.forEach(tag => { + childText += this.inlineTag("span", tag, { "class": "task-tag" }) + }); + } + + if (sectionNumberText) { + childText = this.inlineTag("span", sectionNumberText, { + "class": "section-number" + }) + childText; + headerAttributes["id"] = "header-" + sectionNumberText.replace(/\./g, "-"); + } + + if (taskStatus) + headerAttributes["class"] = "task-status " + taskStatus; + + return this.tag("h" + (this.headerOffset + node.level), + childText, headerAttributes, auxData); + }, + + convertOrderedList: function (node, childText, auxData) { + return this.tag("ol", childText, null, auxData); + }, + + convertUnorderedList: function (node, childText, auxData) { + return this.tag("ul", childText, null, auxData); + }, + + convertDefinitionList: function (node, childText, auxData) { + return this.tag("dl", childText, null, auxData); + }, + + convertDefinitionItem: function (node, childText, auxData, + term, definition) { + return this.tag("dt", term) + this.tag("dd", definition); + }, + + convertListItem: function (node, childText, auxData) { + if (this.exportOptions.suppressCheckboxHandling) { + return this.tag("li", childText, null, auxData); + } else { + var listItemAttributes = {}; + var listItemText = childText; + // Embed checkbox + if (/^\s*\[(X| |-)\]([\s\S]*)/.exec(listItemText)) { + listItemText = RegExp.$2 ; + var checkboxIndicator = RegExp.$1; + + var checkboxAttributes = { type: "checkbox" }; + switch (checkboxIndicator) { + case "X": + checkboxAttributes["checked"] = "true"; + listItemAttributes["data-checkbox-status"] = "done"; + break; + case "-": + listItemAttributes["data-checkbox-status"] = "intermediate"; + break; + default: + listItemAttributes["data-checkbox-status"] = "undone"; + break; + } + + listItemText = this.inlineTag("input", null, checkboxAttributes) + listItemText; + } + + return this.tag("li", listItemText, listItemAttributes, auxData); + } + }, + + convertParagraph: function (node, childText, auxData) { + return this.tag("p", childText, null, auxData); + }, + + convertPreformatted: function (node, childText, auxData) { + return this.tag("pre", childText, null, auxData); + }, + + convertTable: function (node, childText, auxData) { + return this.tag("table", this.tag("tbody", childText), null, auxData); + }, + + convertTableRow: function (node, childText, auxData) { + return this.tag("tr", childText); + }, + + convertTableHeader: function (node, childText, auxData) { + return this.tag("th", childText); + }, + + convertTableCell: function (node, childText, auxData) { + return this.tag("td", childText); + }, + + convertHorizontalRule: function (node, childText, auxData) { + return this.tag("hr", null, null, auxData); + }, + + convertInlineContainer: function (node, childText, auxData) { + return childText; + }, + + convertBold: function (node, childText, auxData) { + return this.inlineTag("b", childText); + }, + + convertItalic: function (node, childText, auxData) { + return this.inlineTag("i", childText); + }, + + convertUnderline: function (node, childText, auxData) { + return this.inlineTag("span", childText, { + style: "text-decoration:underline;" + }); + }, + + convertCode: function (node, childText, auxData) { + return this.inlineTag("code", childText); + }, + + convertDashed: function (node, childText, auxData) { + return this.inlineTag("del", childText); + }, + + convertLink: function (node, childText, auxData) { + var srcParameterStripped = this.stripParametersFromURL(node.src); + if (this.imageExtensionPattern.exec(srcParameterStripped)) { + var imgText = this.getNodeTextContent(node); + return this.inlineTag("img", null, { + src: node.src, + alt: imgText, + title: imgText + }, auxData); + } else { + return this.inlineTag("a", childText, { href: node.src }); + } + }, + + convertQuote: function (node, childText, auxData) { + return this.tag("blockquote", childText, null, auxData); + }, + + convertExample: function (node, childText, auxData) { + return this.tag("pre", childText, null, auxData); + }, + + convertSrc: function (node, childText, auxData) { + var codeLanguage = node.directiveArguments.length + ? node.directiveArguments[0] + : "unknown"; + childText = this.tag("code", childText, { + "class": "language-" + codeLanguage + }, auxData); + return this.tag("pre", childText, { + "class": "prettyprint" + }); + }, + + // @override + convertHTML: function (node, childText, auxData) { + if (node.directiveName === "html:") { + return node.directiveRawValue; + } else if (node.directiveName === "html") { + return node.children.map(function (textNode) { + return textNode.value; + }).join("\n"); + } else { + return childText; + } + }, + + // @implement + convertHeaderBlock: function (headerBlock, level, index) { + level = level || 0; + index = index || 0; + + var contents = []; + + var headerNode = headerBlock.header; + if (headerNode) { + contents.push(this.convertNode(headerNode)); + } + + var blockContent = this.convertNodes(headerBlock.childNodes); + contents.push(blockContent); + + var childBlockContent = headerBlock.childBlocks + .map(function (block, idx) { + return this.convertHeaderBlock(block, level + 1, idx); + }, this) + .join("\n"); + contents.push(childBlockContent); + + var contentsText = contents.join("\n"); + + if (headerNode) { + return this.tag("section", "\n" + contents.join("\n"), { + "class": "block block-level-" + level + }); + } else { + return contentsText; + } + }, + + // ---------------------------------------------------- + // Supplemental methods + // ---------------------------------------------------- + + replaceMap: { + // [replacing pattern, predicate] + "&": ["&", null], + "<": ["<", null], + ">": [">", null], + '"': [""", null], + "'": ["'", null], + "->": ["➔", function (text, insideCodeElement) { + return this.exportOptions.translateSymbolArrow && !insideCodeElement; + }] + }, + + replaceRegexp: null, + + // @implement @override + escapeSpecialChars: function (text, insideCodeElement) { + if (!this.replaceRegexp) { + this.replaceRegexp = new RegExp(Object.keys(this.replaceMap).join("|"), "g"); + } + + var replaceMap = this.replaceMap; + var self = this; + return text.replace(this.replaceRegexp, function (matched) { + if (!replaceMap[matched]) { + throw Error("escapeSpecialChars: Invalid match"); + } + + var predicate = replaceMap[matched][1]; + if (typeof predicate === "function" && + !predicate.call(self, text, insideCodeElement)) { + // Not fullfill the predicate + return matched; + } + + return replaceMap[matched][0]; + }); + }, + + // @implement + postProcess: function (node, currentText, insideCodeElement) { + if (this.exportOptions.exportFromLineNumber && + typeof node.fromLineNumber === "number") { + // Wrap with line number information + currentText = this.inlineTag("div", currentText, { + "data-line-number": node.fromLineNumber + }); + } + return currentText; + }, + + // @implement + makeLink: function (url) { + return "" + decodeURIComponent(url) + ""; + }, + + // @implement + makeSubscript: function (match, body, subscript) { + return "" + + body + + "" + + subscript + + ""; + }, + + // ---------------------------------------------------- + // Specific methods + // ---------------------------------------------------- + + attributesObjectToString: function (attributesObject) { + var attributesString = ""; + for (var attributeName in attributesObject) { + if (attributesObject.hasOwnProperty(attributeName)) { + var attributeValue = attributesObject[attributeName]; + // To avoid id/class name conflicts with other frameworks, + // users can add arbitrary prefix to org-js generated + // ids/classes via exportOptions. + if (attributeName === "class") { + attributeValue = this.orgClassName(attributeValue); + } else if (attributeName === "id") { + attributeValue = this.orgId(attributeValue); + } + attributesString += " " + attributeName + "=\"" + attributeValue + "\""; + } + } + return attributesString; + }, + + inlineTag: function (name, innerText, attributesObject, auxAttributesText) { + attributesObject = attributesObject || {}; + + var htmlString = "<" + name; + // TODO: check duplicated attributes + if (auxAttributesText) + htmlString += " " + auxAttributesText; + htmlString += this.attributesObjectToString(attributesObject); + + if (innerText === null) + return htmlString + "/>"; + + htmlString += ">" + innerText + ""; + + return htmlString; + }, + + tag: function (name, innerText, attributesObject, auxAttributesText) { + return this.inlineTag(name, innerText, attributesObject, auxAttributesText) + "\n"; + } + }; + + if (typeof exports !== "undefined") + exports.ConverterHTML = ConverterHTML; + + return exports; +})(); diff --git a/src/interface/desktop/assets/pico.min.css b/src/interface/desktop/assets/pico.min.css new file mode 100644 index 00000000..61caa585 --- /dev/null +++ b/src/interface/desktop/assets/pico.min.css @@ -0,0 +1,5 @@ +@charset "UTF-8";/*! + * Pico CSS v1.5.10 (https://picocss.com) + * Copyright 2019-2023 - Licensed under MIT + */:root{--font-family:system-ui,-apple-system,"Segoe UI","Roboto","Ubuntu","Cantarell","Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";--line-height:1.5;--font-weight:400;--font-size:16px;--border-radius:0.25rem;--border-width:1px;--outline-width:3px;--spacing:1rem;--typography-spacing-vertical:1.5rem;--block-spacing-vertical:calc(var(--spacing) * 2);--block-spacing-horizontal:var(--spacing);--grid-spacing-vertical:0;--grid-spacing-horizontal:var(--spacing);--form-element-spacing-vertical:0.75rem;--form-element-spacing-horizontal:1rem;--nav-element-spacing-vertical:1rem;--nav-element-spacing-horizontal:0.5rem;--nav-link-spacing-vertical:0.5rem;--nav-link-spacing-horizontal:0.5rem;--form-label-font-weight:var(--font-weight);--transition:0.2s ease-in-out;--modal-overlay-backdrop-filter:blur(0.25rem)}@media (min-width:576px){:root{--font-size:17px}}@media (min-width:768px){:root{--font-size:18px}}@media (min-width:992px){:root{--font-size:19px}}@media (min-width:1200px){:root{--font-size:20px}}@media (min-width:576px){body>footer,body>header,body>main,section{--block-spacing-vertical:calc(var(--spacing) * 2.5)}}@media (min-width:768px){body>footer,body>header,body>main,section{--block-spacing-vertical:calc(var(--spacing) * 3)}}@media (min-width:992px){body>footer,body>header,body>main,section{--block-spacing-vertical:calc(var(--spacing) * 3.5)}}@media (min-width:1200px){body>footer,body>header,body>main,section{--block-spacing-vertical:calc(var(--spacing) * 4)}}@media (min-width:576px){article{--block-spacing-horizontal:calc(var(--spacing) * 1.25)}}@media (min-width:768px){article{--block-spacing-horizontal:calc(var(--spacing) * 1.5)}}@media (min-width:992px){article{--block-spacing-horizontal:calc(var(--spacing) * 1.75)}}@media (min-width:1200px){article{--block-spacing-horizontal:calc(var(--spacing) * 2)}}dialog>article{--block-spacing-vertical:calc(var(--spacing) * 2);--block-spacing-horizontal:var(--spacing)}@media (min-width:576px){dialog>article{--block-spacing-vertical:calc(var(--spacing) * 2.5);--block-spacing-horizontal:calc(var(--spacing) * 1.25)}}@media (min-width:768px){dialog>article{--block-spacing-vertical:calc(var(--spacing) * 3);--block-spacing-horizontal:calc(var(--spacing) * 1.5)}}a{--text-decoration:none}a.contrast,a.secondary{--text-decoration:underline}small{--font-size:0.875em}h1,h2,h3,h4,h5,h6{--font-weight:700}h1{--font-size:2rem;--typography-spacing-vertical:3rem}h2{--font-size:1.75rem;--typography-spacing-vertical:2.625rem}h3{--font-size:1.5rem;--typography-spacing-vertical:2.25rem}h4{--font-size:1.25rem;--typography-spacing-vertical:1.874rem}h5{--font-size:1.125rem;--typography-spacing-vertical:1.6875rem}[type=checkbox],[type=radio]{--border-width:2px}[type=checkbox][role=switch]{--border-width:3px}tfoot td,tfoot th,thead td,thead th{--border-width:3px}:not(thead,tfoot)>*>td{--font-size:0.875em}code,kbd,pre,samp{--font-family:"Menlo","Consolas","Roboto Mono","Ubuntu Monospace","Noto Mono","Oxygen Mono","Liberation Mono",monospace,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji"}kbd{--font-weight:bolder}:root:not([data-theme=dark]),[data-theme=light]{--background-color:#fff;--color:hsl(205, 20%, 32%);--h1-color:hsl(205, 30%, 15%);--h2-color:#24333e;--h3-color:hsl(205, 25%, 23%);--h4-color:#374956;--h5-color:hsl(205, 20%, 32%);--h6-color:#4d606d;--muted-color:hsl(205, 10%, 50%);--muted-border-color:hsl(205, 20%, 94%);--primary:hsl(195, 85%, 41%);--primary-hover:hsl(195, 90%, 32%);--primary-focus:rgba(16, 149, 193, 0.125);--primary-inverse:#fff;--secondary:hsl(205, 15%, 41%);--secondary-hover:hsl(205, 20%, 32%);--secondary-focus:rgba(89, 107, 120, 0.125);--secondary-inverse:#fff;--contrast:hsl(205, 30%, 15%);--contrast-hover:#000;--contrast-focus:rgba(89, 107, 120, 0.125);--contrast-inverse:#fff;--mark-background-color:#fff2ca;--mark-color:#543a26;--ins-color:#388e3c;--del-color:#c62828;--blockquote-border-color:var(--muted-border-color);--blockquote-footer-color:var(--muted-color);--button-box-shadow:0 0 0 rgba(0, 0, 0, 0);--button-hover-box-shadow:0 0 0 rgba(0, 0, 0, 0);--form-element-background-color:transparent;--form-element-border-color:hsl(205, 14%, 68%);--form-element-color:var(--color);--form-element-placeholder-color:var(--muted-color);--form-element-active-background-color:transparent;--form-element-active-border-color:var(--primary);--form-element-focus-color:var(--primary-focus);--form-element-disabled-background-color:hsl(205, 18%, 86%);--form-element-disabled-border-color:hsl(205, 14%, 68%);--form-element-disabled-opacity:0.5;--form-element-invalid-border-color:#c62828;--form-element-invalid-active-border-color:#d32f2f;--form-element-invalid-focus-color:rgba(211, 47, 47, 0.125);--form-element-valid-border-color:#388e3c;--form-element-valid-active-border-color:#43a047;--form-element-valid-focus-color:rgba(67, 160, 71, 0.125);--switch-background-color:hsl(205, 16%, 77%);--switch-color:var(--primary-inverse);--switch-checked-background-color:var(--primary);--range-border-color:hsl(205, 18%, 86%);--range-active-border-color:hsl(205, 16%, 77%);--range-thumb-border-color:var(--background-color);--range-thumb-color:var(--secondary);--range-thumb-hover-color:var(--secondary-hover);--range-thumb-active-color:var(--primary);--table-border-color:var(--muted-border-color);--table-row-stripped-background-color:#f6f8f9;--code-background-color:hsl(205, 20%, 94%);--code-color:var(--muted-color);--code-kbd-background-color:var(--contrast);--code-kbd-color:var(--contrast-inverse);--code-tag-color:hsl(330, 40%, 50%);--code-property-color:hsl(185, 40%, 40%);--code-value-color:hsl(40, 20%, 50%);--code-comment-color:hsl(205, 14%, 68%);--accordion-border-color:var(--muted-border-color);--accordion-close-summary-color:var(--color);--accordion-open-summary-color:var(--muted-color);--card-background-color:var(--background-color);--card-border-color:var(--muted-border-color);--card-box-shadow:0.0145rem 0.029rem 0.174rem rgba(27, 40, 50, 0.01698),0.0335rem 0.067rem 0.402rem rgba(27, 40, 50, 0.024),0.0625rem 0.125rem 0.75rem rgba(27, 40, 50, 0.03),0.1125rem 0.225rem 1.35rem rgba(27, 40, 50, 0.036),0.2085rem 0.417rem 2.502rem rgba(27, 40, 50, 0.04302),0.5rem 1rem 6rem rgba(27, 40, 50, 0.06),0 0 0 0.0625rem rgba(27, 40, 50, 0.015);--card-sectionning-background-color:#fbfbfc;--dropdown-background-color:#fbfbfc;--dropdown-border-color:#e1e6eb;--dropdown-box-shadow:var(--card-box-shadow);--dropdown-color:var(--color);--dropdown-hover-background-color:hsl(205, 20%, 94%);--modal-overlay-background-color:rgba(213, 220, 226, 0.7);--progress-background-color:hsl(205, 18%, 86%);--progress-color:var(--primary);--loading-spinner-opacity:0.5;--tooltip-background-color:var(--contrast);--tooltip-color:var(--contrast-inverse);--icon-checkbox:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(255, 255, 255)' stroke-width='4' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='20 6 9 17 4 12'%3E%3C/polyline%3E%3C/svg%3E");--icon-chevron:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(65, 84, 98)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='6 9 12 15 18 9'%3E%3C/polyline%3E%3C/svg%3E");--icon-chevron-button:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(255, 255, 255)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='6 9 12 15 18 9'%3E%3C/polyline%3E%3C/svg%3E");--icon-chevron-button-inverse:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(255, 255, 255)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='6 9 12 15 18 9'%3E%3C/polyline%3E%3C/svg%3E");--icon-close:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(115, 130, 140)' stroke-width='4' stroke-linecap='round' stroke-linejoin='round'%3E%3Cline x1='18' y1='6' x2='6' y2='18'%3E%3C/line%3E%3Cline x1='6' y1='6' x2='18' y2='18'%3E%3C/line%3E%3C/svg%3E");--icon-date:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(65, 84, 98)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='4' width='18' height='18' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='16' y1='2' x2='16' y2='6'%3E%3C/line%3E%3Cline x1='8' y1='2' x2='8' y2='6'%3E%3C/line%3E%3Cline x1='3' y1='10' x2='21' y2='10'%3E%3C/line%3E%3C/svg%3E");--icon-invalid:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(198, 40, 40)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='8' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='16' x2='12.01' y2='16'%3E%3C/line%3E%3C/svg%3E");--icon-minus:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(255, 255, 255)' stroke-width='4' stroke-linecap='round' stroke-linejoin='round'%3E%3Cline x1='5' y1='12' x2='19' y2='12'%3E%3C/line%3E%3C/svg%3E");--icon-search:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(65, 84, 98)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='11' cy='11' r='8'%3E%3C/circle%3E%3Cline x1='21' y1='21' x2='16.65' y2='16.65'%3E%3C/line%3E%3C/svg%3E");--icon-time:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(65, 84, 98)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cpolyline points='12 6 12 12 16 14'%3E%3C/polyline%3E%3C/svg%3E");--icon-valid:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(56, 142, 60)' stroke-width='3' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='20 6 9 17 4 12'%3E%3C/polyline%3E%3C/svg%3E");color-scheme:light}@media only screen and (prefers-color-scheme:dark){:root:not([data-theme]){--background-color:#11191f;--color:hsl(205, 16%, 77%);--h1-color:hsl(205, 20%, 94%);--h2-color:#e1e6eb;--h3-color:hsl(205, 18%, 86%);--h4-color:#c8d1d8;--h5-color:hsl(205, 16%, 77%);--h6-color:#afbbc4;--muted-color:hsl(205, 10%, 50%);--muted-border-color:#1f2d38;--primary:hsl(195, 85%, 41%);--primary-hover:hsl(195, 80%, 50%);--primary-focus:rgba(16, 149, 193, 0.25);--primary-inverse:#fff;--secondary:hsl(205, 15%, 41%);--secondary-hover:hsl(205, 10%, 50%);--secondary-focus:rgba(115, 130, 140, 0.25);--secondary-inverse:#fff;--contrast:hsl(205, 20%, 94%);--contrast-hover:#fff;--contrast-focus:rgba(115, 130, 140, 0.25);--contrast-inverse:#000;--mark-background-color:#d1c284;--mark-color:#11191f;--ins-color:#388e3c;--del-color:#c62828;--blockquote-border-color:var(--muted-border-color);--blockquote-footer-color:var(--muted-color);--button-box-shadow:0 0 0 rgba(0, 0, 0, 0);--button-hover-box-shadow:0 0 0 rgba(0, 0, 0, 0);--form-element-background-color:#11191f;--form-element-border-color:#374956;--form-element-color:var(--color);--form-element-placeholder-color:var(--muted-color);--form-element-active-background-color:var(--form-element-background-color);--form-element-active-border-color:var(--primary);--form-element-focus-color:var(--primary-focus);--form-element-disabled-background-color:hsl(205, 25%, 23%);--form-element-disabled-border-color:hsl(205, 20%, 32%);--form-element-disabled-opacity:0.5;--form-element-invalid-border-color:#b71c1c;--form-element-invalid-active-border-color:#c62828;--form-element-invalid-focus-color:rgba(198, 40, 40, 0.25);--form-element-valid-border-color:#2e7d32;--form-element-valid-active-border-color:#388e3c;--form-element-valid-focus-color:rgba(56, 142, 60, 0.25);--switch-background-color:#374956;--switch-color:var(--primary-inverse);--switch-checked-background-color:var(--primary);--range-border-color:#24333e;--range-active-border-color:hsl(205, 25%, 23%);--range-thumb-border-color:var(--background-color);--range-thumb-color:var(--secondary);--range-thumb-hover-color:var(--secondary-hover);--range-thumb-active-color:var(--primary);--table-border-color:var(--muted-border-color);--table-row-stripped-background-color:rgba(115, 130, 140, 0.05);--code-background-color:#18232c;--code-color:var(--muted-color);--code-kbd-background-color:var(--contrast);--code-kbd-color:var(--contrast-inverse);--code-tag-color:hsl(330, 30%, 50%);--code-property-color:hsl(185, 30%, 50%);--code-value-color:hsl(40, 10%, 50%);--code-comment-color:#4d606d;--accordion-border-color:var(--muted-border-color);--accordion-active-summary-color:var(--primary);--accordion-close-summary-color:var(--color);--accordion-open-summary-color:var(--muted-color);--card-background-color:#141e26;--card-border-color:var(--card-background-color);--card-box-shadow:0.0145rem 0.029rem 0.174rem rgba(0, 0, 0, 0.01698),0.0335rem 0.067rem 0.402rem rgba(0, 0, 0, 0.024),0.0625rem 0.125rem 0.75rem rgba(0, 0, 0, 0.03),0.1125rem 0.225rem 1.35rem rgba(0, 0, 0, 0.036),0.2085rem 0.417rem 2.502rem rgba(0, 0, 0, 0.04302),0.5rem 1rem 6rem rgba(0, 0, 0, 0.06),0 0 0 0.0625rem rgba(0, 0, 0, 0.015);--card-sectionning-background-color:#18232c;--dropdown-background-color:hsl(205, 30%, 15%);--dropdown-border-color:#24333e;--dropdown-box-shadow:var(--card-box-shadow);--dropdown-color:var(--color);--dropdown-hover-background-color:rgba(36, 51, 62, 0.75);--modal-overlay-background-color:rgba(36, 51, 62, 0.8);--progress-background-color:#24333e;--progress-color:var(--primary);--loading-spinner-opacity:0.5;--tooltip-background-color:var(--contrast);--tooltip-color:var(--contrast-inverse);--icon-checkbox:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(255, 255, 255)' stroke-width='4' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='20 6 9 17 4 12'%3E%3C/polyline%3E%3C/svg%3E");--icon-chevron:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(162, 175, 185)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='6 9 12 15 18 9'%3E%3C/polyline%3E%3C/svg%3E");--icon-chevron-button:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(255, 255, 255)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='6 9 12 15 18 9'%3E%3C/polyline%3E%3C/svg%3E");--icon-chevron-button-inverse:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(0, 0, 0)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='6 9 12 15 18 9'%3E%3C/polyline%3E%3C/svg%3E");--icon-close:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(115, 130, 140)' stroke-width='4' stroke-linecap='round' stroke-linejoin='round'%3E%3Cline x1='18' y1='6' x2='6' y2='18'%3E%3C/line%3E%3Cline x1='6' y1='6' x2='18' y2='18'%3E%3C/line%3E%3C/svg%3E");--icon-date:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(162, 175, 185)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='4' width='18' height='18' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='16' y1='2' x2='16' y2='6'%3E%3C/line%3E%3Cline x1='8' y1='2' x2='8' y2='6'%3E%3C/line%3E%3Cline x1='3' y1='10' x2='21' y2='10'%3E%3C/line%3E%3C/svg%3E");--icon-invalid:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(183, 28, 28)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='8' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='16' x2='12.01' y2='16'%3E%3C/line%3E%3C/svg%3E");--icon-minus:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(255, 255, 255)' stroke-width='4' stroke-linecap='round' stroke-linejoin='round'%3E%3Cline x1='5' y1='12' x2='19' y2='12'%3E%3C/line%3E%3C/svg%3E");--icon-search:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(162, 175, 185)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='11' cy='11' r='8'%3E%3C/circle%3E%3Cline x1='21' y1='21' x2='16.65' y2='16.65'%3E%3C/line%3E%3C/svg%3E");--icon-time:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(162, 175, 185)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cpolyline points='12 6 12 12 16 14'%3E%3C/polyline%3E%3C/svg%3E");--icon-valid:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(46, 125, 50)' stroke-width='3' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='20 6 9 17 4 12'%3E%3C/polyline%3E%3C/svg%3E");color-scheme:dark}}[data-theme=dark]{--background-color:#11191f;--color:hsl(205, 16%, 77%);--h1-color:hsl(205, 20%, 94%);--h2-color:#e1e6eb;--h3-color:hsl(205, 18%, 86%);--h4-color:#c8d1d8;--h5-color:hsl(205, 16%, 77%);--h6-color:#afbbc4;--muted-color:hsl(205, 10%, 50%);--muted-border-color:#1f2d38;--primary:hsl(195, 85%, 41%);--primary-hover:hsl(195, 80%, 50%);--primary-focus:rgba(16, 149, 193, 0.25);--primary-inverse:#fff;--secondary:hsl(205, 15%, 41%);--secondary-hover:hsl(205, 10%, 50%);--secondary-focus:rgba(115, 130, 140, 0.25);--secondary-inverse:#fff;--contrast:hsl(205, 20%, 94%);--contrast-hover:#fff;--contrast-focus:rgba(115, 130, 140, 0.25);--contrast-inverse:#000;--mark-background-color:#d1c284;--mark-color:#11191f;--ins-color:#388e3c;--del-color:#c62828;--blockquote-border-color:var(--muted-border-color);--blockquote-footer-color:var(--muted-color);--button-box-shadow:0 0 0 rgba(0, 0, 0, 0);--button-hover-box-shadow:0 0 0 rgba(0, 0, 0, 0);--form-element-background-color:#11191f;--form-element-border-color:#374956;--form-element-color:var(--color);--form-element-placeholder-color:var(--muted-color);--form-element-active-background-color:var(--form-element-background-color);--form-element-active-border-color:var(--primary);--form-element-focus-color:var(--primary-focus);--form-element-disabled-background-color:hsl(205, 25%, 23%);--form-element-disabled-border-color:hsl(205, 20%, 32%);--form-element-disabled-opacity:0.5;--form-element-invalid-border-color:#b71c1c;--form-element-invalid-active-border-color:#c62828;--form-element-invalid-focus-color:rgba(198, 40, 40, 0.25);--form-element-valid-border-color:#2e7d32;--form-element-valid-active-border-color:#388e3c;--form-element-valid-focus-color:rgba(56, 142, 60, 0.25);--switch-background-color:#374956;--switch-color:var(--primary-inverse);--switch-checked-background-color:var(--primary);--range-border-color:#24333e;--range-active-border-color:hsl(205, 25%, 23%);--range-thumb-border-color:var(--background-color);--range-thumb-color:var(--secondary);--range-thumb-hover-color:var(--secondary-hover);--range-thumb-active-color:var(--primary);--table-border-color:var(--muted-border-color);--table-row-stripped-background-color:rgba(115, 130, 140, 0.05);--code-background-color:#18232c;--code-color:var(--muted-color);--code-kbd-background-color:var(--contrast);--code-kbd-color:var(--contrast-inverse);--code-tag-color:hsl(330, 30%, 50%);--code-property-color:hsl(185, 30%, 50%);--code-value-color:hsl(40, 10%, 50%);--code-comment-color:#4d606d;--accordion-border-color:var(--muted-border-color);--accordion-active-summary-color:var(--primary);--accordion-close-summary-color:var(--color);--accordion-open-summary-color:var(--muted-color);--card-background-color:#141e26;--card-border-color:var(--card-background-color);--card-box-shadow:0.0145rem 0.029rem 0.174rem rgba(0, 0, 0, 0.01698),0.0335rem 0.067rem 0.402rem rgba(0, 0, 0, 0.024),0.0625rem 0.125rem 0.75rem rgba(0, 0, 0, 0.03),0.1125rem 0.225rem 1.35rem rgba(0, 0, 0, 0.036),0.2085rem 0.417rem 2.502rem rgba(0, 0, 0, 0.04302),0.5rem 1rem 6rem rgba(0, 0, 0, 0.06),0 0 0 0.0625rem rgba(0, 0, 0, 0.015);--card-sectionning-background-color:#18232c;--dropdown-background-color:hsl(205, 30%, 15%);--dropdown-border-color:#24333e;--dropdown-box-shadow:var(--card-box-shadow);--dropdown-color:var(--color);--dropdown-hover-background-color:rgba(36, 51, 62, 0.75);--modal-overlay-background-color:rgba(36, 51, 62, 0.8);--progress-background-color:#24333e;--progress-color:var(--primary);--loading-spinner-opacity:0.5;--tooltip-background-color:var(--contrast);--tooltip-color:var(--contrast-inverse);--icon-checkbox:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(255, 255, 255)' stroke-width='4' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='20 6 9 17 4 12'%3E%3C/polyline%3E%3C/svg%3E");--icon-chevron:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(162, 175, 185)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='6 9 12 15 18 9'%3E%3C/polyline%3E%3C/svg%3E");--icon-chevron-button:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(255, 255, 255)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='6 9 12 15 18 9'%3E%3C/polyline%3E%3C/svg%3E");--icon-chevron-button-inverse:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(0, 0, 0)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='6 9 12 15 18 9'%3E%3C/polyline%3E%3C/svg%3E");--icon-close:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(115, 130, 140)' stroke-width='4' stroke-linecap='round' stroke-linejoin='round'%3E%3Cline x1='18' y1='6' x2='6' y2='18'%3E%3C/line%3E%3Cline x1='6' y1='6' x2='18' y2='18'%3E%3C/line%3E%3C/svg%3E");--icon-date:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(162, 175, 185)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='3' y='4' width='18' height='18' rx='2' ry='2'%3E%3C/rect%3E%3Cline x1='16' y1='2' x2='16' y2='6'%3E%3C/line%3E%3Cline x1='8' y1='2' x2='8' y2='6'%3E%3C/line%3E%3Cline x1='3' y1='10' x2='21' y2='10'%3E%3C/line%3E%3C/svg%3E");--icon-invalid:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(183, 28, 28)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cline x1='12' y1='8' x2='12' y2='12'%3E%3C/line%3E%3Cline x1='12' y1='16' x2='12.01' y2='16'%3E%3C/line%3E%3C/svg%3E");--icon-minus:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(255, 255, 255)' stroke-width='4' stroke-linecap='round' stroke-linejoin='round'%3E%3Cline x1='5' y1='12' x2='19' y2='12'%3E%3C/line%3E%3C/svg%3E");--icon-search:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(162, 175, 185)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='11' cy='11' r='8'%3E%3C/circle%3E%3Cline x1='21' y1='21' x2='16.65' y2='16.65'%3E%3C/line%3E%3C/svg%3E");--icon-time:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(162, 175, 185)' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Ccircle cx='12' cy='12' r='10'%3E%3C/circle%3E%3Cpolyline points='12 6 12 12 16 14'%3E%3C/polyline%3E%3C/svg%3E");--icon-valid:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='none' stroke='rgb(46, 125, 50)' stroke-width='3' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='20 6 9 17 4 12'%3E%3C/polyline%3E%3C/svg%3E");color-scheme:dark}[type=checkbox],[type=radio],[type=range],progress{accent-color:var(--primary)}*,::after,::before{box-sizing:border-box;background-repeat:no-repeat}::after,::before{text-decoration:inherit;vertical-align:inherit}:where(:root){-webkit-tap-highlight-color:transparent;-webkit-text-size-adjust:100%;-moz-text-size-adjust:100%;text-size-adjust:100%;background-color:var(--background-color);color:var(--color);font-weight:var(--font-weight);font-size:var(--font-size);line-height:var(--line-height);font-family:var(--font-family);text-rendering:optimizeLegibility;overflow-wrap:break-word;cursor:default;-moz-tab-size:4;-o-tab-size:4;tab-size:4}main{display:block}body{width:100%;margin:0}body>footer,body>header,body>main{width:100%;margin-right:auto;margin-left:auto;padding:var(--block-spacing-vertical) 0}.container,.container-fluid{width:100%;margin-right:auto;margin-left:auto;padding-right:var(--spacing);padding-left:var(--spacing)}@media (min-width:576px){.container{max-width:510px;padding-right:0;padding-left:0}}@media (min-width:768px){.container{max-width:700px}}@media (min-width:992px){.container{max-width:920px}}@media (min-width:1200px){.container{max-width:1130px}}section{margin-bottom:var(--block-spacing-vertical)}.grid{grid-column-gap:var(--grid-spacing-horizontal);grid-row-gap:var(--grid-spacing-vertical);display:grid;grid-template-columns:1fr;margin:0}@media (min-width:992px){.grid{grid-template-columns:repeat(auto-fit,minmax(0%,1fr))}}.grid>*{min-width:0}figure{display:block;margin:0;padding:0;overflow-x:auto}figure figcaption{padding:calc(var(--spacing) * .5) 0;color:var(--muted-color)}b,strong{font-weight:bolder}sub,sup{position:relative;font-size:.75em;line-height:0;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}address,blockquote,dl,figure,form,ol,p,pre,table,ul{margin-top:0;margin-bottom:var(--typography-spacing-vertical);color:var(--color);font-style:normal;font-weight:var(--font-weight);font-size:var(--font-size)}[role=link],a{--color:var(--primary);--background-color:transparent;outline:0;background-color:var(--background-color);color:var(--color);-webkit-text-decoration:var(--text-decoration);text-decoration:var(--text-decoration);transition:background-color var(--transition),color var(--transition),box-shadow var(--transition),-webkit-text-decoration var(--transition);transition:background-color var(--transition),color var(--transition),text-decoration var(--transition),box-shadow var(--transition);transition:background-color var(--transition),color var(--transition),text-decoration var(--transition),box-shadow var(--transition),-webkit-text-decoration var(--transition)}[role=link]:is([aria-current],:hover,:active,:focus),a:is([aria-current],:hover,:active,:focus){--color:var(--primary-hover);--text-decoration:underline}[role=link]:focus,a:focus{--background-color:var(--primary-focus)}[role=link].secondary,a.secondary{--color:var(--secondary)}[role=link].secondary:is([aria-current],:hover,:active,:focus),a.secondary:is([aria-current],:hover,:active,:focus){--color:var(--secondary-hover)}[role=link].secondary:focus,a.secondary:focus{--background-color:var(--secondary-focus)}[role=link].contrast,a.contrast{--color:var(--contrast)}[role=link].contrast:is([aria-current],:hover,:active,:focus),a.contrast:is([aria-current],:hover,:active,:focus){--color:var(--contrast-hover)}[role=link].contrast:focus,a.contrast:focus{--background-color:var(--contrast-focus)}h1,h2,h3,h4,h5,h6{margin-top:0;margin-bottom:var(--typography-spacing-vertical);color:var(--color);font-weight:var(--font-weight);font-size:var(--font-size);font-family:var(--font-family)}h1{--color:var(--h1-color)}h2{--color:var(--h2-color)}h3{--color:var(--h3-color)}h4{--color:var(--h4-color)}h5{--color:var(--h5-color)}h6{--color:var(--h6-color)}:where(address,blockquote,dl,figure,form,ol,p,pre,table,ul)~:is(h1,h2,h3,h4,h5,h6){margin-top:var(--typography-spacing-vertical)}.headings,hgroup{margin-bottom:var(--typography-spacing-vertical)}.headings>*,hgroup>*{margin-bottom:0}.headings>:last-child,hgroup>:last-child{--color:var(--muted-color);--font-weight:unset;font-size:1rem;font-family:unset}p{margin-bottom:var(--typography-spacing-vertical)}small{font-size:var(--font-size)}:where(dl,ol,ul){padding-right:0;padding-left:var(--spacing);-webkit-padding-start:var(--spacing);padding-inline-start:var(--spacing);-webkit-padding-end:0;padding-inline-end:0}:where(dl,ol,ul) li{margin-bottom:calc(var(--typography-spacing-vertical) * .25)}:where(dl,ol,ul) :is(dl,ol,ul){margin:0;margin-top:calc(var(--typography-spacing-vertical) * .25)}ul li{list-style:square}mark{padding:.125rem .25rem;background-color:var(--mark-background-color);color:var(--mark-color);vertical-align:baseline}blockquote{display:block;margin:var(--typography-spacing-vertical) 0;padding:var(--spacing);border-right:none;border-left:.25rem solid var(--blockquote-border-color);-webkit-border-start:0.25rem solid var(--blockquote-border-color);border-inline-start:0.25rem solid var(--blockquote-border-color);-webkit-border-end:none;border-inline-end:none}blockquote footer{margin-top:calc(var(--typography-spacing-vertical) * .5);color:var(--blockquote-footer-color)}abbr[title]{border-bottom:1px dotted;text-decoration:none;cursor:help}ins{color:var(--ins-color);text-decoration:none}del{color:var(--del-color)}::-moz-selection{background-color:var(--primary-focus)}::selection{background-color:var(--primary-focus)}:where(audio,canvas,iframe,img,svg,video){vertical-align:middle}audio,video{display:inline-block}audio:not([controls]){display:none;height:0}:where(iframe){border-style:none}img{max-width:100%;height:auto;border-style:none}:where(svg:not([fill])){fill:currentColor}svg:not(:root){overflow:hidden}button{margin:0;overflow:visible;font-family:inherit;text-transform:none}[type=button],[type=reset],[type=submit],button{-webkit-appearance:button}button{display:block;width:100%;margin-bottom:var(--spacing)}[role=button]{display:inline-block;text-decoration:none}[role=button],button,input[type=button],input[type=reset],input[type=submit]{--background-color:var(--primary);--border-color:var(--primary);--color:var(--primary-inverse);--box-shadow:var(--button-box-shadow, 0 0 0 rgba(0, 0, 0, 0));padding:var(--form-element-spacing-vertical) var(--form-element-spacing-horizontal);border:var(--border-width) solid var(--border-color);border-radius:var(--border-radius);outline:0;background-color:var(--background-color);box-shadow:var(--box-shadow);color:var(--color);font-weight:var(--font-weight);font-size:1rem;line-height:var(--line-height);text-align:center;cursor:pointer;transition:background-color var(--transition),border-color var(--transition),color var(--transition),box-shadow var(--transition)}[role=button]:is([aria-current],:hover,:active,:focus),button:is([aria-current],:hover,:active,:focus),input[type=button]:is([aria-current],:hover,:active,:focus),input[type=reset]:is([aria-current],:hover,:active,:focus),input[type=submit]:is([aria-current],:hover,:active,:focus){--background-color:var(--primary-hover);--border-color:var(--primary-hover);--box-shadow:var(--button-hover-box-shadow, 0 0 0 rgba(0, 0, 0, 0));--color:var(--primary-inverse)}[role=button]:focus,button:focus,input[type=button]:focus,input[type=reset]:focus,input[type=submit]:focus{--box-shadow:var(--button-hover-box-shadow, 0 0 0 rgba(0, 0, 0, 0)),0 0 0 var(--outline-width) var(--primary-focus)}:is(button,input[type=submit],input[type=button],[role=button]).secondary,input[type=reset]{--background-color:var(--secondary);--border-color:var(--secondary);--color:var(--secondary-inverse);cursor:pointer}:is(button,input[type=submit],input[type=button],[role=button]).secondary:is([aria-current],:hover,:active,:focus),input[type=reset]:is([aria-current],:hover,:active,:focus){--background-color:var(--secondary-hover);--border-color:var(--secondary-hover);--color:var(--secondary-inverse)}:is(button,input[type=submit],input[type=button],[role=button]).secondary:focus,input[type=reset]:focus{--box-shadow:var(--button-hover-box-shadow, 0 0 0 rgba(0, 0, 0, 0)),0 0 0 var(--outline-width) var(--secondary-focus)}:is(button,input[type=submit],input[type=button],[role=button]).contrast{--background-color:var(--contrast);--border-color:var(--contrast);--color:var(--contrast-inverse)}:is(button,input[type=submit],input[type=button],[role=button]).contrast:is([aria-current],:hover,:active,:focus){--background-color:var(--contrast-hover);--border-color:var(--contrast-hover);--color:var(--contrast-inverse)}:is(button,input[type=submit],input[type=button],[role=button]).contrast:focus{--box-shadow:var(--button-hover-box-shadow, 0 0 0 rgba(0, 0, 0, 0)),0 0 0 var(--outline-width) var(--contrast-focus)}:is(button,input[type=submit],input[type=button],[role=button]).outline,input[type=reset].outline{--background-color:transparent;--color:var(--primary)}:is(button,input[type=submit],input[type=button],[role=button]).outline:is([aria-current],:hover,:active,:focus),input[type=reset].outline:is([aria-current],:hover,:active,:focus){--background-color:transparent;--color:var(--primary-hover)}:is(button,input[type=submit],input[type=button],[role=button]).outline.secondary,input[type=reset].outline{--color:var(--secondary)}:is(button,input[type=submit],input[type=button],[role=button]).outline.secondary:is([aria-current],:hover,:active,:focus),input[type=reset].outline:is([aria-current],:hover,:active,:focus){--color:var(--secondary-hover)}:is(button,input[type=submit],input[type=button],[role=button]).outline.contrast{--color:var(--contrast)}:is(button,input[type=submit],input[type=button],[role=button]).outline.contrast:is([aria-current],:hover,:active,:focus){--color:var(--contrast-hover)}:where(button,[type=submit],[type=button],[type=reset],[role=button])[disabled],:where(fieldset[disabled]) :is(button,[type=submit],[type=button],[type=reset],[role=button]),a[role=button]:not([href]){opacity:.5;pointer-events:none}input,optgroup,select,textarea{margin:0;font-size:1rem;line-height:var(--line-height);font-family:inherit;letter-spacing:inherit}input{overflow:visible}select{text-transform:none}legend{max-width:100%;padding:0;color:inherit;white-space:normal}textarea{overflow:auto}[type=checkbox],[type=radio]{padding:0}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}::-moz-focus-inner{padding:0;border-style:none}:-moz-focusring{outline:0}:-moz-ui-invalid{box-shadow:none}::-ms-expand{display:none}[type=file],[type=range]{padding:0;border-width:0}input:not([type=checkbox],[type=radio],[type=range]){height:calc(1rem * var(--line-height) + var(--form-element-spacing-vertical) * 2 + var(--border-width) * 2)}fieldset{margin:0;margin-bottom:var(--spacing);padding:0;border:0}fieldset legend,label{display:block;margin-bottom:calc(var(--spacing) * .25);font-weight:var(--form-label-font-weight,var(--font-weight))}input:not([type=checkbox],[type=radio]),select,textarea{width:100%}input:not([type=checkbox],[type=radio],[type=range],[type=file]),select,textarea{-webkit-appearance:none;-moz-appearance:none;appearance:none;padding:var(--form-element-spacing-vertical) var(--form-element-spacing-horizontal)}input,select,textarea{--background-color:var(--form-element-background-color);--border-color:var(--form-element-border-color);--color:var(--form-element-color);--box-shadow:none;border:var(--border-width) solid var(--border-color);border-radius:var(--border-radius);outline:0;background-color:var(--background-color);box-shadow:var(--box-shadow);color:var(--color);font-weight:var(--font-weight);transition:background-color var(--transition),border-color var(--transition),color var(--transition),box-shadow var(--transition)}:where(select,textarea):is(:active,:focus),input:not([type=submit],[type=button],[type=reset],[type=checkbox],[type=radio],[readonly]):is(:active,:focus){--background-color:var(--form-element-active-background-color)}:where(select,textarea):is(:active,:focus),input:not([type=submit],[type=button],[type=reset],[role=switch],[readonly]):is(:active,:focus){--border-color:var(--form-element-active-border-color)}input:not([type=submit],[type=button],[type=reset],[type=range],[type=file],[readonly]):focus,select:focus,textarea:focus{--box-shadow:0 0 0 var(--outline-width) var(--form-element-focus-color)}:where(fieldset[disabled]) :is(input:not([type=submit],[type=button],[type=reset]),select,textarea),input:not([type=submit],[type=button],[type=reset])[disabled],select[disabled],textarea[disabled]{--background-color:var(--form-element-disabled-background-color);--border-color:var(--form-element-disabled-border-color);opacity:var(--form-element-disabled-opacity);pointer-events:none}:where(input,select,textarea):not([type=checkbox],[type=radio],[type=date],[type=datetime-local],[type=month],[type=time],[type=week])[aria-invalid]{padding-right:calc(var(--form-element-spacing-horizontal) + 1.5rem)!important;padding-left:var(--form-element-spacing-horizontal);-webkit-padding-start:var(--form-element-spacing-horizontal)!important;padding-inline-start:var(--form-element-spacing-horizontal)!important;-webkit-padding-end:calc(var(--form-element-spacing-horizontal) + 1.5rem)!important;padding-inline-end:calc(var(--form-element-spacing-horizontal) + 1.5rem)!important;background-position:center right .75rem;background-size:1rem auto;background-repeat:no-repeat}:where(input,select,textarea):not([type=checkbox],[type=radio],[type=date],[type=datetime-local],[type=month],[type=time],[type=week])[aria-invalid=false]{background-image:var(--icon-valid)}:where(input,select,textarea):not([type=checkbox],[type=radio],[type=date],[type=datetime-local],[type=month],[type=time],[type=week])[aria-invalid=true]{background-image:var(--icon-invalid)}:where(input,select,textarea)[aria-invalid=false]{--border-color:var(--form-element-valid-border-color)}:where(input,select,textarea)[aria-invalid=false]:is(:active,:focus){--border-color:var(--form-element-valid-active-border-color)!important;--box-shadow:0 0 0 var(--outline-width) var(--form-element-valid-focus-color)!important}:where(input,select,textarea)[aria-invalid=true]{--border-color:var(--form-element-invalid-border-color)}:where(input,select,textarea)[aria-invalid=true]:is(:active,:focus){--border-color:var(--form-element-invalid-active-border-color)!important;--box-shadow:0 0 0 var(--outline-width) var(--form-element-invalid-focus-color)!important}[dir=rtl] :where(input,select,textarea):not([type=checkbox],[type=radio]):is([aria-invalid],[aria-invalid=true],[aria-invalid=false]){background-position:center left .75rem}input::-webkit-input-placeholder,input::placeholder,select:invalid,textarea::-webkit-input-placeholder,textarea::placeholder{color:var(--form-element-placeholder-color);opacity:1}input:not([type=checkbox],[type=radio]),select,textarea{margin-bottom:var(--spacing)}select::-ms-expand{border:0;background-color:transparent}select:not([multiple],[size]){padding-right:calc(var(--form-element-spacing-horizontal) + 1.5rem);padding-left:var(--form-element-spacing-horizontal);-webkit-padding-start:var(--form-element-spacing-horizontal);padding-inline-start:var(--form-element-spacing-horizontal);-webkit-padding-end:calc(var(--form-element-spacing-horizontal) + 1.5rem);padding-inline-end:calc(var(--form-element-spacing-horizontal) + 1.5rem);background-image:var(--icon-chevron);background-position:center right .75rem;background-size:1rem auto;background-repeat:no-repeat}[dir=rtl] select:not([multiple],[size]){background-position:center left .75rem}:where(input,select,textarea,.grid)+small{display:block;width:100%;margin-top:calc(var(--spacing) * -.75);margin-bottom:var(--spacing);color:var(--muted-color)}label>:where(input,select,textarea){margin-top:calc(var(--spacing) * .25)}[type=checkbox],[type=radio]{-webkit-appearance:none;-moz-appearance:none;appearance:none;width:1.25em;height:1.25em;margin-top:-.125em;margin-right:.375em;margin-left:0;-webkit-margin-start:0;margin-inline-start:0;-webkit-margin-end:.375em;margin-inline-end:.375em;border-width:var(--border-width);font-size:inherit;vertical-align:middle;cursor:pointer}[type=checkbox]::-ms-check,[type=radio]::-ms-check{display:none}[type=checkbox]:checked,[type=checkbox]:checked:active,[type=checkbox]:checked:focus,[type=radio]:checked,[type=radio]:checked:active,[type=radio]:checked:focus{--background-color:var(--primary);--border-color:var(--primary);background-image:var(--icon-checkbox);background-position:center;background-size:.75em auto;background-repeat:no-repeat}[type=checkbox]~label,[type=radio]~label{display:inline-block;margin-right:.375em;margin-bottom:0;cursor:pointer}[type=checkbox]:indeterminate{--background-color:var(--primary);--border-color:var(--primary);background-image:var(--icon-minus);background-position:center;background-size:.75em auto;background-repeat:no-repeat}[type=radio]{border-radius:50%}[type=radio]:checked,[type=radio]:checked:active,[type=radio]:checked:focus{--background-color:var(--primary-inverse);border-width:.35em;background-image:none}[type=checkbox][role=switch]{--background-color:var(--switch-background-color);--border-color:var(--switch-background-color);--color:var(--switch-color);width:2.25em;height:1.25em;border:var(--border-width) solid var(--border-color);border-radius:1.25em;background-color:var(--background-color);line-height:1.25em}[type=checkbox][role=switch]:focus{--background-color:var(--switch-background-color);--border-color:var(--switch-background-color)}[type=checkbox][role=switch]:checked{--background-color:var(--switch-checked-background-color);--border-color:var(--switch-checked-background-color)}[type=checkbox][role=switch]:before{display:block;width:calc(1.25em - (var(--border-width) * 2));height:100%;border-radius:50%;background-color:var(--color);content:"";transition:margin .1s ease-in-out}[type=checkbox][role=switch]:checked{background-image:none}[type=checkbox][role=switch]:checked::before{margin-left:calc(1.125em - var(--border-width));-webkit-margin-start:calc(1.125em - var(--border-width));margin-inline-start:calc(1.125em - var(--border-width))}[type=checkbox]:checked[aria-invalid=false],[type=checkbox][aria-invalid=false],[type=checkbox][role=switch]:checked[aria-invalid=false],[type=checkbox][role=switch][aria-invalid=false],[type=radio]:checked[aria-invalid=false],[type=radio][aria-invalid=false]{--border-color:var(--form-element-valid-border-color)}[type=checkbox]:checked[aria-invalid=true],[type=checkbox][aria-invalid=true],[type=checkbox][role=switch]:checked[aria-invalid=true],[type=checkbox][role=switch][aria-invalid=true],[type=radio]:checked[aria-invalid=true],[type=radio][aria-invalid=true]{--border-color:var(--form-element-invalid-border-color)}[type=color]::-webkit-color-swatch-wrapper{padding:0}[type=color]::-moz-focus-inner{padding:0}[type=color]::-webkit-color-swatch{border:0;border-radius:calc(var(--border-radius) * .5)}[type=color]::-moz-color-swatch{border:0;border-radius:calc(var(--border-radius) * .5)}input:not([type=checkbox],[type=radio],[type=range],[type=file]):is([type=date],[type=datetime-local],[type=month],[type=time],[type=week]){--icon-position:0.75rem;--icon-width:1rem;padding-right:calc(var(--icon-width) + var(--icon-position));background-image:var(--icon-date);background-position:center right var(--icon-position);background-size:var(--icon-width) auto;background-repeat:no-repeat}input:not([type=checkbox],[type=radio],[type=range],[type=file])[type=time]{background-image:var(--icon-time)}[type=date]::-webkit-calendar-picker-indicator,[type=datetime-local]::-webkit-calendar-picker-indicator,[type=month]::-webkit-calendar-picker-indicator,[type=time]::-webkit-calendar-picker-indicator,[type=week]::-webkit-calendar-picker-indicator{width:var(--icon-width);margin-right:calc(var(--icon-width) * -1);margin-left:var(--icon-position);opacity:0}[dir=rtl] :is([type=date],[type=datetime-local],[type=month],[type=time],[type=week]){text-align:right}@-moz-document url-prefix(){[type=date],[type=datetime-local],[type=month],[type=time],[type=week]{padding-right:var(--form-element-spacing-horizontal)!important;background-image:none!important}}[type=file]{--color:var(--muted-color);padding:calc(var(--form-element-spacing-vertical) * .5) 0;border:0;border-radius:0;background:0 0}[type=file]::file-selector-button{--background-color:var(--secondary);--border-color:var(--secondary);--color:var(--secondary-inverse);margin-right:calc(var(--spacing)/ 2);margin-left:0;-webkit-margin-start:0;margin-inline-start:0;-webkit-margin-end:calc(var(--spacing)/ 2);margin-inline-end:calc(var(--spacing)/ 2);padding:calc(var(--form-element-spacing-vertical) * .5) calc(var(--form-element-spacing-horizontal) * .5);border:var(--border-width) solid var(--border-color);border-radius:var(--border-radius);outline:0;background-color:var(--background-color);box-shadow:var(--box-shadow);color:var(--color);font-weight:var(--font-weight);font-size:1rem;line-height:var(--line-height);text-align:center;cursor:pointer;transition:background-color var(--transition),border-color var(--transition),color var(--transition),box-shadow var(--transition)}[type=file]::file-selector-button:is(:hover,:active,:focus){--background-color:var(--secondary-hover);--border-color:var(--secondary-hover)}[type=file]::-webkit-file-upload-button{--background-color:var(--secondary);--border-color:var(--secondary);--color:var(--secondary-inverse);margin-right:calc(var(--spacing)/ 2);margin-left:0;-webkit-margin-start:0;margin-inline-start:0;-webkit-margin-end:calc(var(--spacing)/ 2);margin-inline-end:calc(var(--spacing)/ 2);padding:calc(var(--form-element-spacing-vertical) * .5) calc(var(--form-element-spacing-horizontal) * .5);border:var(--border-width) solid var(--border-color);border-radius:var(--border-radius);outline:0;background-color:var(--background-color);box-shadow:var(--box-shadow);color:var(--color);font-weight:var(--font-weight);font-size:1rem;line-height:var(--line-height);text-align:center;cursor:pointer;-webkit-transition:background-color var(--transition),border-color var(--transition),color var(--transition),box-shadow var(--transition);transition:background-color var(--transition),border-color var(--transition),color var(--transition),box-shadow var(--transition)}[type=file]::-webkit-file-upload-button:is(:hover,:active,:focus){--background-color:var(--secondary-hover);--border-color:var(--secondary-hover)}[type=file]::-ms-browse{--background-color:var(--secondary);--border-color:var(--secondary);--color:var(--secondary-inverse);margin-right:calc(var(--spacing)/ 2);margin-left:0;margin-inline-start:0;margin-inline-end:calc(var(--spacing)/ 2);padding:calc(var(--form-element-spacing-vertical) * .5) calc(var(--form-element-spacing-horizontal) * .5);border:var(--border-width) solid var(--border-color);border-radius:var(--border-radius);outline:0;background-color:var(--background-color);box-shadow:var(--box-shadow);color:var(--color);font-weight:var(--font-weight);font-size:1rem;line-height:var(--line-height);text-align:center;cursor:pointer;-ms-transition:background-color var(--transition),border-color var(--transition),color var(--transition),box-shadow var(--transition);transition:background-color var(--transition),border-color var(--transition),color var(--transition),box-shadow var(--transition)}[type=file]::-ms-browse:is(:hover,:active,:focus){--background-color:var(--secondary-hover);--border-color:var(--secondary-hover)}[type=range]{-webkit-appearance:none;-moz-appearance:none;appearance:none;width:100%;height:1.25rem;background:0 0}[type=range]::-webkit-slider-runnable-track{width:100%;height:.25rem;border-radius:var(--border-radius);background-color:var(--range-border-color);-webkit-transition:background-color var(--transition),box-shadow var(--transition);transition:background-color var(--transition),box-shadow var(--transition)}[type=range]::-moz-range-track{width:100%;height:.25rem;border-radius:var(--border-radius);background-color:var(--range-border-color);-moz-transition:background-color var(--transition),box-shadow var(--transition);transition:background-color var(--transition),box-shadow var(--transition)}[type=range]::-ms-track{width:100%;height:.25rem;border-radius:var(--border-radius);background-color:var(--range-border-color);-ms-transition:background-color var(--transition),box-shadow var(--transition);transition:background-color var(--transition),box-shadow var(--transition)}[type=range]::-webkit-slider-thumb{-webkit-appearance:none;width:1.25rem;height:1.25rem;margin-top:-.5rem;border:2px solid var(--range-thumb-border-color);border-radius:50%;background-color:var(--range-thumb-color);cursor:pointer;-webkit-transition:background-color var(--transition),transform var(--transition);transition:background-color var(--transition),transform var(--transition)}[type=range]::-moz-range-thumb{-webkit-appearance:none;width:1.25rem;height:1.25rem;margin-top:-.5rem;border:2px solid var(--range-thumb-border-color);border-radius:50%;background-color:var(--range-thumb-color);cursor:pointer;-moz-transition:background-color var(--transition),transform var(--transition);transition:background-color var(--transition),transform var(--transition)}[type=range]::-ms-thumb{-webkit-appearance:none;width:1.25rem;height:1.25rem;margin-top:-.5rem;border:2px solid var(--range-thumb-border-color);border-radius:50%;background-color:var(--range-thumb-color);cursor:pointer;-ms-transition:background-color var(--transition),transform var(--transition);transition:background-color var(--transition),transform var(--transition)}[type=range]:focus,[type=range]:hover{--range-border-color:var(--range-active-border-color);--range-thumb-color:var(--range-thumb-hover-color)}[type=range]:active{--range-thumb-color:var(--range-thumb-active-color)}[type=range]:active::-webkit-slider-thumb{transform:scale(1.25)}[type=range]:active::-moz-range-thumb{transform:scale(1.25)}[type=range]:active::-ms-thumb{transform:scale(1.25)}input:not([type=checkbox],[type=radio],[type=range],[type=file])[type=search]{-webkit-padding-start:calc(var(--form-element-spacing-horizontal) + 1.75rem);padding-inline-start:calc(var(--form-element-spacing-horizontal) + 1.75rem);border-radius:5rem;background-image:var(--icon-search);background-position:center left 1.125rem;background-size:1rem auto;background-repeat:no-repeat}input:not([type=checkbox],[type=radio],[type=range],[type=file])[type=search][aria-invalid]{-webkit-padding-start:calc(var(--form-element-spacing-horizontal) + 1.75rem)!important;padding-inline-start:calc(var(--form-element-spacing-horizontal) + 1.75rem)!important;background-position:center left 1.125rem,center right .75rem}input:not([type=checkbox],[type=radio],[type=range],[type=file])[type=search][aria-invalid=false]{background-image:var(--icon-search),var(--icon-valid)}input:not([type=checkbox],[type=radio],[type=range],[type=file])[type=search][aria-invalid=true]{background-image:var(--icon-search),var(--icon-invalid)}[type=search]::-webkit-search-cancel-button{-webkit-appearance:none;display:none}[dir=rtl] :where(input):not([type=checkbox],[type=radio],[type=range],[type=file])[type=search]{background-position:center right 1.125rem}[dir=rtl] :where(input):not([type=checkbox],[type=radio],[type=range],[type=file])[type=search][aria-invalid]{background-position:center right 1.125rem,center left .75rem}:where(table){width:100%;border-collapse:collapse;border-spacing:0;text-indent:0}td,th{padding:calc(var(--spacing)/ 2) var(--spacing);border-bottom:var(--border-width) solid var(--table-border-color);color:var(--color);font-weight:var(--font-weight);font-size:var(--font-size);text-align:left;text-align:start}tfoot td,tfoot th{border-top:var(--border-width) solid var(--table-border-color);border-bottom:0}table[role=grid] tbody tr:nth-child(odd){background-color:var(--table-row-stripped-background-color)}code,kbd,pre,samp{font-size:.875em;font-family:var(--font-family)}pre{-ms-overflow-style:scrollbar;overflow:auto}code,kbd,pre{border-radius:var(--border-radius);background:var(--code-background-color);color:var(--code-color);font-weight:var(--font-weight);line-height:initial}code,kbd{display:inline-block;padding:.375rem .5rem}pre{display:block;margin-bottom:var(--spacing);overflow-x:auto}pre>code{display:block;padding:var(--spacing);background:0 0;font-size:14px;line-height:var(--line-height)}code b{color:var(--code-tag-color);font-weight:var(--font-weight)}code i{color:var(--code-property-color);font-style:normal}code u{color:var(--code-value-color);text-decoration:none}code em{color:var(--code-comment-color);font-style:normal}kbd{background-color:var(--code-kbd-background-color);color:var(--code-kbd-color);vertical-align:baseline}hr{height:0;border:0;border-top:1px solid var(--muted-border-color);color:inherit}[hidden],template{display:none!important}canvas{display:inline-block}details{display:block;margin-bottom:var(--spacing);padding-bottom:var(--spacing);border-bottom:var(--border-width) solid var(--accordion-border-color)}details summary{line-height:1rem;list-style-type:none;cursor:pointer;transition:color var(--transition)}details summary:not([role]){color:var(--accordion-close-summary-color)}details summary::-webkit-details-marker{display:none}details summary::marker{display:none}details summary::-moz-list-bullet{list-style-type:none}details summary::after{display:block;width:1rem;height:1rem;-webkit-margin-start:calc(var(--spacing,1rem) * 0.5);margin-inline-start:calc(var(--spacing,1rem) * .5);float:right;transform:rotate(-90deg);background-image:var(--icon-chevron);background-position:right center;background-size:1rem auto;background-repeat:no-repeat;content:"";transition:transform var(--transition)}details summary:focus{outline:0}details summary:focus:not([role=button]){color:var(--accordion-active-summary-color)}details summary[role=button]{width:100%;text-align:left}details summary[role=button]::after{height:calc(1rem * var(--line-height,1.5));background-image:var(--icon-chevron-button)}details summary[role=button]:not(.outline).contrast::after{background-image:var(--icon-chevron-button-inverse)}details[open]>summary{margin-bottom:calc(var(--spacing))}details[open]>summary:not([role]):not(:focus){color:var(--accordion-open-summary-color)}details[open]>summary::after{transform:rotate(0)}[dir=rtl] details summary{text-align:right}[dir=rtl] details summary::after{float:left;background-position:left center}article{margin:var(--block-spacing-vertical) 0;padding:var(--block-spacing-vertical) var(--block-spacing-horizontal);border-radius:var(--border-radius);background:var(--card-background-color);box-shadow:var(--card-box-shadow)}article>footer,article>header{margin-right:calc(var(--block-spacing-horizontal) * -1);margin-left:calc(var(--block-spacing-horizontal) * -1);padding:calc(var(--block-spacing-vertical) * .66) var(--block-spacing-horizontal);background-color:var(--card-sectionning-background-color)}article>header{margin-top:calc(var(--block-spacing-vertical) * -1);margin-bottom:var(--block-spacing-vertical);border-bottom:var(--border-width) solid var(--card-border-color);border-top-right-radius:var(--border-radius);border-top-left-radius:var(--border-radius)}article>footer{margin-top:var(--block-spacing-vertical);margin-bottom:calc(var(--block-spacing-vertical) * -1);border-top:var(--border-width) solid var(--card-border-color);border-bottom-right-radius:var(--border-radius);border-bottom-left-radius:var(--border-radius)}:root{--scrollbar-width:0px}dialog{display:flex;z-index:999;position:fixed;top:0;right:0;bottom:0;left:0;align-items:center;justify-content:center;width:inherit;min-width:100%;height:inherit;min-height:100%;padding:var(--spacing);border:0;-webkit-backdrop-filter:var(--modal-overlay-backdrop-filter);backdrop-filter:var(--modal-overlay-backdrop-filter);background-color:var(--modal-overlay-background-color);color:var(--color)}dialog article{max-height:calc(100vh - var(--spacing) * 2);overflow:auto}@media (min-width:576px){dialog article{max-width:510px}}@media (min-width:768px){dialog article{max-width:700px}}dialog article>footer,dialog article>header{padding:calc(var(--block-spacing-vertical) * .5) var(--block-spacing-horizontal)}dialog article>header .close{margin:0;margin-left:var(--spacing);float:right}dialog article>footer{text-align:right}dialog article>footer [role=button]{margin-bottom:0}dialog article>footer [role=button]:not(:first-of-type){margin-left:calc(var(--spacing) * .5)}dialog article p:last-of-type{margin:0}dialog article .close{display:block;width:1rem;height:1rem;margin-top:calc(var(--block-spacing-vertical) * -.5);margin-bottom:var(--typography-spacing-vertical);margin-left:auto;background-image:var(--icon-close);background-position:center;background-size:auto 1rem;background-repeat:no-repeat;opacity:.5;transition:opacity var(--transition)}dialog article .close:is([aria-current],:hover,:active,:focus){opacity:1}dialog:not([open]),dialog[open=false]{display:none}.modal-is-open{padding-right:var(--scrollbar-width,0);overflow:hidden;pointer-events:none;touch-action:none}.modal-is-open dialog{pointer-events:auto}:where(.modal-is-opening,.modal-is-closing) dialog,:where(.modal-is-opening,.modal-is-closing) dialog>article{animation-duration:.2s;animation-timing-function:ease-in-out;animation-fill-mode:both}:where(.modal-is-opening,.modal-is-closing) dialog{animation-duration:.8s;animation-name:modal-overlay}:where(.modal-is-opening,.modal-is-closing) dialog>article{animation-delay:.2s;animation-name:modal}.modal-is-closing dialog,.modal-is-closing dialog>article{animation-delay:0s;animation-direction:reverse}@keyframes modal-overlay{from{-webkit-backdrop-filter:none;backdrop-filter:none;background-color:transparent}}@keyframes modal{from{transform:translateY(-100%);opacity:0}}:where(nav li)::before{float:left;content:"​"}nav,nav ul{display:flex}nav{justify-content:space-between}nav ol,nav ul{align-items:center;margin-bottom:0;padding:0;list-style:none}nav ol:first-of-type,nav ul:first-of-type{margin-left:calc(var(--nav-element-spacing-horizontal) * -1)}nav ol:last-of-type,nav ul:last-of-type{margin-right:calc(var(--nav-element-spacing-horizontal) * -1)}nav li{display:inline-block;margin:0;padding:var(--nav-element-spacing-vertical) var(--nav-element-spacing-horizontal)}nav li>*{--spacing:0}nav :where(a,[role=link]){display:inline-block;margin:calc(var(--nav-link-spacing-vertical) * -1) calc(var(--nav-link-spacing-horizontal) * -1);padding:var(--nav-link-spacing-vertical) var(--nav-link-spacing-horizontal);border-radius:var(--border-radius);text-decoration:none}nav :where(a,[role=link]):is([aria-current],:hover,:active,:focus){text-decoration:none}nav[aria-label=breadcrumb]{align-items:center;justify-content:start}nav[aria-label=breadcrumb] ul li:not(:first-child){-webkit-margin-start:var(--nav-link-spacing-horizontal);margin-inline-start:var(--nav-link-spacing-horizontal)}nav[aria-label=breadcrumb] ul li:not(:last-child) ::after{position:absolute;width:calc(var(--nav-link-spacing-horizontal) * 2);-webkit-margin-start:calc(var(--nav-link-spacing-horizontal)/ 2);margin-inline-start:calc(var(--nav-link-spacing-horizontal)/ 2);content:"/";color:var(--muted-color);text-align:center}nav[aria-label=breadcrumb] a[aria-current]{background-color:transparent;color:inherit;text-decoration:none;pointer-events:none}nav [role=button]{margin-right:inherit;margin-left:inherit;padding:var(--nav-link-spacing-vertical) var(--nav-link-spacing-horizontal)}aside li,aside nav,aside ol,aside ul{display:block}aside li{padding:calc(var(--nav-element-spacing-vertical) * .5) var(--nav-element-spacing-horizontal)}aside li a{display:block}aside li [role=button]{margin:inherit}[dir=rtl] nav[aria-label=breadcrumb] ul li:not(:last-child) ::after{content:"\\"}progress{display:inline-block;vertical-align:baseline}progress{-webkit-appearance:none;-moz-appearance:none;display:inline-block;appearance:none;width:100%;height:.5rem;margin-bottom:calc(var(--spacing) * .5);overflow:hidden;border:0;border-radius:var(--border-radius);background-color:var(--progress-background-color);color:var(--progress-color)}progress::-webkit-progress-bar{border-radius:var(--border-radius);background:0 0}progress[value]::-webkit-progress-value{background-color:var(--progress-color)}progress::-moz-progress-bar{background-color:var(--progress-color)}@media (prefers-reduced-motion:no-preference){progress:indeterminate{background:var(--progress-background-color) linear-gradient(to right,var(--progress-color) 30%,var(--progress-background-color) 30%) top left/150% 150% no-repeat;animation:progress-indeterminate 1s linear infinite}progress:indeterminate[value]::-webkit-progress-value{background-color:transparent}progress:indeterminate::-moz-progress-bar{background-color:transparent}}@media (prefers-reduced-motion:no-preference){[dir=rtl] progress:indeterminate{animation-direction:reverse}}@keyframes progress-indeterminate{0%{background-position:200% 0}100%{background-position:-200% 0}}details[role=list],li[role=list]{position:relative}details[role=list] summary+ul,li[role=list]>ul{display:flex;z-index:99;position:absolute;top:auto;right:0;left:0;flex-direction:column;margin:0;padding:0;border:var(--border-width) solid var(--dropdown-border-color);border-radius:var(--border-radius);border-top-right-radius:0;border-top-left-radius:0;background-color:var(--dropdown-background-color);box-shadow:var(--card-box-shadow);color:var(--dropdown-color);white-space:nowrap}details[role=list] summary+ul li,li[role=list]>ul li{width:100%;margin-bottom:0;padding:calc(var(--form-element-spacing-vertical) * .5) var(--form-element-spacing-horizontal);list-style:none}details[role=list] summary+ul li:first-of-type,li[role=list]>ul li:first-of-type{margin-top:calc(var(--form-element-spacing-vertical) * .5)}details[role=list] summary+ul li:last-of-type,li[role=list]>ul li:last-of-type{margin-bottom:calc(var(--form-element-spacing-vertical) * .5)}details[role=list] summary+ul li a,li[role=list]>ul li a{display:block;margin:calc(var(--form-element-spacing-vertical) * -.5) calc(var(--form-element-spacing-horizontal) * -1);padding:calc(var(--form-element-spacing-vertical) * .5) var(--form-element-spacing-horizontal);overflow:hidden;color:var(--dropdown-color);text-decoration:none;text-overflow:ellipsis}details[role=list] summary+ul li a:hover,li[role=list]>ul li a:hover{background-color:var(--dropdown-hover-background-color)}details[role=list] summary::after,li[role=list]>a::after{display:block;width:1rem;height:calc(1rem * var(--line-height,1.5));-webkit-margin-start:0.5rem;margin-inline-start:.5rem;float:right;transform:rotate(0);background-position:right center;background-size:1rem auto;background-repeat:no-repeat;content:""}details[role=list]{padding:0;border-bottom:none}details[role=list] summary{margin-bottom:0}details[role=list] summary:not([role]){height:calc(1rem * var(--line-height) + var(--form-element-spacing-vertical) * 2 + var(--border-width) * 2);padding:var(--form-element-spacing-vertical) var(--form-element-spacing-horizontal);border:var(--border-width) solid var(--form-element-border-color);border-radius:var(--border-radius);background-color:var(--form-element-background-color);color:var(--form-element-placeholder-color);line-height:inherit;cursor:pointer;transition:background-color var(--transition),border-color var(--transition),color var(--transition),box-shadow var(--transition)}details[role=list] summary:not([role]):active,details[role=list] summary:not([role]):focus{border-color:var(--form-element-active-border-color);background-color:var(--form-element-active-background-color)}details[role=list] summary:not([role]):focus{box-shadow:0 0 0 var(--outline-width) var(--form-element-focus-color)}details[role=list][open] summary{border-bottom-right-radius:0;border-bottom-left-radius:0}details[role=list][open] summary::before{display:block;z-index:1;position:fixed;top:0;right:0;bottom:0;left:0;background:0 0;content:"";cursor:default}nav details[role=list] summary,nav li[role=list] a{display:flex;direction:ltr}nav details[role=list] summary+ul,nav li[role=list]>ul{min-width:-moz-fit-content;min-width:fit-content;border-radius:var(--border-radius)}nav details[role=list] summary+ul li a,nav li[role=list]>ul li a{border-radius:0}nav details[role=list] summary,nav details[role=list] summary:not([role]){height:auto;padding:var(--nav-link-spacing-vertical) var(--nav-link-spacing-horizontal)}nav details[role=list][open] summary{border-radius:var(--border-radius)}nav details[role=list] summary+ul{margin-top:var(--outline-width);-webkit-margin-start:0;margin-inline-start:0}nav details[role=list] summary[role=link]{margin-bottom:calc(var(--nav-link-spacing-vertical) * -1);line-height:var(--line-height)}nav details[role=list] summary[role=link]+ul{margin-top:calc(var(--nav-link-spacing-vertical) + var(--outline-width));-webkit-margin-start:calc(var(--nav-link-spacing-horizontal) * -1);margin-inline-start:calc(var(--nav-link-spacing-horizontal) * -1)}li[role=list] a:active~ul,li[role=list] a:focus~ul,li[role=list]:hover>ul{display:flex}li[role=list]>ul{display:none;margin-top:calc(var(--nav-link-spacing-vertical) + var(--outline-width));-webkit-margin-start:calc(var(--nav-element-spacing-horizontal) - var(--nav-link-spacing-horizontal));margin-inline-start:calc(var(--nav-element-spacing-horizontal) - var(--nav-link-spacing-horizontal))}li[role=list]>a::after{background-image:var(--icon-chevron)}label>details[role=list]{margin-top:calc(var(--spacing) * .25);margin-bottom:var(--spacing)}[aria-busy=true]{cursor:progress}[aria-busy=true]:not(input,select,textarea,html)::before{display:inline-block;width:1em;height:1em;border:.1875em solid currentColor;border-radius:1em;border-right-color:transparent;content:"";vertical-align:text-bottom;vertical-align:-.125em;animation:spinner .75s linear infinite;opacity:var(--loading-spinner-opacity)}[aria-busy=true]:not(input,select,textarea,html):not(:empty)::before{margin-right:calc(var(--spacing) * .5);margin-left:0;-webkit-margin-start:0;margin-inline-start:0;-webkit-margin-end:calc(var(--spacing) * .5);margin-inline-end:calc(var(--spacing) * .5)}[aria-busy=true]:not(input,select,textarea,html):empty{text-align:center}a[aria-busy=true],button[aria-busy=true],input[type=button][aria-busy=true],input[type=reset][aria-busy=true],input[type=submit][aria-busy=true]{pointer-events:none}@keyframes spinner{to{transform:rotate(360deg)}}[data-tooltip]{position:relative}[data-tooltip]:not(a,button,input){border-bottom:1px dotted;text-decoration:none;cursor:help}[data-tooltip]::after,[data-tooltip]::before,[data-tooltip][data-placement=top]::after,[data-tooltip][data-placement=top]::before{display:block;z-index:99;position:absolute;bottom:100%;left:50%;padding:.25rem .5rem;overflow:hidden;transform:translate(-50%,-.25rem);border-radius:var(--border-radius);background:var(--tooltip-background-color);content:attr(data-tooltip);color:var(--tooltip-color);font-style:normal;font-weight:var(--font-weight);font-size:.875rem;text-decoration:none;text-overflow:ellipsis;white-space:nowrap;opacity:0;pointer-events:none}[data-tooltip]::after,[data-tooltip][data-placement=top]::after{padding:0;transform:translate(-50%,0);border-top:.3rem solid;border-right:.3rem solid transparent;border-left:.3rem solid transparent;border-radius:0;background-color:transparent;content:"";color:var(--tooltip-background-color)}[data-tooltip][data-placement=bottom]::after,[data-tooltip][data-placement=bottom]::before{top:100%;bottom:auto;transform:translate(-50%,.25rem)}[data-tooltip][data-placement=bottom]:after{transform:translate(-50%,-.3rem);border:.3rem solid transparent;border-bottom:.3rem solid}[data-tooltip][data-placement=left]::after,[data-tooltip][data-placement=left]::before{top:50%;right:100%;bottom:auto;left:auto;transform:translate(-.25rem,-50%)}[data-tooltip][data-placement=left]:after{transform:translate(.3rem,-50%);border:.3rem solid transparent;border-left:.3rem solid}[data-tooltip][data-placement=right]::after,[data-tooltip][data-placement=right]::before{top:50%;right:auto;bottom:auto;left:100%;transform:translate(.25rem,-50%)}[data-tooltip][data-placement=right]:after{transform:translate(-.3rem,-50%);border:.3rem solid transparent;border-right:.3rem solid}[data-tooltip]:focus::after,[data-tooltip]:focus::before,[data-tooltip]:hover::after,[data-tooltip]:hover::before{opacity:1}@media (hover:hover) and (pointer:fine){[data-tooltip]:hover::after,[data-tooltip]:hover::before,[data-tooltip][data-placement=bottom]:focus::after,[data-tooltip][data-placement=bottom]:focus::before,[data-tooltip][data-placement=bottom]:hover [data-tooltip]:focus::after,[data-tooltip][data-placement=bottom]:hover [data-tooltip]:focus::before{animation-duration:.2s;animation-name:tooltip-slide-top}[data-tooltip]:hover::after,[data-tooltip][data-placement=bottom]:focus::after,[data-tooltip][data-placement=bottom]:hover [data-tooltip]:focus::after{animation-name:tooltip-caret-slide-top}[data-tooltip][data-placement=bottom]:focus::after,[data-tooltip][data-placement=bottom]:focus::before,[data-tooltip][data-placement=bottom]:hover::after,[data-tooltip][data-placement=bottom]:hover::before{animation-duration:.2s;animation-name:tooltip-slide-bottom}[data-tooltip][data-placement=bottom]:focus::after,[data-tooltip][data-placement=bottom]:hover::after{animation-name:tooltip-caret-slide-bottom}[data-tooltip][data-placement=left]:focus::after,[data-tooltip][data-placement=left]:focus::before,[data-tooltip][data-placement=left]:hover::after,[data-tooltip][data-placement=left]:hover::before{animation-duration:.2s;animation-name:tooltip-slide-left}[data-tooltip][data-placement=left]:focus::after,[data-tooltip][data-placement=left]:hover::after{animation-name:tooltip-caret-slide-left}[data-tooltip][data-placement=right]:focus::after,[data-tooltip][data-placement=right]:focus::before,[data-tooltip][data-placement=right]:hover::after,[data-tooltip][data-placement=right]:hover::before{animation-duration:.2s;animation-name:tooltip-slide-right}[data-tooltip][data-placement=right]:focus::after,[data-tooltip][data-placement=right]:hover::after{animation-name:tooltip-caret-slide-right}}@keyframes tooltip-slide-top{from{transform:translate(-50%,.75rem);opacity:0}to{transform:translate(-50%,-.25rem);opacity:1}}@keyframes tooltip-caret-slide-top{from{opacity:0}50%{transform:translate(-50%,-.25rem);opacity:0}to{transform:translate(-50%,0);opacity:1}}@keyframes tooltip-slide-bottom{from{transform:translate(-50%,-.75rem);opacity:0}to{transform:translate(-50%,.25rem);opacity:1}}@keyframes tooltip-caret-slide-bottom{from{opacity:0}50%{transform:translate(-50%,-.5rem);opacity:0}to{transform:translate(-50%,-.3rem);opacity:1}}@keyframes tooltip-slide-left{from{transform:translate(.75rem,-50%);opacity:0}to{transform:translate(-.25rem,-50%);opacity:1}}@keyframes tooltip-caret-slide-left{from{opacity:0}50%{transform:translate(.05rem,-50%);opacity:0}to{transform:translate(.3rem,-50%);opacity:1}}@keyframes tooltip-slide-right{from{transform:translate(-.75rem,-50%);opacity:0}to{transform:translate(.25rem,-50%);opacity:1}}@keyframes tooltip-caret-slide-right{from{opacity:0}50%{transform:translate(-.05rem,-50%);opacity:0}to{transform:translate(-.3rem,-50%);opacity:1}}[aria-controls]{cursor:pointer}[aria-disabled=true],[disabled]{cursor:not-allowed}[aria-hidden=false][hidden]{display:initial}[aria-hidden=false][hidden]:not(:focus){clip:rect(0,0,0,0);position:absolute}[tabindex],a,area,button,input,label,select,summary,textarea{-ms-touch-action:manipulation}[dir=rtl]{direction:rtl}@media (prefers-reduced-motion:reduce){:not([aria-busy=true]),:not([aria-busy=true])::after,:not([aria-busy=true])::before{background-attachment:initial!important;animation-duration:1ms!important;animation-delay:-1ms!important;animation-iteration-count:1!important;scroll-behavior:auto!important;transition-delay:0s!important;transition-duration:0s!important}} +/*# sourceMappingURL=pico.min.css.map */ diff --git a/src/interface/desktop/chat.html b/src/interface/desktop/chat.html new file mode 100644 index 00000000..21a1a416 --- /dev/null +++ b/src/interface/desktop/chat.html @@ -0,0 +1,559 @@ + + + + + Khoj - Chat + + + + + + + +
+
+ +
+ + +
+ + +
+ + + + + + + diff --git a/src/interface/desktop/config.html b/src/interface/desktop/config.html new file mode 100644 index 00000000..09610296 --- /dev/null +++ b/src/interface/desktop/config.html @@ -0,0 +1,342 @@ + + + + + Khoj - Search + + + + + + + + + +
+ +
+ + +
+
+
+
+
+ File +

+ Host +

+
+
+ +
+
+ File +

+ Files + +

+
+
+
+
+
+ Folder +

+ Folders + +

+
+
+
+
+
+ +
+
+ +
+
+
+
+
+
+ + + + + + + diff --git a/src/interface/desktop/index.html b/src/interface/desktop/index.html new file mode 100644 index 00000000..6b3c7e32 --- /dev/null +++ b/src/interface/desktop/index.html @@ -0,0 +1,499 @@ + + + + + Khoj - Search + + + + + + + + + + + + +
+ + +
+ + + + + +
+ + + + + + diff --git a/src/interface/desktop/main.js b/src/interface/desktop/main.js new file mode 100644 index 00000000..097b799c --- /dev/null +++ b/src/interface/desktop/main.js @@ -0,0 +1,359 @@ +const { app, BrowserWindow, ipcMain } = require('electron'); +const todesktop = require("@todesktop/runtime"); + +todesktop.init(); + +const fs = require('fs'); +const {dialog} = require('electron'); + +const cron = require('cron').CronJob; +const axios = require('axios'); +const { Readable } = require('stream'); + +const KHOJ_URL = 'http://127.0.0.1:42110' + +const Store = require('electron-store'); + +const validFileTypes = ['org', 'md', 'markdown', 'txt', 'html', 'xml', 'pdf'] + +const binaryFileTypes = ['pdf', 'png', 'jpg', 'jpeg'] + +const schema = { + files: { + type: 'array', + items: { + type: 'object', + properties: { + path: { + type: 'string' + } + } + }, + default: [] + }, + folders: { + type: 'array', + items: { + type: 'object', + properties: { + path: { + type: 'string' + } + } + }, + default: [] + }, + hostURL: { + type: 'string', + default: KHOJ_URL + }, + lastSync: { + type: 'array', + items: { + type: 'object', + properties: { + path: { + type: 'string' + }, + datetime: { + type: 'string' + } + } + } + } +}; + +var state = {} + +const store = new Store({schema}); + +console.log(store); + +// include the Node.js 'path' module at the top of your file +const path = require('path'); + +function handleSetTitle (event, title) { + const webContents = event.sender + const win = BrowserWindow.fromWebContents(webContents) + win.setTitle(title) + dialog.showOpenDialog({properties: ['openFile', 'openDirectory'] }).then(function (response) { + if (!response.canceled) { + // handle fully qualified file name + console.log(response.filePaths[0]); + } else { + console.log("no file selected"); + } + }); +} + +function pushDataToKhoj () { + let filesToPush = []; + const files = store.get('files'); + const folders = store.get('folders'); + state = { + completed: true + } + + if (files) { + for (file of files) { + filesToPush.push(file.path); + } + } + if (folders) { + for (folder of folders) { + const files = fs.readdirSync(folder.path, { withFileTypes: true }); + for (file of files) { + if (file.isFile() && validFileTypes.includes(file.name.split('.').pop())) { + filesToPush.push(path.join(folder.path, file.name)); + } + } + } + } + + let data = { + files: [] + } + + const lastSync = store.get('lastSync') || []; + + for (file of filesToPush) { + const stats = fs.statSync(file); + if (stats.mtime.toISOString() < lastSync.find((syncedFile) => syncedFile.path === file)?.datetime) { + continue; + } + try { + let rawData; + // If the file is a PDF or IMG file, read it as a binary file + if (binaryFileTypes.includes(file.split('.').pop())) { + rawData = fs.readFileSync(file).toString('base64'); + } else { + rawData = fs.readFileSync(file, 'utf8'); + } + + data.files.push({ + path: file, + content: rawData + }); + state[file] = { + success: true, + } + } catch (err) { + console.error(err); + state[file] = { + success: false, + error: err + } + } + } + + for (const syncedFile of lastSync) { + if (!filesToPush.includes(syncedFile.path)) { + data.files.push({ + path: syncedFile.path, + content: "" + }); + } + } + + const headers = { 'x-api-key': 'secret', 'Content-Type': 'application/json' }; + + const stream = new Readable({ + read() { + this.push(JSON.stringify(data)); + this.push(null); + } + }); + + const hostURL = store.get('hostURL') || KHOJ_URL; + + axios.post(`${hostURL}/v1/indexer/batch`, stream, { headers }) + .then(response => { + console.log(response.data); + const win = BrowserWindow.getAllWindows()[0]; + win.webContents.send('update-state', state); + let lastSync = []; + for (const file of filesToPush) { + lastSync.push({ + path: file, + datetime: new Date().toISOString() + }); + } + store.set('lastSync', lastSync); + }) + .catch(error => { + console.error(error); + state['completed'] = false + const win = BrowserWindow.getAllWindows()[0]; + win.webContents.send('update-state', state); + }); + +} + +pushDataToKhoj(); + +async function handleFileOpen (event, key) { + const { canceled, filePaths } = await dialog.showOpenDialog({properties: ['openFile', 'openDirectory'], filters: [{ name: "Valid Khoj Files", extensions: validFileTypes}] }); + if (!canceled) { + const files = store.get('files') || []; + const folders = store.get('folders') || []; + + for (const filePath of filePaths) { + console.log(filePath); + if (fs.existsSync(filePath)) { + const stats = fs.statSync(filePath); + if (stats.isFile()) { + console.log(`${filePath} is a file.`); + + if (files.find((file) => file.path === filePath)) { + continue; + } + + files.push({path: filePath}); + store.set('files', files); + } else if (stats.isDirectory()) { + console.log(`${filePath} is a directory.`); + + if (folders.find((folder) => folder.path === filePath)) { + continue; + } + + folders.push({path: filePath}); + store.set('folders', folders); + } + + } else { + console.log(`${filePath} does not exist.`); + } + } + return { + files: store.get('files'), + folders: store.get('folders') + } + } +} + +async function getFiles () { + return store.get('files'); +} + +async function getFolders () { + return store.get('folders'); +} + +async function setURL (event, url) { + store.set('hostURL', url); + return store.get('hostURL'); +} + +async function getURL () { + return store.get('hostURL'); +} + +async function removeFile (event, filePath) { + const files = store.get('files'); + const newFiles = files.filter((file) => file.path !== filePath); + store.set('files', newFiles); + return newFiles; +} + +async function removeFolder (event, folderPath) { + const folders = store.get('folders'); + const newFolders = folders.filter((folder) => folder.path !== folderPath); + store.set('folders', newFolders); + return newFolders; +} + +async function syncData () { + try { + pushDataToKhoj(); + const date = new Date(); + console.log('Pushing data to Khoj at: ', date); + } catch (err) { + console.error(err); + } +} + +const createWindow = () => { + const win = new BrowserWindow({ + width: 800, + height: 800, + // titleBarStyle: 'hidden', + webPreferences: { + preload: path.join(__dirname, 'preload.js'), + nodeIntegration: true, + } + }) + + const job = new cron('0 */10 * * * *', function() { + try { + pushDataToKhoj(); + const date = new Date(); + console.log('Pushing data to Khoj at: ', date); + win.webContents.send('update-state', state); + } catch (err) { + console.error(err); + } + }); + + win.setResizable(true); + win.setOpacity(0.95); + win.setBackgroundColor('#FFFFFF'); + win.setHasShadow(true); + + job.start(); + + win.loadFile('index.html') +} + +app.whenReady().then(() => { + ipcMain.on('set-title', handleSetTitle); + + ipcMain.handle('getStoreValue', handleFileOpen); + + ipcMain.on('update-state', (event, arg) => { + console.log(arg); + event.reply('update-state', arg); + }); + + ipcMain.handle('getFiles', getFiles); + ipcMain.handle('getFolders', getFolders); + + ipcMain.handle('removeFile', removeFile); + ipcMain.handle('removeFolder', removeFolder); + + ipcMain.handle('setURL', setURL); + ipcMain.handle('getURL', getURL); + + ipcMain.handle('syncData', syncData); + + createWindow() + + app.setAboutPanelOptions({ + applicationName: "Khoj", + applicationVersion: "0.0.1", + version: "0.0.1", + authors: "Khoj Team", + website: "https://khoj.dev", + iconPath: path.join(__dirname, 'assets', 'khoj.png') + }); + + app.on('ready', async() => { + try { + const result = await todesktop.autoUpdater.checkForUpdates(); + if (result.updateInfo) { + console.log("Update found:", result.updateInfo.version); + todesktop.autoUpdater.restartAndInstall(); + } + } catch (e) { + console.log("Update check failed:", e); + } + }) + + app.on('activate', () => { + if (BrowserWindow.getAllWindows().length === 0) createWindow() + }) +}) + +app.on('window-all-closed', () => { + if (process.platform !== 'darwin') app.quit() +}) diff --git a/src/interface/desktop/package.json b/src/interface/desktop/package.json new file mode 100644 index 00000000..34e27d09 --- /dev/null +++ b/src/interface/desktop/package.json @@ -0,0 +1,25 @@ +{ + "name": "Khoj", + "homepage": ".", + "productName": "Khoj", + "version": "1.0.0", + "description": "Scaffolding for the desktop entrypoint to Khoj", + "main": "main.js", + "repository": "\"https://github.com/khoj-ai/khoj\"", + "author": "Khoj ", + "license": "MIT", + "private": false, + "devDependencies": { + "electron": "26.1.0" + }, + "scripts": { + "start": "yarn electron ." + }, + "dependencies": { + "@todesktop/runtime": "^1.3.0", + "axios": "^1.5.0", + "cron": "^2.4.3", + "electron-store": "^8.1.0", + "fs": "^0.0.1-security" + } +} diff --git a/src/interface/desktop/preload.js b/src/interface/desktop/preload.js new file mode 100644 index 00000000..8b4ec59e --- /dev/null +++ b/src/interface/desktop/preload.js @@ -0,0 +1,49 @@ +window.addEventListener('DOMContentLoaded', () => { + const replaceText = (selector, text) => { + const element = document.getElementById(selector) + if (element) element.innerText = text + } + + for (const dependency of ['chrome', 'node', 'electron']) { + replaceText(`${dependency}-version`, process.versions[dependency]) + } +}) + +const { contextBridge, ipcRenderer } = require('electron'); + +contextBridge.exposeInMainWorld('electronAPI', { + setTitle: (title) => ipcRenderer.send('set-title', title) +}) + +contextBridge.exposeInMainWorld('storeValueAPI', { + getStoreValue: (key) => ipcRenderer.invoke('getStoreValue', key) +}) + +contextBridge.exposeInMainWorld('getFilesAPI', { + getFiles: () => ipcRenderer.invoke('getFiles') +}) + +contextBridge.exposeInMainWorld('getFoldersAPI', { + getFolders: () => ipcRenderer.invoke('getFolders') +}) + +contextBridge.exposeInMainWorld('updateStateAPI', { + onUpdateState: (callback) => ipcRenderer.on('update-state', callback) +}) + +contextBridge.exposeInMainWorld('removeFileAPI', { + removeFile: (filePath) => ipcRenderer.invoke('removeFile', filePath) +}) + +contextBridge.exposeInMainWorld('removeFolderAPI', { + removeFolder: (folderPath) => ipcRenderer.invoke('removeFolder', folderPath) +}) + +contextBridge.exposeInMainWorld('hostURLAPI', { + setURL: (url) => ipcRenderer.invoke('setURL', url), + getURL: () => ipcRenderer.invoke('getURL') +}) + +contextBridge.exposeInMainWorld('syncDataAPI', { + syncData: () => ipcRenderer.invoke('syncData') +}) diff --git a/src/interface/desktop/renderer.js b/src/interface/desktop/renderer.js new file mode 100644 index 00000000..a60805cb --- /dev/null +++ b/src/interface/desktop/renderer.js @@ -0,0 +1,177 @@ +const getButton = document.getElementById('update-data') +const showKey = document.getElementById('show-key'); + +async function removeFile(filePath) { + const updatedFiles = await window.removeFileAPI.removeFile(filePath); + + let currentFilesElement = document.getElementById("current-files"); + currentFilesElement.innerHTML = ''; + for (const file of updatedFiles) { + console.log(file); + let fileElement = makeFileElement(file); + currentFilesElement.appendChild(fileElement); + } +} + +async function removeFolder(folderPath) { + const updatedFolders = await window.removeFolderAPI.removeFolder(folderPath); + + let currentFoldersElement = document.getElementById("current-folders"); + currentFoldersElement.innerHTML = ''; + for (const folder of updatedFolders) { + console.log(folder); + let folderElement = makeFolderElement(folder); + currentFoldersElement.appendChild(folderElement); + } +} + +const toggleFilesButton = document.getElementById('toggle-files'); +const currentFiles = document.getElementById('current-files'); + +const toggleFilesSVG = document.getElementById('toggle-files-svg'); + +toggleFilesButton.addEventListener('click', () => { + if (currentFiles.style.display === 'none') { + currentFiles.style.display = 'block'; + toggleFilesSVG.style.transform = 'rotate(0deg)'; + } else { + currentFiles.style.display = 'none'; + toggleFilesSVG.style.transform = 'rotate(180deg)'; + } +}); + +const toggleFoldersButton = document.getElementById('toggle-folders'); +const currentFolders = document.getElementById('current-folders'); + +const toggleFoldersSVG = document.getElementById('toggle-folders-svg'); + + +toggleFoldersButton.addEventListener('click', () => { + if (currentFolders.style.display === 'none') { + currentFolders.style.display = 'block'; + toggleFoldersSVG.style.transform = 'rotate(0deg)'; + } else { + currentFolders.style.display = 'none'; + toggleFoldersSVG.style.transform = 'rotate(180deg)'; + } +}); + +function makeFileElement(file) { + let fileElement = document.createElement("div"); + fileElement.classList.add("file-element"); + let fileNameElement = document.createElement("div"); + fileNameElement.classList.add("content-name"); + fileNameElement.innerHTML = file.path; + fileElement.appendChild(fileNameElement); + + let buttonContainer = document.createElement("div"); + buttonContainer.classList.add("remove-button-container"); + let removeFileButton = document.createElement("button"); + removeFileButton.classList.add("remove-file-button"); + removeFileButton.innerHTML = "🗑️"; + removeFileButton.addEventListener("click", () => { + removeFile(file.path); + }); + buttonContainer.appendChild(removeFileButton); + fileElement.appendChild(buttonContainer); + return fileElement; +} + +function makeFolderElement(folder) { + let folderElement = document.createElement("div"); + folderElement.classList.add("folder-element"); + let folderNameElement = document.createElement("div"); + folderNameElement.classList.add("content-name"); + folderNameElement.innerHTML = folder.path; + folderElement.appendChild(folderNameElement); + + let buttonContainer = document.createElement("div"); + buttonContainer.classList.add("remove-button-container"); + let removeFolderButton = document.createElement("button"); + removeFolderButton.classList.add("remove-folder-button"); + removeFolderButton.innerHTML = "🗑️"; + removeFolderButton.addEventListener("click", () => { + removeFolder(folder.path); + }); + buttonContainer.appendChild(removeFolderButton); + folderElement.appendChild(buttonContainer); + return folderElement; +} + +(async function() { + const files = await window.getFilesAPI.getFiles(); + let currentFilesElement = document.getElementById("current-files"); + for (const file of files) { + console.log(file); + let fileElement = makeFileElement(file); + currentFilesElement.appendChild(fileElement); + } + + const folders = await window.getFoldersAPI.getFolders(); + let currentFoldersElement = document.getElementById("current-folders"); + for (const folder of folders) { + let folderElement = makeFolderElement(folder); + currentFoldersElement.appendChild(folderElement); + } +})(); + +getButton.addEventListener('click', async () => { + const key = 'foo'; + const value = await window.storeValueAPI.getStoreValue(key); + console.log(value); + let currentFilesElement = document.getElementById("current-files"); + let currentFoldersElement = document.getElementById("current-folders"); + + if (value.files) { + currentFilesElement.innerHTML = ''; + value.files.forEach((file) => { + let fileElement = makeFileElement(file); + currentFilesElement.appendChild(fileElement); + }); + } + + if (value.folders) { + currentFoldersElement.innerHTML = ''; + value.folders.forEach((folder) => { + let folderElement = makeFolderElement(folder); + currentFoldersElement.appendChild(folderElement); + }); + } +}); + +window.updateStateAPI.onUpdateState((event, state) => { + console.log("state was updated", state); + let syncStatusElement = document.getElementById("sync-status"); + const currentTime = new Date(); + if (state.completed == false) { + syncStatusElement.innerHTML = `Sync was unsuccessful at ${currentTime.toLocaleTimeString()}. Contact team@khoj.dev to report this issue.`; + return; + } + syncStatusElement.innerHTML = `Last synced at ${currentTime.toLocaleTimeString()}`; +}); + +const urlInput = document.getElementById('khoj-host-url'); +(async function() { + const url = await window.hostURLAPI.getURL(); + urlInput.value = url; +})(); + +urlInput.addEventListener('blur', async () => { + const urlInputValue = urlInput.value; + + // Check if it's a valid URL + try { + new URL(urlInputValue); + } catch (e) { + console.log(e); + return; + } + + const url = await window.hostURLAPI.setURL(urlInput.value.trim()); + urlInput.value = url; +}); + +const syncButton = document.getElementById('sync-data'); +syncButton.addEventListener('click', async () => { + await window.syncDataAPI.syncData(); +}); diff --git a/src/interface/desktop/todesktop.json b/src/interface/desktop/todesktop.json new file mode 100644 index 00000000..3313a0f5 --- /dev/null +++ b/src/interface/desktop/todesktop.json @@ -0,0 +1,6 @@ +{ + "id": "", + "icon": "./assets/icons/favicon-128x128.png", + "appPath": ".", + "schemaVersion": 1 +} diff --git a/src/interface/desktop/yarn.lock b/src/interface/desktop/yarn.lock new file mode 100644 index 00000000..9fe6be69 --- /dev/null +++ b/src/interface/desktop/yarn.lock @@ -0,0 +1,1205 @@ +# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. +# yarn lockfile v1 + + +"@electron/get@^2.0.0": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@electron/get/-/get-2.0.2.tgz#ae2a967b22075e9c25aaf00d5941cd79c21efd7e" + integrity sha512-eFZVFoRXb3GFGd7Ak7W4+6jBl9wBtiZ4AaYOse97ej6mKj5tkyO0dUnUChs1IhJZtx1BENo4/p4WUTXpi6vT+g== + dependencies: + debug "^4.1.1" + env-paths "^2.2.0" + fs-extra "^8.1.0" + got "^11.8.5" + progress "^2.0.3" + semver "^6.2.0" + sumchecker "^3.0.1" + optionalDependencies: + global-agent "^3.0.0" + +"@nodelib/fs.scandir@2.1.5": + version "2.1.5" + resolved "https://registry.yarnpkg.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz#7619c2eb21b25483f6d167548b4cfd5a7488c3d5" + integrity sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g== + dependencies: + "@nodelib/fs.stat" "2.0.5" + run-parallel "^1.1.9" + +"@nodelib/fs.stat@2.0.5", "@nodelib/fs.stat@^2.0.2": + version "2.0.5" + resolved "https://registry.yarnpkg.com/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz#5bd262af94e9d25bd1e71b05deed44876a222e8b" + integrity sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A== + +"@nodelib/fs.walk@^1.2.3": + version "1.2.8" + resolved "https://registry.yarnpkg.com/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz#e95737e8bb6746ddedf69c556953494f196fe69a" + integrity sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg== + dependencies: + "@nodelib/fs.scandir" "2.1.5" + fastq "^1.6.0" + +"@sindresorhus/is@^4.0.0": + version "4.6.0" + resolved "https://registry.yarnpkg.com/@sindresorhus/is/-/is-4.6.0.tgz#3c7c9c46e678feefe7a2e5bb609d3dbd665ffb3f" + integrity sha512-t09vSN3MdfsyCHoFcTRCH/iUtG7OJ0CsjzB8cjAmKc/va/kIgeDI/TxsigdncE/4be734m0cvIYwNaV4i2XqAw== + +"@szmarczak/http-timer@^4.0.5": + version "4.0.6" + resolved "https://registry.yarnpkg.com/@szmarczak/http-timer/-/http-timer-4.0.6.tgz#b4a914bb62e7c272d4e5989fe4440f812ab1d807" + integrity sha512-4BAffykYOgO+5nzBWYwE3W90sBgLJoUPRWWcL8wlyiM8IB8ipJz3UMJ9KXQd1RKQXpKp8Tutn80HZtWsu2u76w== + dependencies: + defer-to-connect "^2.0.0" + +"@todesktop/runtime@^1.3.0": + version "1.3.0" + resolved "https://registry.yarnpkg.com/@todesktop/runtime/-/runtime-1.3.0.tgz#7baa64fd5c2e4daa591bda96270a0e39947ec3c7" + integrity sha512-a5USs4VxnqvtNqFR6F3bCaQ56W6WFO4VOPPaXefCYiCxcsFMYb4IulXGkYjvcpkU/MFGWzmVnzba6UwK7eQMUQ== + dependencies: + del "^6.0.0" + electron-updater "^4.6.1" + eventemitter2 "^6.4.5" + execa "^5.0.0" + lodash.once "^4.1.1" + semver "^7.3.2" + +"@types/cacheable-request@^6.0.1": + version "6.0.3" + resolved "https://registry.yarnpkg.com/@types/cacheable-request/-/cacheable-request-6.0.3.tgz#a430b3260466ca7b5ca5bfd735693b36e7a9d183" + integrity sha512-IQ3EbTzGxIigb1I3qPZc1rWJnH0BmSKv5QYTalEwweFvyBDLSAe24zP0le/hyi7ecGfZVlIVAg4BZqb8WBwKqw== + dependencies: + "@types/http-cache-semantics" "*" + "@types/keyv" "^3.1.4" + "@types/node" "*" + "@types/responselike" "^1.0.0" + +"@types/http-cache-semantics@*": + version "4.0.1" + resolved "https://registry.yarnpkg.com/@types/http-cache-semantics/-/http-cache-semantics-4.0.1.tgz#0ea7b61496902b95890dc4c3a116b60cb8dae812" + integrity sha512-SZs7ekbP8CN0txVG2xVRH6EgKmEm31BOxA07vkFaETzZz1xh+cbt8BcI0slpymvwhx5dlFnQG2rTlPVQn+iRPQ== + +"@types/keyv@^3.1.4": + version "3.1.4" + resolved "https://registry.yarnpkg.com/@types/keyv/-/keyv-3.1.4.tgz#3ccdb1c6751b0c7e52300bcdacd5bcbf8faa75b6" + integrity sha512-BQ5aZNSCpj7D6K2ksrRCTmKRLEpnPvWDiLPfoGyhZ++8YtiK9d/3DBKPJgry359X/P1PfruyYwvnvwFjuEiEIg== + dependencies: + "@types/node" "*" + +"@types/luxon@~3.3.0": + version "3.3.1" + resolved "https://registry.yarnpkg.com/@types/luxon/-/luxon-3.3.1.tgz#08727da7d81ee6a6c702b9dc6c8f86be010eb4dc" + integrity sha512-XOS5nBcgEeP2PpcqJHjCWhUCAzGfXIU8ILOSLpx2FhxqMW9KdxgCGXNOEKGVBfveKtIpztHzKK5vSRVLyW/NqA== + +"@types/node@*": + version "20.5.7" + resolved "https://registry.yarnpkg.com/@types/node/-/node-20.5.7.tgz#4b8ecac87fbefbc92f431d09c30e176fc0a7c377" + integrity sha512-dP7f3LdZIysZnmvP3ANJYTSwg+wLLl8p7RqniVlV7j+oXSXAbt9h0WIBFmJy5inWZoX9wZN6eXx+YXd9Rh3RBA== + +"@types/node@^18.11.18": + version "18.17.12" + resolved "https://registry.yarnpkg.com/@types/node/-/node-18.17.12.tgz#c6bd7413a13e6ad9cfb7e97dd5c4e904c1821e50" + integrity sha512-d6xjC9fJ/nSnfDeU0AMDsaJyb1iHsqCSOdi84w4u+SlN/UgQdY5tRhpMzaFYsI4mnpvgTivEaQd0yOUhAtOnEQ== + +"@types/responselike@^1.0.0": + version "1.0.0" + resolved "https://registry.yarnpkg.com/@types/responselike/-/responselike-1.0.0.tgz#251f4fe7d154d2bad125abe1b429b23afd262e29" + integrity sha512-85Y2BjiufFzaMIlvJDvTTB8Fxl2xfLo4HgmHzVBz08w4wDePCTjYw66PdrolO0kzli3yam/YCgRufyo1DdQVTA== + dependencies: + "@types/node" "*" + +"@types/semver@^7.3.6": + version "7.5.1" + resolved "https://registry.yarnpkg.com/@types/semver/-/semver-7.5.1.tgz#0480eeb7221eb9bc398ad7432c9d7e14b1a5a367" + integrity sha512-cJRQXpObxfNKkFAZbJl2yjWtJCqELQIdShsogr1d2MilP8dKD9TE/nEKHkJgUNHdGKCQaf9HbIynuV2csLGVLg== + +"@types/yauzl@^2.9.1": + version "2.10.0" + resolved "https://registry.yarnpkg.com/@types/yauzl/-/yauzl-2.10.0.tgz#b3248295276cf8c6f153ebe6a9aba0c988cb2599" + integrity sha512-Cn6WYCm0tXv8p6k+A8PvbDG763EDpBoTzHdA+Q/MF6H3sapGjCm9NzoaJncJS9tUKSuCoDs9XHxYYsQDgxR6kw== + dependencies: + "@types/node" "*" + +aggregate-error@^3.0.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/aggregate-error/-/aggregate-error-3.1.0.tgz#92670ff50f5359bdb7a3e0d40d0ec30c5737687a" + integrity sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA== + dependencies: + clean-stack "^2.0.0" + indent-string "^4.0.0" + +ajv-formats@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/ajv-formats/-/ajv-formats-2.1.1.tgz#6e669400659eb74973bbf2e33327180a0996b520" + integrity sha512-Wx0Kx52hxE7C18hkMEggYlEifqWZtYaRgouJor+WMdPnQyEK13vgEWyVNup7SoeeoLMsr4kf5h6dOW11I15MUA== + dependencies: + ajv "^8.0.0" + +ajv@^8.0.0, ajv@^8.6.3: + version "8.12.0" + resolved "https://registry.yarnpkg.com/ajv/-/ajv-8.12.0.tgz#d1a0527323e22f53562c567c00991577dfbe19d1" + integrity sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA== + dependencies: + fast-deep-equal "^3.1.1" + json-schema-traverse "^1.0.0" + require-from-string "^2.0.2" + uri-js "^4.2.2" + +argparse@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/argparse/-/argparse-2.0.1.tgz#246f50f3ca78a3240f6c997e8a9bd1eac49e4b38" + integrity sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q== + +array-union@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/array-union/-/array-union-2.1.0.tgz#b798420adbeb1de828d84acd8a2e23d3efe85e8d" + integrity sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw== + +asynckit@^0.4.0: + version "0.4.0" + resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79" + integrity sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q== + +atomically@^1.7.0: + version "1.7.0" + resolved "https://registry.yarnpkg.com/atomically/-/atomically-1.7.0.tgz#c07a0458432ea6dbc9a3506fffa424b48bccaafe" + integrity sha512-Xcz9l0z7y9yQ9rdDaxlmaI4uJHf/T8g9hOEzJcsEqX2SjCj4J20uK7+ldkDHMbpJDK76wF7xEIgxc/vSlsfw5w== + +axios@^1.5.0: + version "1.5.0" + resolved "https://registry.yarnpkg.com/axios/-/axios-1.5.0.tgz#f02e4af823e2e46a9768cfc74691fdd0517ea267" + integrity sha512-D4DdjDo5CY50Qms0qGQTTw6Q44jl7zRwY7bthds06pUGfChBCTcQs+N743eFWGEd6pRTMd6A+I87aWyFV5wiZQ== + dependencies: + follow-redirects "^1.15.0" + form-data "^4.0.0" + proxy-from-env "^1.1.0" + +balanced-match@^1.0.0: + version "1.0.2" + resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee" + integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw== + +boolean@^3.0.1: + version "3.2.0" + resolved "https://registry.yarnpkg.com/boolean/-/boolean-3.2.0.tgz#9e5294af4e98314494cbb17979fa54ca159f116b" + integrity sha512-d0II/GO9uf9lfUHH2BQsjxzRJZBdsjgsBiW4BvhWk/3qoKwQFjIDVN19PfX8F2D/r9PCMTtLWjYVCFrpeYUzsw== + +brace-expansion@^1.1.7: + version "1.1.11" + resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd" + integrity sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA== + dependencies: + balanced-match "^1.0.0" + concat-map "0.0.1" + +braces@^3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.2.tgz#3454e1a462ee8d599e236df336cd9ea4f8afe107" + integrity sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A== + dependencies: + fill-range "^7.0.1" + +buffer-crc32@~0.2.3: + version "0.2.13" + resolved "https://registry.yarnpkg.com/buffer-crc32/-/buffer-crc32-0.2.13.tgz#0d333e3f00eac50aa1454abd30ef8c2a5d9a7242" + integrity sha512-VO9Ht/+p3SN7SKWqcrgEzjGbRSJYTx+Q1pTQC0wrWqHx0vpJraQ6GtHx8tvcg1rlK1byhU5gccxgOgj7B0TDkQ== + +builder-util-runtime@8.9.2: + version "8.9.2" + resolved "https://registry.yarnpkg.com/builder-util-runtime/-/builder-util-runtime-8.9.2.tgz#a9669ae5b5dcabfe411ded26678e7ae997246c28" + integrity sha512-rhuKm5vh7E0aAmT6i8aoSfEjxzdYEFX7zDApK+eNgOhjofnWb74d9SRJv0H/8nsgOkos0TZ4zxW0P8J4N7xQ2A== + dependencies: + debug "^4.3.2" + sax "^1.2.4" + +cacheable-lookup@^5.0.3: + version "5.0.4" + resolved "https://registry.yarnpkg.com/cacheable-lookup/-/cacheable-lookup-5.0.4.tgz#5a6b865b2c44357be3d5ebc2a467b032719a7005" + integrity sha512-2/kNscPhpcxrOigMZzbiWF7dz8ilhb/nIHU3EyZiXWXpeq/au8qJ8VhdftMkty3n7Gj6HIGalQG8oiBNB3AJgA== + +cacheable-request@^7.0.2: + version "7.0.4" + resolved "https://registry.yarnpkg.com/cacheable-request/-/cacheable-request-7.0.4.tgz#7a33ebf08613178b403635be7b899d3e69bbe817" + integrity sha512-v+p6ongsrp0yTGbJXjgxPow2+DL93DASP4kXCDKb8/bwRtt9OEF3whggkkDkGNzgcWy2XaF4a8nZglC7uElscg== + dependencies: + clone-response "^1.0.2" + get-stream "^5.1.0" + http-cache-semantics "^4.0.0" + keyv "^4.0.0" + lowercase-keys "^2.0.0" + normalize-url "^6.0.1" + responselike "^2.0.0" + +clean-stack@^2.0.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/clean-stack/-/clean-stack-2.2.0.tgz#ee8472dbb129e727b31e8a10a427dee9dfe4008b" + integrity sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A== + +clone-response@^1.0.2: + version "1.0.3" + resolved "https://registry.yarnpkg.com/clone-response/-/clone-response-1.0.3.tgz#af2032aa47816399cf5f0a1d0db902f517abb8c3" + integrity sha512-ROoL94jJH2dUVML2Y/5PEDNaSHgeOdSDicUyS7izcF63G6sTc/FTjLub4b8Il9S8S0beOfYt0TaA5qvFK+w0wA== + dependencies: + mimic-response "^1.0.0" + +combined-stream@^1.0.8: + version "1.0.8" + resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.8.tgz#c3d45a8b34fd730631a110a8a2520682b31d5a7f" + integrity sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg== + dependencies: + delayed-stream "~1.0.0" + +concat-map@0.0.1: + version "0.0.1" + resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" + integrity sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg== + +conf@^10.2.0: + version "10.2.0" + resolved "https://registry.yarnpkg.com/conf/-/conf-10.2.0.tgz#838e757be963f1a2386dfe048a98f8f69f7b55d6" + integrity sha512-8fLl9F04EJqjSqH+QjITQfJF8BrOVaYr1jewVgSRAEWePfxT0sku4w2hrGQ60BC/TNLGQ2pgxNlTbWQmMPFvXg== + dependencies: + ajv "^8.6.3" + ajv-formats "^2.1.1" + atomically "^1.7.0" + debounce-fn "^4.0.0" + dot-prop "^6.0.1" + env-paths "^2.2.1" + json-schema-typed "^7.0.3" + onetime "^5.1.2" + pkg-up "^3.1.0" + semver "^7.3.5" + +cron@^2.4.3: + version "2.4.3" + resolved "https://registry.yarnpkg.com/cron/-/cron-2.4.3.tgz#4e43d8d9a6373b8f28d876c4e9a47c14422d8652" + integrity sha512-YBvExkQYF7w0PxyeFLRyr817YVDhGxaCi5/uRRMqa4aWD3IFKRd+uNbpW1VWMdqQy8PZ7CElc+accXJcauPKzQ== + dependencies: + "@types/luxon" "~3.3.0" + luxon "~3.3.0" + +cross-spawn@^7.0.3: + version "7.0.3" + resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-7.0.3.tgz#f73a85b9d5d41d045551c177e2882d4ac85728a6" + integrity sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w== + dependencies: + path-key "^3.1.0" + shebang-command "^2.0.0" + which "^2.0.1" + +debounce-fn@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/debounce-fn/-/debounce-fn-4.0.0.tgz#ed76d206d8a50e60de0dd66d494d82835ffe61c7" + integrity sha512-8pYCQiL9Xdcg0UPSD3d+0KMlOjp+KGU5EPwYddgzQ7DATsg4fuUDjQtsYLmWjnk2obnNHgV3vE2Y4jejSOJVBQ== + dependencies: + mimic-fn "^3.0.0" + +debug@^4.1.0, debug@^4.1.1, debug@^4.3.2: + version "4.3.4" + resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865" + integrity sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ== + dependencies: + ms "2.1.2" + +decompress-response@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/decompress-response/-/decompress-response-6.0.0.tgz#ca387612ddb7e104bd16d85aab00d5ecf09c66fc" + integrity sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ== + dependencies: + mimic-response "^3.1.0" + +defer-to-connect@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/defer-to-connect/-/defer-to-connect-2.0.1.tgz#8016bdb4143e4632b77a3449c6236277de520587" + integrity sha512-4tvttepXG1VaYGrRibk5EwJd1t4udunSOVMdLSAL6mId1ix438oPwPZMALY41FCijukO1L0twNcGsdzS7dHgDg== + +define-properties@^1.1.3: + version "1.2.0" + resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.2.0.tgz#52988570670c9eacedd8064f4a990f2405849bd5" + integrity sha512-xvqAVKGfT1+UAvPwKTVw/njhdQ8ZhXK4lI0bCIuCMrp2up9nPnaDftrLtmpTazqd1o+UY4zgzU+avtMbDP+ldA== + dependencies: + has-property-descriptors "^1.0.0" + object-keys "^1.1.1" + +del@^6.0.0: + version "6.1.1" + resolved "https://registry.yarnpkg.com/del/-/del-6.1.1.tgz#3b70314f1ec0aa325c6b14eb36b95786671edb7a" + integrity sha512-ua8BhapfP0JUJKC/zV9yHHDW/rDoDxP4Zhn3AkA6/xT6gY7jYXJiaeyBZznYVujhZZET+UgcbZiQ7sN3WqcImg== + dependencies: + globby "^11.0.1" + graceful-fs "^4.2.4" + is-glob "^4.0.1" + is-path-cwd "^2.2.0" + is-path-inside "^3.0.2" + p-map "^4.0.0" + rimraf "^3.0.2" + slash "^3.0.0" + +delayed-stream@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619" + integrity sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ== + +detect-node@^2.0.4: + version "2.1.0" + resolved "https://registry.yarnpkg.com/detect-node/-/detect-node-2.1.0.tgz#c9c70775a49c3d03bc2c06d9a73be550f978f8b1" + integrity sha512-T0NIuQpnTvFDATNuHN5roPwSBG83rFsuO+MXXH9/3N1eFbn4wcPjttvjMLEPWJ0RGUYgQE7cGgS3tNxbqCGM7g== + +dir-glob@^3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/dir-glob/-/dir-glob-3.0.1.tgz#56dbf73d992a4a93ba1584f4534063fd2e41717f" + integrity sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA== + dependencies: + path-type "^4.0.0" + +dot-prop@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/dot-prop/-/dot-prop-6.0.1.tgz#fc26b3cf142b9e59b74dbd39ed66ce620c681083" + integrity sha512-tE7ztYzXHIeyvc7N+hR3oi7FIbf/NIjVP9hmAt3yMXzrQ072/fpjGLx2GxNxGxUl5V73MEqYzioOMoVhGMJ5cA== + dependencies: + is-obj "^2.0.0" + +electron-store@^8.1.0: + version "8.1.0" + resolved "https://registry.yarnpkg.com/electron-store/-/electron-store-8.1.0.tgz#46a398f2bd9aa83c4a9daaae28380e2b3b9c7597" + integrity sha512-2clHg/juMjOH0GT9cQ6qtmIvK183B39ZXR0bUoPwKwYHJsEF3quqyDzMFUAu+0OP8ijmN2CbPRAelhNbWUbzwA== + dependencies: + conf "^10.2.0" + type-fest "^2.17.0" + +electron-updater@^4.6.1: + version "4.6.5" + resolved "https://registry.yarnpkg.com/electron-updater/-/electron-updater-4.6.5.tgz#e9a75458bbfd6bb41a58a829839e150ad2eb2d3d" + integrity sha512-kdTly8O9mSZfm9fslc1mnCY+mYOeaYRy7ERa2Fed240u01BKll3aiupzkd07qKw69KvhBSzuHroIW3mF0D8DWA== + dependencies: + "@types/semver" "^7.3.6" + builder-util-runtime "8.9.2" + fs-extra "^10.0.0" + js-yaml "^4.1.0" + lazy-val "^1.0.5" + lodash.escaperegexp "^4.1.2" + lodash.isequal "^4.5.0" + semver "^7.3.5" + +electron@^26.1.0: + version "26.1.0" + resolved "https://registry.yarnpkg.com/electron/-/electron-26.1.0.tgz#d26fefba5a5c68069b07a117d87aee1c4e5d172d" + integrity sha512-qEh19H09Pysn3ibms5nZ0haIh5pFoOd7/5Ww7gzmAwDQOulRi8Sa2naeueOyIb1GKpf+6L4ix3iceYRAuA5r5Q== + dependencies: + "@electron/get" "^2.0.0" + "@types/node" "^18.11.18" + extract-zip "^2.0.1" + +end-of-stream@^1.1.0: + version "1.4.4" + resolved "https://registry.yarnpkg.com/end-of-stream/-/end-of-stream-1.4.4.tgz#5ae64a5f45057baf3626ec14da0ca5e4b2431eb0" + integrity sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q== + dependencies: + once "^1.4.0" + +env-paths@^2.2.0, env-paths@^2.2.1: + version "2.2.1" + resolved "https://registry.yarnpkg.com/env-paths/-/env-paths-2.2.1.tgz#420399d416ce1fbe9bc0a07c62fa68d67fd0f8f2" + integrity sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A== + +es6-error@^4.1.1: + version "4.1.1" + resolved "https://registry.yarnpkg.com/es6-error/-/es6-error-4.1.1.tgz#9e3af407459deed47e9a91f9b885a84eb05c561d" + integrity sha512-Um/+FxMr9CISWh0bi5Zv0iOD+4cFh5qLeks1qhAopKVAJw3drgKbKySikp7wGhDL0HPeaja0P5ULZrxLkniUVg== + +escape-string-regexp@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz#14ba83a5d373e3d311e5afca29cf5bfad965bf34" + integrity sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA== + +eventemitter2@^6.4.5: + version "6.4.9" + resolved "https://registry.yarnpkg.com/eventemitter2/-/eventemitter2-6.4.9.tgz#41f2750781b4230ed58827bc119d293471ecb125" + integrity sha512-JEPTiaOt9f04oa6NOkc4aH+nVp5I3wEjpHbIPqfgCdD5v5bUzy7xQqwcVO2aDQgOWhI28da57HksMrzK9HlRxg== + +execa@^5.0.0: + version "5.1.1" + resolved "https://registry.yarnpkg.com/execa/-/execa-5.1.1.tgz#f80ad9cbf4298f7bd1d4c9555c21e93741c411dd" + integrity sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg== + dependencies: + cross-spawn "^7.0.3" + get-stream "^6.0.0" + human-signals "^2.1.0" + is-stream "^2.0.0" + merge-stream "^2.0.0" + npm-run-path "^4.0.1" + onetime "^5.1.2" + signal-exit "^3.0.3" + strip-final-newline "^2.0.0" + +extract-zip@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/extract-zip/-/extract-zip-2.0.1.tgz#663dca56fe46df890d5f131ef4a06d22bb8ba13a" + integrity sha512-GDhU9ntwuKyGXdZBUgTIe+vXnWj0fppUEtMDL0+idd5Sta8TGpHssn/eusA9mrPr9qNDym6SxAYZjNvCn/9RBg== + dependencies: + debug "^4.1.1" + get-stream "^5.1.0" + yauzl "^2.10.0" + optionalDependencies: + "@types/yauzl" "^2.9.1" + +fast-deep-equal@^3.1.1: + version "3.1.3" + resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525" + integrity sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q== + +fast-glob@^3.2.9: + version "3.3.1" + resolved "https://registry.yarnpkg.com/fast-glob/-/fast-glob-3.3.1.tgz#784b4e897340f3dbbef17413b3f11acf03c874c4" + integrity sha512-kNFPyjhh5cKjrUltxs+wFx+ZkbRaxxmZ+X0ZU31SOsxCEtP9VPgtq2teZw1DebupL5GmDaNQ6yKMMVcM41iqDg== + dependencies: + "@nodelib/fs.stat" "^2.0.2" + "@nodelib/fs.walk" "^1.2.3" + glob-parent "^5.1.2" + merge2 "^1.3.0" + micromatch "^4.0.4" + +fastq@^1.6.0: + version "1.15.0" + resolved "https://registry.yarnpkg.com/fastq/-/fastq-1.15.0.tgz#d04d07c6a2a68fe4599fea8d2e103a937fae6b3a" + integrity sha512-wBrocU2LCXXa+lWBt8RoIRD89Fi8OdABODa/kEnyeyjS5aZO5/GNvI5sEINADqP/h8M29UHTHUb53sUu5Ihqdw== + dependencies: + reusify "^1.0.4" + +fd-slicer@~1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/fd-slicer/-/fd-slicer-1.1.0.tgz#25c7c89cb1f9077f8891bbe61d8f390eae256f1e" + integrity sha512-cE1qsB/VwyQozZ+q1dGxR8LBYNZeofhEdUNGSMbQD3Gw2lAzX9Zb3uIU6Ebc/Fmyjo9AWWfnn0AUCHqtevs/8g== + dependencies: + pend "~1.2.0" + +fill-range@^7.0.1: + version "7.0.1" + resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.0.1.tgz#1919a6a7c75fe38b2c7c77e5198535da9acdda40" + integrity sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ== + dependencies: + to-regex-range "^5.0.1" + +find-up@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/find-up/-/find-up-3.0.0.tgz#49169f1d7993430646da61ecc5ae355c21c97b73" + integrity sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg== + dependencies: + locate-path "^3.0.0" + +follow-redirects@^1.15.0: + version "1.15.2" + resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.15.2.tgz#b460864144ba63f2681096f274c4e57026da2c13" + integrity sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA== + +form-data@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/form-data/-/form-data-4.0.0.tgz#93919daeaf361ee529584b9b31664dc12c9fa452" + integrity sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww== + dependencies: + asynckit "^0.4.0" + combined-stream "^1.0.8" + mime-types "^2.1.12" + +fs-extra@^10.0.0: + version "10.1.0" + resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-10.1.0.tgz#02873cfbc4084dde127eaa5f9905eef2325d1abf" + integrity sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ== + dependencies: + graceful-fs "^4.2.0" + jsonfile "^6.0.1" + universalify "^2.0.0" + +fs-extra@^8.1.0: + version "8.1.0" + resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-8.1.0.tgz#49d43c45a88cd9677668cb7be1b46efdb8d2e1c0" + integrity sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g== + dependencies: + graceful-fs "^4.2.0" + jsonfile "^4.0.0" + universalify "^0.1.0" + +fs.realpath@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f" + integrity sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw== + +fs@^0.0.1-security: + version "0.0.1-security" + resolved "https://registry.yarnpkg.com/fs/-/fs-0.0.1-security.tgz#8a7bd37186b6dddf3813f23858b57ecaaf5e41d4" + integrity sha512-3XY9e1pP0CVEUCdj5BmfIZxRBTSDycnbqhIOGec9QYtmVH2fbLpj86CFWkrNOkt/Fvty4KZG5lTglL9j/gJ87w== + +function-bind@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.1.tgz#a56899d3ea3c9bab874bb9773b7c5ede92f4895d" + integrity sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A== + +get-intrinsic@^1.1.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/get-intrinsic/-/get-intrinsic-1.2.1.tgz#d295644fed4505fc9cde952c37ee12b477a83d82" + integrity sha512-2DcsyfABl+gVHEfCOaTrWgyt+tb6MSEGmKq+kI5HwLbIYgjgmMcV8KQ41uaKz1xxUcn9tJtgFbQUEVcEbd0FYw== + dependencies: + function-bind "^1.1.1" + has "^1.0.3" + has-proto "^1.0.1" + has-symbols "^1.0.3" + +get-stream@^5.1.0: + version "5.2.0" + resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-5.2.0.tgz#4966a1795ee5ace65e706c4b7beb71257d6e22d3" + integrity sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA== + dependencies: + pump "^3.0.0" + +get-stream@^6.0.0: + version "6.0.1" + resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-6.0.1.tgz#a262d8eef67aced57c2852ad6167526a43cbf7b7" + integrity sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg== + +glob-parent@^5.1.2: + version "5.1.2" + resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-5.1.2.tgz#869832c58034fe68a4093c17dc15e8340d8401c4" + integrity sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow== + dependencies: + is-glob "^4.0.1" + +glob@^7.1.3: + version "7.2.3" + resolved "https://registry.yarnpkg.com/glob/-/glob-7.2.3.tgz#b8df0fb802bbfa8e89bd1d938b4e16578ed44f2b" + integrity sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q== + dependencies: + fs.realpath "^1.0.0" + inflight "^1.0.4" + inherits "2" + minimatch "^3.1.1" + once "^1.3.0" + path-is-absolute "^1.0.0" + +global-agent@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/global-agent/-/global-agent-3.0.0.tgz#ae7cd31bd3583b93c5a16437a1afe27cc33a1ab6" + integrity sha512-PT6XReJ+D07JvGoxQMkT6qji/jVNfX/h364XHZOWeRzy64sSFr+xJ5OX7LI3b4MPQzdL4H8Y8M0xzPpsVMwA8Q== + dependencies: + boolean "^3.0.1" + es6-error "^4.1.1" + matcher "^3.0.0" + roarr "^2.15.3" + semver "^7.3.2" + serialize-error "^7.0.1" + +globalthis@^1.0.1: + version "1.0.3" + resolved "https://registry.yarnpkg.com/globalthis/-/globalthis-1.0.3.tgz#5852882a52b80dc301b0660273e1ed082f0b6ccf" + integrity sha512-sFdI5LyBiNTHjRd7cGPWapiHWMOXKyuBNX/cWJ3NfzrZQVa8GI/8cofCl74AOVqq9W5kNmguTIzJ/1s2gyI9wA== + dependencies: + define-properties "^1.1.3" + +globby@^11.0.1: + version "11.1.0" + resolved "https://registry.yarnpkg.com/globby/-/globby-11.1.0.tgz#bd4be98bb042f83d796f7e3811991fbe82a0d34b" + integrity sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g== + dependencies: + array-union "^2.1.0" + dir-glob "^3.0.1" + fast-glob "^3.2.9" + ignore "^5.2.0" + merge2 "^1.4.1" + slash "^3.0.0" + +got@^11.8.5: + version "11.8.6" + resolved "https://registry.yarnpkg.com/got/-/got-11.8.6.tgz#276e827ead8772eddbcfc97170590b841823233a" + integrity sha512-6tfZ91bOr7bOXnK7PRDCGBLa1H4U080YHNaAQ2KsMGlLEzRbk44nsZF2E1IeRc3vtJHPVbKCYgdFbaGO2ljd8g== + dependencies: + "@sindresorhus/is" "^4.0.0" + "@szmarczak/http-timer" "^4.0.5" + "@types/cacheable-request" "^6.0.1" + "@types/responselike" "^1.0.0" + cacheable-lookup "^5.0.3" + cacheable-request "^7.0.2" + decompress-response "^6.0.0" + http2-wrapper "^1.0.0-beta.5.2" + lowercase-keys "^2.0.0" + p-cancelable "^2.0.0" + responselike "^2.0.0" + +graceful-fs@^4.1.6, graceful-fs@^4.2.0, graceful-fs@^4.2.4: + version "4.2.11" + resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.11.tgz#4183e4e8bf08bb6e05bbb2f7d2e0c8f712ca40e3" + integrity sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ== + +has-property-descriptors@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/has-property-descriptors/-/has-property-descriptors-1.0.0.tgz#610708600606d36961ed04c196193b6a607fa861" + integrity sha512-62DVLZGoiEBDHQyqG4w9xCuZ7eJEwNmJRWw2VY84Oedb7WFcA27fiEVe8oUQx9hAUJ4ekurquucTGwsyO1XGdQ== + dependencies: + get-intrinsic "^1.1.1" + +has-proto@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/has-proto/-/has-proto-1.0.1.tgz#1885c1305538958aff469fef37937c22795408e0" + integrity sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg== + +has-symbols@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.0.3.tgz#bb7b2c4349251dce87b125f7bdf874aa7c8b39f8" + integrity sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A== + +has@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/has/-/has-1.0.3.tgz#722d7cbfc1f6aa8241f16dd814e011e1f41e8796" + integrity sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw== + dependencies: + function-bind "^1.1.1" + +http-cache-semantics@^4.0.0: + version "4.1.1" + resolved "https://registry.yarnpkg.com/http-cache-semantics/-/http-cache-semantics-4.1.1.tgz#abe02fcb2985460bf0323be664436ec3476a6d5a" + integrity sha512-er295DKPVsV82j5kw1Gjt+ADA/XYHsajl82cGNQG2eyoPkvgUhX+nDIyelzhIWbbsXP39EHcI6l5tYs2FYqYXQ== + +http2-wrapper@^1.0.0-beta.5.2: + version "1.0.3" + resolved "https://registry.yarnpkg.com/http2-wrapper/-/http2-wrapper-1.0.3.tgz#b8f55e0c1f25d4ebd08b3b0c2c079f9590800b3d" + integrity sha512-V+23sDMr12Wnz7iTcDeJr3O6AIxlnvT/bmaAAAP/Xda35C90p9599p0F1eHR/N1KILWSoWVAiOMFjBBXaXSMxg== + dependencies: + quick-lru "^5.1.1" + resolve-alpn "^1.0.0" + +human-signals@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/human-signals/-/human-signals-2.1.0.tgz#dc91fcba42e4d06e4abaed33b3e7a3c02f514ea0" + integrity sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw== + +ignore@^5.2.0: + version "5.2.4" + resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.2.4.tgz#a291c0c6178ff1b960befe47fcdec301674a6324" + integrity sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ== + +indent-string@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/indent-string/-/indent-string-4.0.0.tgz#624f8f4497d619b2d9768531d58f4122854d7251" + integrity sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg== + +inflight@^1.0.4: + version "1.0.6" + resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9" + integrity sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA== + dependencies: + once "^1.3.0" + wrappy "1" + +inherits@2: + version "2.0.4" + resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c" + integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== + +is-extglob@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/is-extglob/-/is-extglob-2.1.1.tgz#a88c02535791f02ed37c76a1b9ea9773c833f8c2" + integrity sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ== + +is-glob@^4.0.1: + version "4.0.3" + resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-4.0.3.tgz#64f61e42cbbb2eec2071a9dac0b28ba1e65d5084" + integrity sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg== + dependencies: + is-extglob "^2.1.1" + +is-number@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/is-number/-/is-number-7.0.0.tgz#7535345b896734d5f80c4d06c50955527a14f12b" + integrity sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng== + +is-obj@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/is-obj/-/is-obj-2.0.0.tgz#473fb05d973705e3fd9620545018ca8e22ef4982" + integrity sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w== + +is-path-cwd@^2.2.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/is-path-cwd/-/is-path-cwd-2.2.0.tgz#67d43b82664a7b5191fd9119127eb300048a9fdb" + integrity sha512-w942bTcih8fdJPJmQHFzkS76NEP8Kzzvmw92cXsazb8intwLqPibPPdXf4ANdKV3rYMuuQYGIWtvz9JilB3NFQ== + +is-path-inside@^3.0.2: + version "3.0.3" + resolved "https://registry.yarnpkg.com/is-path-inside/-/is-path-inside-3.0.3.tgz#d231362e53a07ff2b0e0ea7fed049161ffd16283" + integrity sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ== + +is-stream@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/is-stream/-/is-stream-2.0.1.tgz#fac1e3d53b97ad5a9d0ae9cef2389f5810a5c077" + integrity sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg== + +isexe@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" + integrity sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw== + +js-yaml@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-4.1.0.tgz#c1fb65f8f5017901cdd2c951864ba18458a10602" + integrity sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA== + dependencies: + argparse "^2.0.1" + +json-buffer@3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/json-buffer/-/json-buffer-3.0.1.tgz#9338802a30d3b6605fbe0613e094008ca8c05a13" + integrity sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ== + +json-schema-traverse@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz#ae7bcb3656ab77a73ba5c49bf654f38e6b6860e2" + integrity sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug== + +json-schema-typed@^7.0.3: + version "7.0.3" + resolved "https://registry.yarnpkg.com/json-schema-typed/-/json-schema-typed-7.0.3.tgz#23ff481b8b4eebcd2ca123b4fa0409e66469a2d9" + integrity sha512-7DE8mpG+/fVw+dTpjbxnx47TaMnDfOI1jwft9g1VybltZCduyRQPJPvc+zzKY9WPHxhPWczyFuYa6I8Mw4iU5A== + +json-stringify-safe@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb" + integrity sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA== + +jsonfile@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-4.0.0.tgz#8771aae0799b64076b76640fca058f9c10e33ecb" + integrity sha512-m6F1R3z8jjlf2imQHS2Qez5sjKWQzbuuhuJ/FKYFRZvPE3PuHcSMVZzfsLhGVOkfd20obL5SWEBew5ShlquNxg== + optionalDependencies: + graceful-fs "^4.1.6" + +jsonfile@^6.0.1: + version "6.1.0" + resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-6.1.0.tgz#bc55b2634793c679ec6403094eb13698a6ec0aae" + integrity sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ== + dependencies: + universalify "^2.0.0" + optionalDependencies: + graceful-fs "^4.1.6" + +keyv@^4.0.0: + version "4.5.3" + resolved "https://registry.yarnpkg.com/keyv/-/keyv-4.5.3.tgz#00873d2b046df737963157bd04f294ca818c9c25" + integrity sha512-QCiSav9WaX1PgETJ+SpNnx2PRRapJ/oRSXM4VO5OGYGSjrxbKPVFVhB3l2OCbLCk329N8qyAtsJjSjvVBWzEug== + dependencies: + json-buffer "3.0.1" + +lazy-val@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/lazy-val/-/lazy-val-1.0.5.tgz#6cf3b9f5bc31cee7ee3e369c0832b7583dcd923d" + integrity sha512-0/BnGCCfyUMkBpeDgWihanIAF9JmZhHBgUhEqzvf+adhNGLoP6TaiI5oF8oyb3I45P+PcnrqihSf01M0l0G5+Q== + +locate-path@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-3.0.0.tgz#dbec3b3ab759758071b58fe59fc41871af21400e" + integrity sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A== + dependencies: + p-locate "^3.0.0" + path-exists "^3.0.0" + +lodash.escaperegexp@^4.1.2: + version "4.1.2" + resolved "https://registry.yarnpkg.com/lodash.escaperegexp/-/lodash.escaperegexp-4.1.2.tgz#64762c48618082518ac3df4ccf5d5886dae20347" + integrity sha512-TM9YBvyC84ZxE3rgfefxUWiQKLilstD6k7PTGt6wfbtXF8ixIJLOL3VYyV/z+ZiPLsVxAsKAFVwWlWeb2Y8Yyw== + +lodash.isequal@^4.5.0: + version "4.5.0" + resolved "https://registry.yarnpkg.com/lodash.isequal/-/lodash.isequal-4.5.0.tgz#415c4478f2bcc30120c22ce10ed3226f7d3e18e0" + integrity sha512-pDo3lu8Jhfjqls6GkMgpahsF9kCyayhgykjyLMNFTKWrpVdAQtYyB4muAMWozBB4ig/dtWAmsMxLEI8wuz+DYQ== + +lodash.once@^4.1.1: + version "4.1.1" + resolved "https://registry.yarnpkg.com/lodash.once/-/lodash.once-4.1.1.tgz#0dd3971213c7c56df880977d504c88fb471a97ac" + integrity sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg== + +lowercase-keys@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/lowercase-keys/-/lowercase-keys-2.0.0.tgz#2603e78b7b4b0006cbca2fbcc8a3202558ac9479" + integrity sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA== + +lru-cache@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-6.0.0.tgz#6d6fe6570ebd96aaf90fcad1dafa3b2566db3a94" + integrity sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA== + dependencies: + yallist "^4.0.0" + +luxon@~3.3.0: + version "3.3.0" + resolved "https://registry.yarnpkg.com/luxon/-/luxon-3.3.0.tgz#d73ab5b5d2b49a461c47cedbc7e73309b4805b48" + integrity sha512-An0UCfG/rSiqtAIiBPO0Y9/zAnHUZxAMiCpTd5h2smgsj7GGmcenvrvww2cqNA8/4A5ZrD1gJpHN2mIHZQF+Mg== + +matcher@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/matcher/-/matcher-3.0.0.tgz#bd9060f4c5b70aa8041ccc6f80368760994f30ca" + integrity sha512-OkeDaAZ/bQCxeFAozM55PKcKU0yJMPGifLwV4Qgjitu+5MoAfSQN4lsLJeXZ1b8w0x+/Emda6MZgXS1jvsapng== + dependencies: + escape-string-regexp "^4.0.0" + +merge-stream@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/merge-stream/-/merge-stream-2.0.0.tgz#52823629a14dd00c9770fb6ad47dc6310f2c1f60" + integrity sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w== + +merge2@^1.3.0, merge2@^1.4.1: + version "1.4.1" + resolved "https://registry.yarnpkg.com/merge2/-/merge2-1.4.1.tgz#4368892f885e907455a6fd7dc55c0c9d404990ae" + integrity sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg== + +micromatch@^4.0.4: + version "4.0.5" + resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-4.0.5.tgz#bc8999a7cbbf77cdc89f132f6e467051b49090c6" + integrity sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA== + dependencies: + braces "^3.0.2" + picomatch "^2.3.1" + +mime-db@1.52.0: + version "1.52.0" + resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.52.0.tgz#bbabcdc02859f4987301c856e3387ce5ec43bf70" + integrity sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg== + +mime-types@^2.1.12: + version "2.1.35" + resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.35.tgz#381a871b62a734450660ae3deee44813f70d959a" + integrity sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw== + dependencies: + mime-db "1.52.0" + +mimic-fn@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-2.1.0.tgz#7ed2c2ccccaf84d3ffcb7a69b57711fc2083401b" + integrity sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg== + +mimic-fn@^3.0.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-3.1.0.tgz#65755145bbf3e36954b949c16450427451d5ca74" + integrity sha512-Ysbi9uYW9hFyfrThdDEQuykN4Ey6BuwPD2kpI5ES/nFTDn/98yxYNLZJcgUAKPT/mcrLLKaGzJR9YVxJrIdASQ== + +mimic-response@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/mimic-response/-/mimic-response-1.0.1.tgz#4923538878eef42063cb8a3e3b0798781487ab1b" + integrity sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ== + +mimic-response@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/mimic-response/-/mimic-response-3.1.0.tgz#2d1d59af9c1b129815accc2c46a022a5ce1fa3c9" + integrity sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ== + +minimatch@^3.1.1: + version "3.1.2" + resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.1.2.tgz#19cd194bfd3e428f049a70817c038d89ab4be35b" + integrity sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw== + dependencies: + brace-expansion "^1.1.7" + +ms@2.1.2: + version "2.1.2" + resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009" + integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w== + +normalize-url@^6.0.1: + version "6.1.0" + resolved "https://registry.yarnpkg.com/normalize-url/-/normalize-url-6.1.0.tgz#40d0885b535deffe3f3147bec877d05fe4c5668a" + integrity sha512-DlL+XwOy3NxAQ8xuC0okPgK46iuVNAK01YN7RueYBqqFeGsBjV9XmCAzAdgt+667bCl5kPh9EqKKDwnaPG1I7A== + +npm-run-path@^4.0.1: + version "4.0.1" + resolved "https://registry.yarnpkg.com/npm-run-path/-/npm-run-path-4.0.1.tgz#b7ecd1e5ed53da8e37a55e1c2269e0b97ed748ea" + integrity sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw== + dependencies: + path-key "^3.0.0" + +object-keys@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/object-keys/-/object-keys-1.1.1.tgz#1c47f272df277f3b1daf061677d9c82e2322c60e" + integrity sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA== + +once@^1.3.0, once@^1.3.1, once@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1" + integrity sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w== + dependencies: + wrappy "1" + +onetime@^5.1.2: + version "5.1.2" + resolved "https://registry.yarnpkg.com/onetime/-/onetime-5.1.2.tgz#d0e96ebb56b07476df1dd9c4806e5237985ca45e" + integrity sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg== + dependencies: + mimic-fn "^2.1.0" + +p-cancelable@^2.0.0: + version "2.1.1" + resolved "https://registry.yarnpkg.com/p-cancelable/-/p-cancelable-2.1.1.tgz#aab7fbd416582fa32a3db49859c122487c5ed2cf" + integrity sha512-BZOr3nRQHOntUjTrH8+Lh54smKHoHyur8We1V8DSMVrl5A2malOOwuJRnKRDjSnkoeBh4at6BwEnb5I7Jl31wg== + +p-limit@^2.0.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-2.3.0.tgz#3dd33c647a214fdfffd835933eb086da0dc21db1" + integrity sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w== + dependencies: + p-try "^2.0.0" + +p-locate@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-3.0.0.tgz#322d69a05c0264b25997d9f40cd8a891ab0064a4" + integrity sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ== + dependencies: + p-limit "^2.0.0" + +p-map@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/p-map/-/p-map-4.0.0.tgz#bb2f95a5eda2ec168ec9274e06a747c3e2904d2b" + integrity sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ== + dependencies: + aggregate-error "^3.0.0" + +p-try@^2.0.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/p-try/-/p-try-2.2.0.tgz#cb2868540e313d61de58fafbe35ce9004d5540e6" + integrity sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ== + +path-exists@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-3.0.0.tgz#ce0ebeaa5f78cb18925ea7d810d7b59b010fd515" + integrity sha512-bpC7GYwiDYQ4wYLe+FA8lhRjhQCMcQGuSgGGqDkg/QerRWw9CmGRT0iSOVRSZJ29NMLZgIzqaljJ63oaL4NIJQ== + +path-is-absolute@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f" + integrity sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg== + +path-key@^3.0.0, path-key@^3.1.0: + version "3.1.1" + resolved "https://registry.yarnpkg.com/path-key/-/path-key-3.1.1.tgz#581f6ade658cbba65a0d3380de7753295054f375" + integrity sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q== + +path-type@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/path-type/-/path-type-4.0.0.tgz#84ed01c0a7ba380afe09d90a8c180dcd9d03043b" + integrity sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw== + +pend@~1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/pend/-/pend-1.2.0.tgz#7a57eb550a6783f9115331fcf4663d5c8e007a50" + integrity sha512-F3asv42UuXchdzt+xXqfW1OGlVBe+mxa2mqI0pg5yAHZPvFmY3Y6drSf/GQ1A86WgWEN9Kzh/WrgKa6iGcHXLg== + +picomatch@^2.3.1: + version "2.3.1" + resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.3.1.tgz#3ba3833733646d9d3e4995946c1365a67fb07a42" + integrity sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA== + +pkg-up@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/pkg-up/-/pkg-up-3.1.0.tgz#100ec235cc150e4fd42519412596a28512a0def5" + integrity sha512-nDywThFk1i4BQK4twPQ6TA4RT8bDY96yeuCVBWL3ePARCiEKDRSrNGbFIgUJpLp+XeIR65v8ra7WuJOFUBtkMA== + dependencies: + find-up "^3.0.0" + +progress@^2.0.3: + version "2.0.3" + resolved "https://registry.yarnpkg.com/progress/-/progress-2.0.3.tgz#7e8cf8d8f5b8f239c1bc68beb4eb78567d572ef8" + integrity sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA== + +proxy-from-env@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/proxy-from-env/-/proxy-from-env-1.1.0.tgz#e102f16ca355424865755d2c9e8ea4f24d58c3e2" + integrity sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg== + +pump@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/pump/-/pump-3.0.0.tgz#b4a2116815bde2f4e1ea602354e8c75565107a64" + integrity sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww== + dependencies: + end-of-stream "^1.1.0" + once "^1.3.1" + +punycode@^2.1.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.3.0.tgz#f67fa67c94da8f4d0cfff981aee4118064199b8f" + integrity sha512-rRV+zQD8tVFys26lAGR9WUuS4iUAngJScM+ZRSKtvl5tKeZ2t5bvdNFdNHBW9FWR4guGHlgmsZ1G7BSm2wTbuA== + +queue-microtask@^1.2.2: + version "1.2.3" + resolved "https://registry.yarnpkg.com/queue-microtask/-/queue-microtask-1.2.3.tgz#4929228bbc724dfac43e0efb058caf7b6cfb6243" + integrity sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A== + +quick-lru@^5.1.1: + version "5.1.1" + resolved "https://registry.yarnpkg.com/quick-lru/-/quick-lru-5.1.1.tgz#366493e6b3e42a3a6885e2e99d18f80fb7a8c932" + integrity sha512-WuyALRjWPDGtt/wzJiadO5AXY+8hZ80hVpe6MyivgraREW751X3SbhRvG3eLKOYN+8VEvqLcf3wdnt44Z4S4SA== + +require-from-string@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/require-from-string/-/require-from-string-2.0.2.tgz#89a7fdd938261267318eafe14f9c32e598c36909" + integrity sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw== + +resolve-alpn@^1.0.0: + version "1.2.1" + resolved "https://registry.yarnpkg.com/resolve-alpn/-/resolve-alpn-1.2.1.tgz#b7adbdac3546aaaec20b45e7d8265927072726f9" + integrity sha512-0a1F4l73/ZFZOakJnQ3FvkJ2+gSTQWz/r2KE5OdDY0TxPm5h4GkqkWWfM47T7HsbnOtcJVEF4epCVy6u7Q3K+g== + +responselike@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/responselike/-/responselike-2.0.1.tgz#9a0bc8fdc252f3fb1cca68b016591059ba1422bc" + integrity sha512-4gl03wn3hj1HP3yzgdI7d3lCkF95F21Pz4BPGvKHinyQzALR5CapwC8yIi0Rh58DEMQ/SguC03wFj2k0M/mHhw== + dependencies: + lowercase-keys "^2.0.0" + +reusify@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/reusify/-/reusify-1.0.4.tgz#90da382b1e126efc02146e90845a88db12925d76" + integrity sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw== + +rimraf@^3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-3.0.2.tgz#f1a5402ba6220ad52cc1282bac1ae3aa49fd061a" + integrity sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA== + dependencies: + glob "^7.1.3" + +roarr@^2.15.3: + version "2.15.4" + resolved "https://registry.yarnpkg.com/roarr/-/roarr-2.15.4.tgz#f5fe795b7b838ccfe35dc608e0282b9eba2e7afd" + integrity sha512-CHhPh+UNHD2GTXNYhPWLnU8ONHdI+5DI+4EYIAOaiD63rHeYlZvyh8P+in5999TTSFgUYuKUAjzRI4mdh/p+2A== + dependencies: + boolean "^3.0.1" + detect-node "^2.0.4" + globalthis "^1.0.1" + json-stringify-safe "^5.0.1" + semver-compare "^1.0.0" + sprintf-js "^1.1.2" + +run-parallel@^1.1.9: + version "1.2.0" + resolved "https://registry.yarnpkg.com/run-parallel/-/run-parallel-1.2.0.tgz#66d1368da7bdf921eb9d95bd1a9229e7f21a43ee" + integrity sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA== + dependencies: + queue-microtask "^1.2.2" + +sax@^1.2.4: + version "1.2.4" + resolved "https://registry.yarnpkg.com/sax/-/sax-1.2.4.tgz#2816234e2378bddc4e5354fab5caa895df7100d9" + integrity sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw== + +semver-compare@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/semver-compare/-/semver-compare-1.0.0.tgz#0dee216a1c941ab37e9efb1788f6afc5ff5537fc" + integrity sha512-YM3/ITh2MJ5MtzaM429anh+x2jiLVjqILF4m4oyQB18W7Ggea7BfqdH/wGMK7dDiMghv/6WG7znWMwUDzJiXow== + +semver@^6.2.0: + version "6.3.1" + resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4" + integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA== + +semver@^7.3.2, semver@^7.3.5: + version "7.5.4" + resolved "https://registry.yarnpkg.com/semver/-/semver-7.5.4.tgz#483986ec4ed38e1c6c48c34894a9182dbff68a6e" + integrity sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA== + dependencies: + lru-cache "^6.0.0" + +serialize-error@^7.0.1: + version "7.0.1" + resolved "https://registry.yarnpkg.com/serialize-error/-/serialize-error-7.0.1.tgz#f1360b0447f61ffb483ec4157c737fab7d778e18" + integrity sha512-8I8TjW5KMOKsZQTvoxjuSIa7foAwPWGOts+6o7sgjz41/qMD9VQHEDxi6PBvK2l0MXUmqZyNpUK+T2tQaaElvw== + dependencies: + type-fest "^0.13.1" + +shebang-command@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-2.0.0.tgz#ccd0af4f8835fbdc265b82461aaf0c36663f34ea" + integrity sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA== + dependencies: + shebang-regex "^3.0.0" + +shebang-regex@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/shebang-regex/-/shebang-regex-3.0.0.tgz#ae16f1644d873ecad843b0307b143362d4c42172" + integrity sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A== + +signal-exit@^3.0.3: + version "3.0.7" + resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.7.tgz#a9a1767f8af84155114eaabd73f99273c8f59ad9" + integrity sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ== + +slash@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/slash/-/slash-3.0.0.tgz#6539be870c165adbd5240220dbe361f1bc4d4634" + integrity sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q== + +sprintf-js@^1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.1.2.tgz#da1765262bf8c0f571749f2ad6c26300207ae673" + integrity sha512-VE0SOVEHCk7Qc8ulkWw3ntAzXuqf7S2lvwQaDLRnUeIEaKNQJzV6BwmLKhOqT61aGhfUMrXeaBk+oDGCzvhcug== + +strip-final-newline@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/strip-final-newline/-/strip-final-newline-2.0.0.tgz#89b852fb2fcbe936f6f4b3187afb0a12c1ab58ad" + integrity sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA== + +sumchecker@^3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/sumchecker/-/sumchecker-3.0.1.tgz#6377e996795abb0b6d348e9b3e1dfb24345a8e42" + integrity sha512-MvjXzkz/BOfyVDkG0oFOtBxHX2u3gKbMHIF/dXblZsgD3BWOFLmHovIpZY7BykJdAjcqRCBi1WYBNdEC9yI7vg== + dependencies: + debug "^4.1.0" + +to-regex-range@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-5.0.1.tgz#1648c44aae7c8d988a326018ed72f5b4dd0392e4" + integrity sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ== + dependencies: + is-number "^7.0.0" + +type-fest@^0.13.1: + version "0.13.1" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.13.1.tgz#0172cb5bce80b0bd542ea348db50c7e21834d934" + integrity sha512-34R7HTnG0XIJcBSn5XhDd7nNFPRcXYRZrBB2O2jdKqYODldSzBAqzsWoZYYvduky73toYS/ESqxPvkDf/F0XMg== + +type-fest@^2.17.0: + version "2.19.0" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-2.19.0.tgz#88068015bb33036a598b952e55e9311a60fd3a9b" + integrity sha512-RAH822pAdBgcNMAfWnCBU3CFZcfZ/i1eZjwFU/dsLKumyuuP3niueg2UAukXYF0E2AAoc82ZSSf9J0WQBinzHA== + +universalify@^0.1.0: + version "0.1.2" + resolved "https://registry.yarnpkg.com/universalify/-/universalify-0.1.2.tgz#b646f69be3942dabcecc9d6639c80dc105efaa66" + integrity sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg== + +universalify@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/universalify/-/universalify-2.0.0.tgz#75a4984efedc4b08975c5aeb73f530d02df25717" + integrity sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ== + +uri-js@^4.2.2: + version "4.4.1" + resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e" + integrity sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg== + dependencies: + punycode "^2.1.0" + +which@^2.0.1: + version "2.0.2" + resolved "https://registry.yarnpkg.com/which/-/which-2.0.2.tgz#7c6a8dd0a636a0327e10b59c9286eee93f3f51b1" + integrity sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA== + dependencies: + isexe "^2.0.0" + +wrappy@1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" + integrity sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ== + +yallist@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/yallist/-/yallist-4.0.0.tgz#9bb92790d9c0effec63be73519e11a35019a3a72" + integrity sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A== + +yauzl@^2.10.0: + version "2.10.0" + resolved "https://registry.yarnpkg.com/yauzl/-/yauzl-2.10.0.tgz#c7eb17c93e112cb1086fa6d8e51fb0667b79a5f9" + integrity sha512-p4a9I6X6nu6IhoGmBqAcbJy1mlC4j27vEPZX9F4L4/vZT3Lyq1VkFHw/V/PUcB9Buo+DG3iHkT0x3Qya58zc3g== + dependencies: + buffer-crc32 "~0.2.3" + fd-slicer "~1.1.0" diff --git a/src/khoj/configure.py b/src/khoj/configure.py index d35b04f8..6a358c1f 100644 --- a/src/khoj/configure.py +++ b/src/khoj/configure.py @@ -105,7 +105,7 @@ def configure_routes(app): app.mount("/static", StaticFiles(directory=constants.web_directory), name="static") app.include_router(api, prefix="/api") app.include_router(api_beta, prefix="/api/beta") - app.include_router(indexer, prefix="/indexer") + app.include_router(indexer, prefix="/v1/indexer") app.include_router(web_client) diff --git a/src/khoj/processor/github/github_to_jsonl.py b/src/khoj/processor/github/github_to_jsonl.py index 6abc137f..bcd2e530 100644 --- a/src/khoj/processor/github/github_to_jsonl.py +++ b/src/khoj/processor/github/github_to_jsonl.py @@ -37,7 +37,7 @@ class GithubToJsonl(TextToJsonl): else: return - def process(self, previous_entries=[], files=None): + def process(self, previous_entries=[], files=None, full_corpus=True): if self.config.pat_token is None or self.config.pat_token == "": logger.error(f"Github PAT token is not set. Skipping github content") raise ValueError("Github PAT token is not set. Skipping github content") diff --git a/src/khoj/processor/jsonl/jsonl_to_jsonl.py b/src/khoj/processor/jsonl/jsonl_to_jsonl.py index 1ace9b8a..4a6fab99 100644 --- a/src/khoj/processor/jsonl/jsonl_to_jsonl.py +++ b/src/khoj/processor/jsonl/jsonl_to_jsonl.py @@ -16,7 +16,7 @@ logger = logging.getLogger(__name__) class JsonlToJsonl(TextToJsonl): # Define Functions - def process(self, previous_entries=[], files: dict[str, str] = {}): + def process(self, previous_entries=[], files: dict[str, str] = {}, full_corpus: bool = True): # Extract required fields from config input_jsonl_files, input_jsonl_filter, output_file = ( self.config.input_files, diff --git a/src/khoj/processor/markdown/markdown_to_jsonl.py b/src/khoj/processor/markdown/markdown_to_jsonl.py index 4cb5a525..c2f0f0bf 100644 --- a/src/khoj/processor/markdown/markdown_to_jsonl.py +++ b/src/khoj/processor/markdown/markdown_to_jsonl.py @@ -22,10 +22,17 @@ class MarkdownToJsonl(TextToJsonl): self.config = config # Define Functions - def process(self, previous_entries=[], files=None): + def process(self, previous_entries=[], files=None, full_corpus: bool = True): # Extract required fields from config output_file = self.config.compressed_jsonl + if not full_corpus: + deletion_file_names = set([file for file in files if files[file] == ""]) + files_to_process = set(files) - deletion_file_names + files = {file: files[file] for file in files_to_process} + else: + deletion_file_names = None + # Extract Entries from specified Markdown files with timer("Parse entries from Markdown files into dictionaries", logger): current_entries = MarkdownToJsonl.convert_markdown_entries_to_maps( @@ -39,7 +46,7 @@ class MarkdownToJsonl(TextToJsonl): # Identify, mark and merge any new entries with previous entries with timer("Identify new or updated entries", logger): entries_with_ids = TextToJsonl.mark_entries_for_update( - current_entries, previous_entries, key="compiled", logger=logger + current_entries, previous_entries, key="compiled", logger=logger, deletion_filenames=deletion_file_names ) with timer("Write markdown entries to JSONL file", logger): diff --git a/src/khoj/processor/notion/notion_to_jsonl.py b/src/khoj/processor/notion/notion_to_jsonl.py index ab53c974..0df56c37 100644 --- a/src/khoj/processor/notion/notion_to_jsonl.py +++ b/src/khoj/processor/notion/notion_to_jsonl.py @@ -80,7 +80,7 @@ class NotionToJsonl(TextToJsonl): self.body_params = {"page_size": 100} - def process(self, previous_entries=[], files=None): + def process(self, previous_entries=[], files=None, full_corpus=True): current_entries = [] # Get all pages diff --git a/src/khoj/processor/org_mode/org_to_jsonl.py b/src/khoj/processor/org_mode/org_to_jsonl.py index 34db4798..2f22add4 100644 --- a/src/khoj/processor/org_mode/org_to_jsonl.py +++ b/src/khoj/processor/org_mode/org_to_jsonl.py @@ -21,11 +21,20 @@ class OrgToJsonl(TextToJsonl): self.config = config # Define Functions - def process(self, previous_entries: List[Entry] = [], files: dict[str, str] = None) -> List[Tuple[int, Entry]]: + def process( + self, previous_entries: List[Entry] = [], files: dict[str, str] = None, full_corpus: bool = True + ) -> List[Tuple[int, Entry]]: # Extract required fields from config output_file = self.config.compressed_jsonl index_heading_entries = self.config.index_heading_entries + if not full_corpus: + deletion_file_names = set([file for file in files if files[file] == ""]) + files_to_process = set(files) - deletion_file_names + files = {file: files[file] for file in files_to_process} + else: + deletion_file_names = None + # Extract Entries from specified Org files with timer("Parse entries from org files into OrgNode objects", logger): entry_nodes, file_to_entries = self.extract_org_entries(files) @@ -39,7 +48,7 @@ class OrgToJsonl(TextToJsonl): # Identify, mark and merge any new entries with previous entries with timer("Identify new or updated entries", logger): entries_with_ids = TextToJsonl.mark_entries_for_update( - current_entries, previous_entries, key="compiled", logger=logger + current_entries, previous_entries, key="compiled", logger=logger, deletion_filenames=deletion_file_names ) # Process Each Entry from All Notes Files diff --git a/src/khoj/processor/pdf/pdf_to_jsonl.py b/src/khoj/processor/pdf/pdf_to_jsonl.py index ad52637e..77c34617 100644 --- a/src/khoj/processor/pdf/pdf_to_jsonl.py +++ b/src/khoj/processor/pdf/pdf_to_jsonl.py @@ -2,9 +2,10 @@ import os import logging from typing import List +import base64 # External Packages -from langchain.document_loaders import PyPDFLoader +from langchain.document_loaders import PyMuPDFLoader # Internal Packages from khoj.processor.text_to_jsonl import TextToJsonl @@ -18,10 +19,17 @@ logger = logging.getLogger(__name__) class PdfToJsonl(TextToJsonl): # Define Functions - def process(self, previous_entries=[], files=dict[str, str]): + def process(self, previous_entries=[], files: dict[str, str] = None, full_corpus: bool = True): # Extract required fields from config output_file = self.config.compressed_jsonl + if not full_corpus: + deletion_file_names = set([file for file in files if files[file] == ""]) + files_to_process = set(files) - deletion_file_names + files = {file: files[file] for file in files_to_process} + else: + deletion_file_names = None + # Extract Entries from specified Pdf files with timer("Parse entries from PDF files into dictionaries", logger): current_entries = PdfToJsonl.convert_pdf_entries_to_maps(*PdfToJsonl.extract_pdf_entries(files)) @@ -33,7 +41,7 @@ class PdfToJsonl(TextToJsonl): # Identify, mark and merge any new entries with previous entries with timer("Identify new or updated entries", logger): entries_with_ids = TextToJsonl.mark_entries_for_update( - current_entries, previous_entries, key="compiled", logger=logger + current_entries, previous_entries, key="compiled", logger=logger, deletion_filenames=deletion_file_names ) with timer("Write PDF entries to JSONL file", logger): @@ -55,9 +63,11 @@ class PdfToJsonl(TextToJsonl): for pdf_file in pdf_files: try: # Write the PDF file to a temporary file, as it is stored in byte format in the pdf_file object and the PyPDFLoader expects a file path - with open(f"{pdf_file}.pdf", "wb") as f: - f.write(pdf_files[pdf_file]) - loader = PyPDFLoader(f"{pdf_file}.pdf") + tmp_file = f"tmp_pdf_file.pdf" + with open(f"{tmp_file}", "wb") as f: + bytes = base64.b64decode(pdf_files[pdf_file]) + f.write(bytes) + loader = PyMuPDFLoader(f"{tmp_file}") pdf_entries_per_file = [page.page_content for page in loader.load()] entry_to_location_map += zip(pdf_entries_per_file, [pdf_file] * len(pdf_entries_per_file)) entries.extend(pdf_entries_per_file) @@ -65,8 +75,8 @@ class PdfToJsonl(TextToJsonl): logger.warning(f"Unable to process file: {pdf_file}. This file will not be indexed.") logger.warning(e) finally: - if os.path.exists(f"{pdf_file}.pdf"): - os.remove(f"{pdf_file}.pdf") + if os.path.exists(f"{tmp_file}"): + os.remove(f"{tmp_file}") return entries, dict(entry_to_location_map) diff --git a/src/khoj/processor/plaintext/plaintext_to_jsonl.py b/src/khoj/processor/plaintext/plaintext_to_jsonl.py index f488faed..3acb656e 100644 --- a/src/khoj/processor/plaintext/plaintext_to_jsonl.py +++ b/src/khoj/processor/plaintext/plaintext_to_jsonl.py @@ -15,9 +15,18 @@ logger = logging.getLogger(__name__) class PlaintextToJsonl(TextToJsonl): # Define Functions - def process(self, previous_entries: List[Entry] = [], files: dict[str, str] = None) -> List[Tuple[int, Entry]]: + def process( + self, previous_entries: List[Entry] = [], files: dict[str, str] = None, full_corpus: bool = True + ) -> List[Tuple[int, Entry]]: output_file = self.config.compressed_jsonl + if not full_corpus: + deletion_file_names = set([file for file in files if files[file] == ""]) + files_to_process = set(files) - deletion_file_names + files = {file: files[file] for file in files_to_process} + else: + deletion_file_names = None + # Extract Entries from specified plaintext files with timer("Parse entries from plaintext files", logger): current_entries = PlaintextToJsonl.convert_plaintext_entries_to_maps(files) @@ -29,7 +38,7 @@ class PlaintextToJsonl(TextToJsonl): # Identify, mark and merge any new entries with previous entries with timer("Identify new or updated entries", logger): entries_with_ids = TextToJsonl.mark_entries_for_update( - current_entries, previous_entries, key="compiled", logger=logger + current_entries, previous_entries, key="compiled", logger=logger, deletion_filenames=deletion_file_names ) with timer("Write entries to JSONL file", logger): diff --git a/src/khoj/processor/text_to_jsonl.py b/src/khoj/processor/text_to_jsonl.py index 09371994..98f5986f 100644 --- a/src/khoj/processor/text_to_jsonl.py +++ b/src/khoj/processor/text_to_jsonl.py @@ -2,7 +2,7 @@ from abc import ABC, abstractmethod import hashlib import logging -from typing import Callable, List, Tuple +from typing import Callable, List, Tuple, Set from khoj.utils.helpers import timer # Internal Packages @@ -17,7 +17,9 @@ class TextToJsonl(ABC): self.config = config @abstractmethod - def process(self, previous_entries: List[Entry] = [], files: dict[str, str] = None) -> List[Tuple[int, Entry]]: + def process( + self, previous_entries: List[Entry] = [], files: dict[str, str] = None, full_corpus: bool = True + ) -> List[Tuple[int, Entry]]: ... @staticmethod @@ -62,12 +64,21 @@ class TextToJsonl(ABC): @staticmethod def mark_entries_for_update( - current_entries: List[Entry], previous_entries: List[Entry], key="compiled", logger: logging.Logger = None - ) -> List[Tuple[int, Entry]]: + current_entries: List[Entry], + previous_entries: List[Entry], + key="compiled", + logger: logging.Logger = None, + deletion_filenames: Set[str] = None, + ): # Hash all current and previous entries to identify new entries with timer("Hash previous, current entries", logger): current_entry_hashes = list(map(TextToJsonl.hash_func(key), current_entries)) previous_entry_hashes = list(map(TextToJsonl.hash_func(key), previous_entries)) + if deletion_filenames is not None: + deletion_entries = [entry for entry in previous_entries if entry.file in deletion_filenames] + deletion_entry_hashes = list(map(TextToJsonl.hash_func(key), deletion_entries)) + else: + deletion_entry_hashes = [] with timer("Identify, Mark, Combine new, existing entries", logger): hash_to_current_entries = dict(zip(current_entry_hashes, current_entries)) @@ -77,6 +88,19 @@ class TextToJsonl(ABC): new_entry_hashes = set(current_entry_hashes) - set(previous_entry_hashes) # All entries that exist in both current and previous sets are kept existing_entry_hashes = set(current_entry_hashes) & set(previous_entry_hashes) + # All entries that exist in the previous set but not in the current set should be preserved + remaining_entry_hashes = set(previous_entry_hashes) - set(current_entry_hashes) + # All entries that exist in the previous set and also in the deletions set should be removed + to_delete_entry_hashes = set(previous_entry_hashes) & set(deletion_entry_hashes) + + preserving_entry_hashes = existing_entry_hashes + + if deletion_filenames is not None: + preserving_entry_hashes = ( + (existing_entry_hashes | remaining_entry_hashes) + if len(deletion_entry_hashes) == 0 + else (set(previous_entry_hashes) - to_delete_entry_hashes) + ) # load new entries in the order in which they are processed for a stable sort new_entries = [ @@ -90,7 +114,7 @@ class TextToJsonl(ABC): # Set id of existing entries to their previous ids to reuse their existing encoded embeddings existing_entries = [ (previous_entry_hashes.index(entry_hash), hash_to_previous_entries[entry_hash]) - for entry_hash in existing_entry_hashes + for entry_hash in preserving_entry_hashes ] existing_entries_sorted = sorted(existing_entries, key=lambda e: e[0]) diff --git a/src/khoj/routers/indexer.py b/src/khoj/routers/indexer.py index eaff7bf8..ca2d884e 100644 --- a/src/khoj/routers/indexer.py +++ b/src/khoj/routers/indexer.py @@ -1,6 +1,7 @@ # Standard Packages import logging -from typing import Optional, Union +import sys +from typing import Optional, Union, Dict # External Packages from fastapi import APIRouter, HTTPException, Header, Request, Body, Response @@ -15,10 +16,11 @@ from khoj.processor.pdf.pdf_to_jsonl import PdfToJsonl from khoj.processor.github.github_to_jsonl import GithubToJsonl from khoj.processor.notion.notion_to_jsonl import NotionToJsonl from khoj.processor.plaintext.plaintext_to_jsonl import PlaintextToJsonl -from khoj.utils.rawconfig import ContentConfig +from khoj.utils.rawconfig import ContentConfig, TextContentConfig from khoj.search_type import text_search, image_search from khoj.utils.config import SearchModels -from khoj.utils.helpers import LRU +from khoj.utils.constants import default_config +from khoj.utils.helpers import LRU, get_file_type from khoj.utils.rawconfig import ( ContentConfig, ) @@ -35,11 +37,20 @@ logger = logging.getLogger(__name__) indexer = APIRouter() +class File(BaseModel): + path: str + content: Union[str, bytes] + + class IndexBatchRequest(BaseModel): - org: Optional[dict[str, str]] - pdf: Optional[dict[str, str]] - plaintext: Optional[dict[str, str]] - markdown: Optional[dict[str, str]] + files: list[File] + + +class IndexerInput(BaseModel): + org: Optional[dict[str, str]] = None + markdown: Optional[dict[str, str]] = None + pdf: Optional[dict[str, bytes]] = None + plaintext: Optional[dict[str, str]] = None @indexer.post("/batch") @@ -54,24 +65,65 @@ async def index_batch( state.config_lock.acquire() try: logger.info(f"Received batch indexing request") - index_batch_request_acc = "" + index_batch_request_acc = b"" async for chunk in request.stream(): - index_batch_request_acc += chunk.decode() + index_batch_request_acc += chunk + data_bytes = sys.getsizeof(index_batch_request_acc) + unit = "KB" + data_size = data_bytes / 1024 + if data_size > 1000: + unit = "MB" + data_size = data_size / 1024 + if data_size > 1000: + unit = "GB" + data_size = data_size / 1024 + data_size_metric = f"{data_size:.2f} {unit}" + logger.info(f"Received {data_size_metric} of data") index_batch_request = IndexBatchRequest.parse_raw(index_batch_request_acc) - logger.info(f"Received batch indexing request size: {len(index_batch_request.dict())}") + logger.info(f"Received {len(index_batch_request.files)} files") + + org_files: Dict[str, str] = {} + markdown_files: Dict[str, str] = {} + pdf_files: Dict[str, str] = {} + plaintext_files: Dict[str, str] = {} + + for file in index_batch_request.files: + file_type = get_file_type(file.path) + dict_to_update = None + if file_type == "org": + dict_to_update = org_files + elif file_type == "markdown": + dict_to_update = markdown_files + elif file_type == "pdf": + dict_to_update = pdf_files + elif file_type == "plaintext": + dict_to_update = plaintext_files + + if dict_to_update is not None: + dict_to_update[file.path] = file.content + else: + logger.info(f"Skipping unsupported streamed file: {file.path}") + + indexer_input = IndexerInput( + org=org_files, + markdown=markdown_files, + pdf=pdf_files, + plaintext=plaintext_files, + ) # Extract required fields from config state.content_index = configure_content( state.content_index, state.config.content_type, - index_batch_request.dict(), + indexer_input.dict(), state.search_models, regenerate=regenerate, t=search_type, + full_corpus=False, ) except Exception as e: - logger.error(f"Failed to process batch indexing request: {e}") + logger.error(f"Failed to process batch indexing request: {e}", exc_info=True) finally: state.config_lock.release() return Response(content="OK", status_code=200) @@ -84,6 +136,7 @@ def configure_content( search_models: SearchModels, regenerate: bool = False, t: Optional[Union[state.SearchType, str]] = None, + full_corpus: bool = True, ) -> Optional[ContentIndex]: # Run Validation Checks if content_config is None: @@ -105,10 +158,17 @@ def configure_content( # Initialize Org Notes Search if ( (t == None or t == state.SearchType.Org.value) - and content_config.org + and (content_config.org or files["org"]) and search_models.text_search - and files["org"] ): + if content_config.org == None: + logger.info("🦄 No configuration for orgmode notes. Using default configuration.") + default_configuration = default_config["content-type"]["org"] # type: ignore + content_config.org = TextContentConfig( + compressed_jsonl=default_configuration["compressed-jsonl"], + embeddings_file=default_configuration["embeddings-file"], + ) + logger.info("🦄 Setting up search for orgmode notes") # Extract Entries, Generate Notes Embeddings content_index.org = text_search.setup( @@ -118,15 +178,27 @@ def configure_content( search_models.text_search.bi_encoder, regenerate=regenerate, filters=[DateFilter(), WordFilter(), FileFilter()], + full_corpus=full_corpus, ) + except Exception as e: + logger.error(f"🚨 Failed to setup org: {e}", exc_info=True) + try: # Initialize Markdown Search if ( (t == None or t == state.SearchType.Markdown.value) - and content_config.markdown + and (content_config.markdown or files["markdown"]) and search_models.text_search and files["markdown"] ): + if content_config.markdown == None: + logger.info("💎 No configuration for markdown notes. Using default configuration.") + default_configuration = default_config["content-type"]["markdown"] # type: ignore + content_config.markdown = TextContentConfig( + compressed_jsonl=default_configuration["compressed-jsonl"], + embeddings_file=default_configuration["embeddings-file"], + ) + logger.info("💎 Setting up search for markdown notes") # Extract Entries, Generate Markdown Embeddings content_index.markdown = text_search.setup( @@ -136,15 +208,28 @@ def configure_content( search_models.text_search.bi_encoder, regenerate=regenerate, filters=[DateFilter(), WordFilter(), FileFilter()], + full_corpus=full_corpus, ) + except Exception as e: + logger.error(f"🚨 Failed to setup markdown: {e}", exc_info=True) + + try: # Initialize PDF Search if ( (t == None or t == state.SearchType.Pdf.value) - and content_config.pdf + and (content_config.pdf or files["pdf"]) and search_models.text_search and files["pdf"] ): + if content_config.pdf == None: + logger.info("🖨️ No configuration for pdf notes. Using default configuration.") + default_configuration = default_config["content-type"]["pdf"] # type: ignore + content_config.pdf = TextContentConfig( + compressed_jsonl=default_configuration["compressed-jsonl"], + embeddings_file=default_configuration["embeddings-file"], + ) + logger.info("🖨️ Setting up search for pdf") # Extract Entries, Generate PDF Embeddings content_index.pdf = text_search.setup( @@ -154,15 +239,28 @@ def configure_content( search_models.text_search.bi_encoder, regenerate=regenerate, filters=[DateFilter(), WordFilter(), FileFilter()], + full_corpus=full_corpus, ) + except Exception as e: + logger.error(f"🚨 Failed to setup PDF: {e}", exc_info=True) + + try: # Initialize Plaintext Search if ( (t == None or t == state.SearchType.Plaintext.value) - and content_config.plaintext + and (content_config.plaintext or files["plaintext"]) and search_models.text_search and files["plaintext"] ): + if content_config.plaintext == None: + logger.info("📄 No configuration for plaintext notes. Using default configuration.") + default_configuration = default_config["content-type"]["plaintext"] # type: ignore + content_config.plaintext = TextContentConfig( + compressed_jsonl=default_configuration["compressed-jsonl"], + embeddings_file=default_configuration["embeddings-file"], + ) + logger.info("📄 Setting up search for plaintext") # Extract Entries, Generate Plaintext Embeddings content_index.plaintext = text_search.setup( @@ -172,8 +270,13 @@ def configure_content( search_models.text_search.bi_encoder, regenerate=regenerate, filters=[DateFilter(), WordFilter(), FileFilter()], + full_corpus=full_corpus, ) + except Exception as e: + logger.error(f"🚨 Failed to setup plaintext: {e}", exc_info=True) + + try: # Initialize Image Search if (t == None or t == state.SearchType.Image.value) and content_config.image and search_models.image_search: logger.info("🌄 Setting up search for images") @@ -182,6 +285,10 @@ def configure_content( content_config.image, search_models.image_search.image_encoder, regenerate=regenerate ) + except Exception as e: + logger.error(f"🚨 Failed to setup images: {e}", exc_info=True) + + try: if (t == None or t == state.SearchType.Github.value) and content_config.github and search_models.text_search: logger.info("🐙 Setting up search for github") # Extract Entries, Generate Github Embeddings @@ -192,8 +299,13 @@ def configure_content( search_models.text_search.bi_encoder, regenerate=regenerate, filters=[DateFilter(), WordFilter(), FileFilter()], + full_corpus=full_corpus, ) + except Exception as e: + logger.error(f"🚨 Failed to setup GitHub: {e}", exc_info=True) + + try: # Initialize Notion Search if (t == None or t in state.SearchType.Notion.value) and content_config.notion and search_models.text_search: logger.info("🔌 Setting up search for notion") @@ -204,8 +316,13 @@ def configure_content( search_models.text_search.bi_encoder, regenerate=regenerate, filters=[DateFilter(), WordFilter(), FileFilter()], + full_corpus=full_corpus, ) + except Exception as e: + logger.error(f"🚨 Failed to setup GitHub: {e}", exc_info=True) + + try: # Initialize External Plugin Search if (t == None or t in state.SearchType) and content_config.plugins and search_models.text_search: logger.info("🔌 Setting up search for plugins") @@ -218,11 +335,11 @@ def configure_content( search_models.text_search.bi_encoder, regenerate=regenerate, filters=[DateFilter(), WordFilter(), FileFilter()], + full_corpus=full_corpus, ) except Exception as e: - logger.error(f"🚨 Failed to setup search: {e}", exc_info=True) - raise e + logger.error(f"🚨 Failed to setup Plugin: {e}", exc_info=True) # Invalidate Query Cache state.query_cache = LRU() diff --git a/src/khoj/search_type/text_search.py b/src/khoj/search_type/text_search.py index 468a7584..b790b000 100644 --- a/src/khoj/search_type/text_search.py +++ b/src/khoj/search_type/text_search.py @@ -192,19 +192,24 @@ def setup( regenerate: bool, filters: List[BaseFilter] = [], normalize: bool = True, + full_corpus: bool = True, ) -> TextContent: # Map notes in text files to (compressed) JSONL formatted file config.compressed_jsonl = resolve_absolute_path(config.compressed_jsonl) previous_entries = [] if config.compressed_jsonl.exists() and not regenerate: previous_entries = extract_entries(config.compressed_jsonl) - entries_with_indices = text_to_jsonl(config).process(previous_entries=previous_entries, files=files) + entries_with_indices = text_to_jsonl(config).process( + previous_entries=previous_entries, files=files, full_corpus=full_corpus + ) # Extract Updated Entries entries = extract_entries(config.compressed_jsonl) if is_none_or_empty(entries): config_params = ", ".join([f"{key}={value}" for key, value in config.dict().items()]) - raise ValueError(f"No valid entries found in specified files: {config_params}") + raise ValueError( + f"No valid entries found in specified configuration: {config_params}, with files: {files.keys()}" + ) # Compute or Load Embeddings config.embeddings_file = resolve_absolute_path(config.embeddings_file) diff --git a/src/khoj/utils/fs_syncer.py b/src/khoj/utils/fs_syncer.py index 7aa5af2b..d303d39b 100644 --- a/src/khoj/utils/fs_syncer.py +++ b/src/khoj/utils/fs_syncer.py @@ -1,5 +1,6 @@ import logging import glob +import base64 from typing import Optional from bs4 import BeautifulSoup @@ -209,7 +210,7 @@ def get_pdf_files(config: TextContentConfig): for file in all_pdf_files: with open(file, "rb") as f: try: - filename_to_content_map[file] = f.read() + filename_to_content_map[file] = base64.b64encode(f.read()).decode("utf-8") except Exception as e: logger.warning(f"Unable to read file: {file} as PDF. Skipping file.") logger.warning(e, exc_info=True) diff --git a/src/khoj/utils/helpers.py b/src/khoj/utils/helpers.py index 9bd139d4..fcdca7cb 100644 --- a/src/khoj/utils/helpers.py +++ b/src/khoj/utils/helpers.py @@ -66,6 +66,22 @@ def merge_dicts(priority_dict: dict, default_dict: dict): return merged_dict +def get_file_type(filepath: str) -> str: + "Get file type from file path" + file_type = Path(filepath).suffix[1:] + + if file_type in ["md", "markdown"]: + return "markdown" + elif file_type in ["org", "orgmode"]: + return "org" + elif file_type in ["txt", "text", "html", "xml", "htm", "rst"]: + return "plaintext" + elif file_type in ["pdf"]: + return "pdf" + + return file_type + + def load_model( model_name: str, model_type, model_dir=None, device: str = None ) -> Union[BaseEncoder, SentenceTransformer, CrossEncoder]: diff --git a/src/khoj/utils/rawconfig.py b/src/khoj/utils/rawconfig.py index 13c9b1cf..0a916db4 100644 --- a/src/khoj/utils/rawconfig.py +++ b/src/khoj/utils/rawconfig.py @@ -32,14 +32,6 @@ class TextContentConfig(TextConfigBase): input_filter: Optional[List[str]] index_heading_entries: Optional[bool] = False - @validator("input_filter") - def input_filter_or_files_required(cls, input_filter, values, **kwargs): - if is_none_or_empty(input_filter) and ("input_files" not in values or values["input_files"] is None): - raise ValueError( - "Either input_filter or input_files required in all content-type. section of Khoj config file" - ) - return input_filter - class GithubRepoConfig(ConfigBase): name: str @@ -63,16 +55,6 @@ class ImageContentConfig(ConfigBase): use_xmp_metadata: bool batch_size: int - @validator("input_filter") - def input_filter_or_directories_required(cls, input_filter, values, **kwargs): - if is_none_or_empty(input_filter) and ( - "input_directories" not in values or values["input_directories"] is None - ): - raise ValueError( - "Either input_filter or input_directories required in all content-type.image section of Khoj config file" - ) - return input_filter - class ContentConfig(ConfigBase): org: Optional[TextContentConfig] diff --git a/tests/test_client.py b/tests/test_client.py index b2fad8b1..d2497f73 100644 --- a/tests/test_client.py +++ b/tests/test_client.py @@ -66,7 +66,7 @@ def test_index_batch(client): headers = {"x-api-key": "secret"} # Act - response = client.post("/indexer/batch", json=request_body, headers=headers) + response = client.post("/v1/indexer/batch", json=request_body, headers=headers) # Assert assert response.status_code == 200 @@ -81,7 +81,7 @@ def test_regenerate_with_valid_content_type(client): headers = {"x-api-key": "secret"} # Act - response = client.post(f"/indexer/batch?search_type={content_type}", json=request_body, headers=headers) + response = client.post(f"/v1/indexer/batch?search_type={content_type}", json=request_body, headers=headers) # Assert assert response.status_code == 200, f"Returned status: {response.status_code} for content type: {content_type}" @@ -97,7 +97,7 @@ def test_regenerate_with_github_fails_without_pat(client): headers = {"x-api-key": "secret"} # Act - response = client.post(f"/indexer/batch?search_type=github", json=request_body, headers=headers) + response = client.post(f"/v1/indexer/batch?search_type=github", json=request_body, headers=headers) # Assert assert response.status_code == 200, f"Returned status: {response.status_code} for content type: github" diff --git a/tests/test_pdf_to_jsonl.py b/tests/test_pdf_to_jsonl.py index b9b26986..bacce37c 100644 --- a/tests/test_pdf_to_jsonl.py +++ b/tests/test_pdf_to_jsonl.py @@ -1,6 +1,7 @@ # Standard Packages import json import os +import base64 # Internal Packages from khoj.processor.pdf.pdf_to_jsonl import PdfToJsonl @@ -15,7 +16,7 @@ def test_single_page_pdf_to_jsonl(): # Extract Entries from specified Pdf files # Read singlepage.pdf into memory as bytes with open("tests/data/pdf/singlepage.pdf", "rb") as f: - pdf_bytes = f.read() + pdf_bytes = base64.b64encode(f.read()).decode("utf-8") data = {"tests/data/pdf/singlepage.pdf": pdf_bytes} entries, entry_to_file_map = PdfToJsonl.extract_pdf_entries(pdf_files=data) @@ -35,7 +36,7 @@ def test_multi_page_pdf_to_jsonl(): # Act # Extract Entries from specified Pdf files with open("tests/data/pdf/multipage.pdf", "rb") as f: - pdf_bytes = f.read() + pdf_bytes = base64.b64encode(f.read()).decode("utf-8") data = {"tests/data/pdf/multipage.pdf": pdf_bytes} entries, entry_to_file_map = PdfToJsonl.extract_pdf_entries(pdf_files=data) diff --git a/tests/test_rawconfig.py b/tests/test_rawconfig.py index 50baa92a..3f3028b5 100644 --- a/tests/test_rawconfig.py +++ b/tests/test_rawconfig.py @@ -6,18 +6,6 @@ from khoj.utils.rawconfig import TextContentConfig, ImageContentConfig # Test -# ---------------------------------------------------------------------------------------------------- -def test_input_file_or_filter_required_in_text_content_config(): - # Act - with pytest.raises(ValueError): - TextContentConfig( - input_files=None, - input_filter=None, - compressed_jsonl="notes.jsonl", - embeddings_file="note_embeddings.pt", - ) - - # ---------------------------------------------------------------------------------------------------- def test_input_filter_or_directories_required_in_image_content_config(): # Act diff --git a/tests/test_text_search.py b/tests/test_text_search.py index f8c0b688..b1a9aa4d 100644 --- a/tests/test_text_search.py +++ b/tests/test_text_search.py @@ -131,6 +131,65 @@ def test_entry_chunking_by_max_tokens(org_config_with_only_new_file: TextContent assert len(initial_notes_model.corpus_embeddings) == 2 +# ---------------------------------------------------------------------------------------------------- +# @pytest.mark.skip(reason="Flaky due to compressed_jsonl file being rewritten by other tests") +def test_entry_chunking_by_max_tokens_not_full_corpus( + org_config_with_only_new_file: TextContentConfig, search_models: SearchModels +): + # Arrange + # Insert org-mode entry with size exceeding max token limit to new org file + data = { + "readme.org": """ +* Khoj + /Allow natural language search on user content like notes, images using transformer based models/ + + All data is processed locally. User can interface with khoj app via [[./interface/emacs/khoj.el][Emacs]], API or Commandline + +** Dependencies + - Python3 + - [[https://docs.conda.io/en/latest/miniconda.html#latest-miniconda-installer-links][Miniconda]] + +** Install + #+begin_src shell + git clone https://github.com/khoj-ai/khoj && cd khoj + conda env create -f environment.yml + conda activate khoj + #+end_src""" + } + text_search.setup( + OrgToJsonl, + data, + org_config_with_only_new_file, + search_models.text_search.bi_encoder, + regenerate=False, + ) + + max_tokens = 256 + new_file_to_index = Path(org_config_with_only_new_file.input_files[0]) + with open(new_file_to_index, "w") as f: + f.write(f"* Entry more than {max_tokens} words\n") + for index in range(max_tokens + 1): + f.write(f"{index} ") + + data = get_org_files(org_config_with_only_new_file) + + # Act + # reload embeddings, entries, notes model after adding new org-mode file + initial_notes_model = text_search.setup( + OrgToJsonl, + data, + org_config_with_only_new_file, + search_models.text_search.bi_encoder, + regenerate=False, + full_corpus=False, + ) + + # Assert + # verify newly added org-mode entry is split by max tokens + assert len(initial_notes_model.entries) == 5 + assert len(initial_notes_model.corpus_embeddings) == 5 + + # ---------------------------------------------------------------------------------------------------- def test_regenerate_index_with_new_entry( content_config: ContentConfig, search_models: SearchModels, new_org_file: Path