diff --git a/tasks.md b/tasks.md index 3e508f3..d948bbd 100644 --- a/tasks.md +++ b/tasks.md @@ -243,8 +243,7 @@ kaller dem direkte. Samme verktøy, to brukere. - [x] 21.1 `synops-transcribe`: Whisper-transkribering. Input: `--cas-hash --model [--initial-prompt ]`. Output: JSON med segmenter. Skriver segmenter til PG, oppdaterer node metadata. Erstatter `transcribe.rs`. - [x] 21.2 `synops-audio`: FFmpeg-prosessering. Input: `--cas-hash --edl `. Output: ny CAS-hash. Erstatter `audio.rs`. Inkluder parametervalidering (fase 17.2–17.3). -- [~] 21.3 `synops-render`: Tera HTML-rendering. Input: `--node-id --theme `. Output: CAS-hash for rendret HTML. Erstatter `publishing.rs`. - > Påbegynt: 2026-03-18T09:10 +- [x] 21.3 `synops-render`: Tera HTML-rendering. Input: `--node-id --theme `. Output: CAS-hash for rendret HTML. Erstatter `publishing.rs`. - [ ] 21.4 `synops-rss`: RSS/Atom-generering. Input: `--collection-id `. Output: XML til stdout. Erstatter `rss.rs`. - [ ] 21.5 `synops-tts`: Tekst-til-tale. Input: `--text --voice `. Output: CAS-hash for lydfil. Erstatter `tts.rs`. - [ ] 21.6 `synops-summarize`: AI-oppsummering. Input: `--communication-id `. Output: sammendrag som tekst. Erstatter `summarize.rs`. diff --git a/tools/README.md b/tools/README.md index 3ea8c84..b51127e 100644 --- a/tools/README.md +++ b/tools/README.md @@ -9,6 +9,7 @@ eller maskinrommet-API. Ligger i PATH via symlink eller direkte kall. |---------|-------------|--------| | `synops-transcribe` | Whisper-transkribering av lydfil fra CAS | Ferdig | | `synops-audio` | FFmpeg lydprosessering med EDL (cut, normalize, EQ, m.m.) | Ferdig | +| `synops-render` | Tera HTML-rendering til CAS (artikler, forsider) | Ferdig | ## Konvensjoner - Navnekonvensjon: `synops-` (f.eks. `synops-context`) diff --git a/tools/synops-render/Cargo.lock b/tools/synops-render/Cargo.lock new file mode 100644 index 0000000..392bf12 --- /dev/null +++ b/tools/synops-render/Cargo.lock @@ -0,0 +1,2722 @@ +# This file is automatically @generated by Cargo. +# It is not intended for manual editing. +version = 4 + +[[package]] +name = "aho-corasick" +version = "1.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ddd31a130427c27518df266943a5308ed92d4b226cc639f5a8f1002816174301" +dependencies = [ + "memchr", +] + +[[package]] +name = "allocator-api2" +version = "0.2.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "683d7910e743518b0e34f1186f92494becacb047c7b6bf616c96772180fef923" + +[[package]] +name = "android_system_properties" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "819e7219dbd41043ac279b19830f2efc897156490d7fd6ea916720117ee66311" +dependencies = [ + "libc", +] + +[[package]] +name = "anstream" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "824a212faf96e9acacdbd09febd34438f8f711fb84e09a8916013cd7815ca28d" +dependencies = [ + "anstyle", + "anstyle-parse", + "anstyle-query", + "anstyle-wincon", + "colorchoice", + "is_terminal_polyfill", + "utf8parse", +] + +[[package]] +name = "anstyle" +version = "1.0.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "940b3a0ca603d1eade50a4846a2afffd5ef57a9feac2c0e2ec2e14f9ead76000" + +[[package]] +name = "anstyle-parse" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "52ce7f38b242319f7cabaa6813055467063ecdc9d355bbb4ce0c68908cd8130e" +dependencies = [ + "utf8parse", +] + +[[package]] +name = "anstyle-query" +version = "1.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "40c48f72fd53cd289104fc64099abca73db4166ad86ea0b4341abe65af83dadc" +dependencies = [ + "windows-sys 0.61.2", +] + +[[package]] +name = "anstyle-wincon" +version = "3.0.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "291e6a250ff86cd4a820112fb8898808a366d8f9f58ce16d1f538353ad55747d" +dependencies = [ + "anstyle", + "once_cell_polyfill", + "windows-sys 0.61.2", +] + +[[package]] +name = "anyhow" +version = "1.0.102" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f202df86484c868dbad7eaa557ef785d5c66295e41b460ef922eca0723b842c" + +[[package]] +name = "atoi" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f28d99ec8bfea296261ca1af174f24225171fea9664ba9003cbebee704810528" +dependencies = [ + "num-traits", +] + +[[package]] +name = "autocfg" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c08606f8c3cbf4ce6ec8e28fb0014a2c086708fe954eaa885384a6165172e7e8" + +[[package]] +name = "base64" +version = "0.22.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72b3254f16251a8381aa12e40e3c4d2f0199f8c6508fbecb9d91f575e0fbb8c6" + +[[package]] +name = "base64ct" +version = "1.8.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2af50177e190e07a26ab74f8b1efbfe2ef87da2116221318cb1c2e82baf7de06" + +[[package]] +name = "bitflags" +version = "2.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "843867be96c8daad0d758b57df9392b6d8d271134fce549de6ce169ff98a92af" +dependencies = [ + "serde_core", +] + +[[package]] +name = "block-buffer" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71" +dependencies = [ + "generic-array", +] + +[[package]] +name = "bstr" +version = "1.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "63044e1ae8e69f3b5a92c736ca6269b8d12fa7efe39bf34ddb06d102cf0e2cab" +dependencies = [ + "memchr", + "serde", +] + +[[package]] +name = "bumpalo" +version = "3.20.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d20789868f4b01b2f2caec9f5c4e0213b41e3e5702a50157d699ae31ced2fcb" + +[[package]] +name = "byteorder" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" + +[[package]] +name = "bytes" +version = "1.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e748733b7cbc798e1434b6ac524f0c1ff2ab456fe201501e6497c8417a4fc33" + +[[package]] +name = "cc" +version = "1.2.57" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7a0dd1ca384932ff3641c8718a02769f1698e7563dc6974ffd03346116310423" +dependencies = [ + "find-msvc-tools", + "shlex", +] + +[[package]] +name = "cfg-if" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9330f8b2ff13f34540b44e946ef35111825727b38d33286ef986142615121801" + +[[package]] +name = "chrono" +version = "0.4.44" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c673075a2e0e5f4a1dde27ce9dee1ea4558c7ffe648f576438a20ca1d2acc4b0" +dependencies = [ + "iana-time-zone", + "js-sys", + "num-traits", + "serde", + "wasm-bindgen", + "windows-link", +] + +[[package]] +name = "chrono-tz" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93698b29de5e97ad0ae26447b344c482a7284c737d9ddc5f9e52b74a336671bb" +dependencies = [ + "chrono", + "chrono-tz-build", + "phf", +] + +[[package]] +name = "chrono-tz-build" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0c088aee841df9c3041febbb73934cfc39708749bf96dc827e3359cd39ef11b1" +dependencies = [ + "parse-zoneinfo", + "phf", + "phf_codegen", +] + +[[package]] +name = "clap" +version = "4.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b193af5b67834b676abd72466a96c1024e6a6ad978a1f484bd90b85c94041351" +dependencies = [ + "clap_builder", + "clap_derive", +] + +[[package]] +name = "clap_builder" +version = "4.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "714a53001bf66416adb0e2ef5ac857140e7dc3a0c48fb28b2f10762fc4b5069f" +dependencies = [ + "anstream", + "anstyle", + "clap_lex", + "strsim", +] + +[[package]] +name = "clap_derive" +version = "4.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1110bd8a634a1ab8cb04345d8d878267d57c3cf1b38d91b71af6686408bbca6a" +dependencies = [ + "heck", + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "clap_lex" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c8d4a3bb8b1e0c1050499d1815f5ab16d04f0959b233085fb31653fbfc9d98f9" + +[[package]] +name = "colorchoice" +version = "1.0.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d07550c9036bf2ae0c684c4297d503f838287c83c53686d05370d0e139ae570" + +[[package]] +name = "concurrent-queue" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ca0197aee26d1ae37445ee532fefce43251d24cc7c166799f4d46817f1d3973" +dependencies = [ + "crossbeam-utils", +] + +[[package]] +name = "const-oid" +version = "0.9.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c2459377285ad874054d797f3ccebf984978aa39129f6eafde5cdc8315b612f8" + +[[package]] +name = "core-foundation-sys" +version = "0.8.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "773648b94d0e5d620f64f280777445740e61fe701025087ec8b57f45c791888b" + +[[package]] +name = "cpufeatures" +version = "0.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "59ed5838eebb26a2bb2e58f6d5b5316989ae9d08bab10e0e6d103e656d1b0280" +dependencies = [ + "libc", +] + +[[package]] +name = "crc" +version = "3.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5eb8a2a1cd12ab0d987a5d5e825195d372001a4094a0376319d5a0ad71c1ba0d" +dependencies = [ + "crc-catalog", +] + +[[package]] +name = "crc-catalog" +version = "2.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "19d374276b40fb8bbdee95aef7c7fa6b5316ec764510eb64b8dd0e2ed0d7e7f5" + +[[package]] +name = "crossbeam-deque" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9dd111b7b7f7d55b72c0a6ae361660ee5853c9af73f70c3c2ef6858b950e2e51" +dependencies = [ + "crossbeam-epoch", + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-epoch" +version = "0.9.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5b82ac4a3c2ca9c3460964f020e1402edd5753411d7737aa39c3714ad1b5420e" +dependencies = [ + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-queue" +version = "0.3.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0f58bbc28f91df819d0aa2a2c00cd19754769c2fad90579b3592b1c9ba7a3115" +dependencies = [ + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-utils" +version = "0.8.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d0a5c400df2834b80a4c3327b3aad3a4c4cd4de0629063962b03235697506a28" + +[[package]] +name = "crypto-common" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "78c8292055d1c1df0cce5d180393dc8cce0abec0a7102adb6c7b1eef6016d60a" +dependencies = [ + "generic-array", + "typenum", +] + +[[package]] +name = "der" +version = "0.7.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e7c1832837b905bbfb5101e07cc24c8deddf52f93225eee6ead5f4d63d53ddcb" +dependencies = [ + "const-oid", + "pem-rfc7468", + "zeroize", +] + +[[package]] +name = "deunicode" +version = "1.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "abd57806937c9cc163efc8ea3910e00a62e2aeb0b8119f1793a978088f8f6b04" + +[[package]] +name = "digest" +version = "0.10.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292" +dependencies = [ + "block-buffer", + "const-oid", + "crypto-common", + "subtle", +] + +[[package]] +name = "displaydoc" +version = "0.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "97369cbbc041bc366949bc74d34658d6cda5621039731c6310521892a3a20ae0" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "dotenvy" +version = "0.15.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1aaf95b3e5c8f23aa320147307562d361db0ae0d51242340f558153b4eb2439b" + +[[package]] +name = "either" +version = "1.15.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "48c757948c5ede0e46177b7add2e67155f70e33c07fea8284df6576da70b3719" +dependencies = [ + "serde", +] + +[[package]] +name = "equivalent" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "877a4ace8713b0bcf2a4e7eec82529c029f1d0619886d18145fea96c3ffe5c0f" + +[[package]] +name = "errno" +version = "0.3.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "39cab71617ae0d63f51a36d69f866391735b51691dbda63cf6f96d042b63efeb" +dependencies = [ + "libc", + "windows-sys 0.61.2", +] + +[[package]] +name = "etcetera" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "136d1b5283a1ab77bd9257427ffd09d8667ced0570b6f938942bc7568ed5b943" +dependencies = [ + "cfg-if", + "home", + "windows-sys 0.48.0", +] + +[[package]] +name = "event-listener" +version = "5.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e13b66accf52311f30a0db42147dadea9850cb48cd070028831ae5f5d4b856ab" +dependencies = [ + "concurrent-queue", + "parking", + "pin-project-lite", +] + +[[package]] +name = "find-msvc-tools" +version = "0.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5baebc0774151f905a1a2cc41989300b1e6fbb29aff0ceffa1064fdd3088d582" + +[[package]] +name = "flume" +version = "0.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da0e4dd2a88388a1f4ccc7c9ce104604dab68d9f408dc34cd45823d5a9069095" +dependencies = [ + "futures-core", + "futures-sink", + "spin", +] + +[[package]] +name = "foldhash" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d9c4f5dac5e15c24eb999c26181a6ca40b39fe946cbe4c263c7209467bc83af2" + +[[package]] +name = "form_urlencoded" +version = "1.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cb4cb245038516f5f85277875cdaa4f7d2c9a0fa0468de06ed190163b1581fcf" +dependencies = [ + "percent-encoding", +] + +[[package]] +name = "futures-channel" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "07bbe89c50d7a535e539b8c17bc0b49bdb77747034daa8087407d655f3f7cc1d" +dependencies = [ + "futures-core", + "futures-sink", +] + +[[package]] +name = "futures-core" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7e3450815272ef58cec6d564423f6e755e25379b217b0bc688e295ba24df6b1d" + +[[package]] +name = "futures-executor" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "baf29c38818342a3b26b5b923639e7b1f4a61fc5e76102d4b1981c6dc7a7579d" +dependencies = [ + "futures-core", + "futures-task", + "futures-util", +] + +[[package]] +name = "futures-intrusive" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d930c203dd0b6ff06e0201a4a2fe9149b43c684fd4420555b26d21b1a02956f" +dependencies = [ + "futures-core", + "lock_api", + "parking_lot", +] + +[[package]] +name = "futures-io" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cecba35d7ad927e23624b22ad55235f2239cfa44fd10428eecbeba6d6a717718" + +[[package]] +name = "futures-sink" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c39754e157331b013978ec91992bde1ac089843443c49cbc7f46150b0fad0893" + +[[package]] +name = "futures-task" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "037711b3d59c33004d3856fbdc83b99d4ff37a24768fa1be9ce3538a1cde4393" + +[[package]] +name = "futures-util" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "389ca41296e6190b48053de0321d02a77f32f8a5d2461dd38762c0593805c6d6" +dependencies = [ + "futures-core", + "futures-io", + "futures-sink", + "futures-task", + "memchr", + "pin-project-lite", + "slab", +] + +[[package]] +name = "generic-array" +version = "0.14.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85649ca51fd72272d7821adaf274ad91c288277713d9c18820d8499a7ff69e9a" +dependencies = [ + "typenum", + "version_check", +] + +[[package]] +name = "getrandom" +version = "0.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ff2abc00be7fca6ebc474524697ae276ad847ad0a6b3faa4bcb027e9a4614ad0" +dependencies = [ + "cfg-if", + "libc", + "wasi", +] + +[[package]] +name = "getrandom" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0de51e6874e94e7bf76d726fc5d13ba782deca734ff60d5bb2fb2607c7406555" +dependencies = [ + "cfg-if", + "libc", + "r-efi", + "wasip2", + "wasip3", +] + +[[package]] +name = "globset" +version = "0.4.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "52dfc19153a48bde0cbd630453615c8151bce3a5adfac7a0aebfbf0a1e1f57e3" +dependencies = [ + "aho-corasick", + "bstr", + "log", + "regex-automata", + "regex-syntax", +] + +[[package]] +name = "globwalk" +version = "0.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0bf760ebf69878d9fd8f110c89703d90ce35095324d1f1edcb595c63945ee757" +dependencies = [ + "bitflags", + "ignore", + "walkdir", +] + +[[package]] +name = "hashbrown" +version = "0.15.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9229cfe53dfd69f0609a49f65461bd93001ea1ef889cd5529dd176593f5338a1" +dependencies = [ + "allocator-api2", + "equivalent", + "foldhash", +] + +[[package]] +name = "hashbrown" +version = "0.16.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "841d1cc9bed7f9236f321df977030373f4a4163ae1a7dbfe1a51a2c1a51d9100" + +[[package]] +name = "hashlink" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7382cf6263419f2d8df38c55d7da83da5c18aef87fc7a7fc1fb1e344edfe14c1" +dependencies = [ + "hashbrown 0.15.5", +] + +[[package]] +name = "heck" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2304e00983f87ffb38b55b444b5e3b60a884b5d30c0fca7d82fe33449bbe55ea" + +[[package]] +name = "hex" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70" + +[[package]] +name = "hkdf" +version = "0.12.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7b5f8eb2ad728638ea2c7d47a21db23b7b58a72ed6a38256b8a1849f15fbbdf7" +dependencies = [ + "hmac", +] + +[[package]] +name = "hmac" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c49c37c09c17a53d937dfbb742eb3a961d65a994e6bcdcf37e7399d0cc8ab5e" +dependencies = [ + "digest", +] + +[[package]] +name = "home" +version = "0.5.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cc627f471c528ff0c4a49e1d5e60450c8f6461dd6d10ba9dcd3a61d3dff7728d" +dependencies = [ + "windows-sys 0.61.2", +] + +[[package]] +name = "humansize" +version = "2.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6cb51c9a029ddc91b07a787f1d86b53ccfa49b0e86688c946ebe8d3555685dd7" +dependencies = [ + "libm", +] + +[[package]] +name = "iana-time-zone" +version = "0.1.65" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e31bc9ad994ba00e440a8aa5c9ef0ec67d5cb5e5cb0cc7f8b744a35b389cc470" +dependencies = [ + "android_system_properties", + "core-foundation-sys", + "iana-time-zone-haiku", + "js-sys", + "log", + "wasm-bindgen", + "windows-core", +] + +[[package]] +name = "iana-time-zone-haiku" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f31827a206f56af32e590ba56d5d2d085f558508192593743f16b2306495269f" +dependencies = [ + "cc", +] + +[[package]] +name = "icu_collections" +version = "2.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4c6b649701667bbe825c3b7e6388cb521c23d88644678e83c0c4d0a621a34b43" +dependencies = [ + "displaydoc", + "potential_utf", + "yoke", + "zerofrom", + "zerovec", +] + +[[package]] +name = "icu_locale_core" +version = "2.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "edba7861004dd3714265b4db54a3c390e880ab658fec5f7db895fae2046b5bb6" +dependencies = [ + "displaydoc", + "litemap", + "tinystr", + "writeable", + "zerovec", +] + +[[package]] +name = "icu_normalizer" +version = "2.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5f6c8828b67bf8908d82127b2054ea1b4427ff0230ee9141c54251934ab1b599" +dependencies = [ + "icu_collections", + "icu_normalizer_data", + "icu_properties", + "icu_provider", + "smallvec", + "zerovec", +] + +[[package]] +name = "icu_normalizer_data" +version = "2.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7aedcccd01fc5fe81e6b489c15b247b8b0690feb23304303a9e560f37efc560a" + +[[package]] +name = "icu_properties" +version = "2.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "020bfc02fe870ec3a66d93e677ccca0562506e5872c650f893269e08615d74ec" +dependencies = [ + "icu_collections", + "icu_locale_core", + "icu_properties_data", + "icu_provider", + "zerotrie", + "zerovec", +] + +[[package]] +name = "icu_properties_data" +version = "2.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "616c294cf8d725c6afcd8f55abc17c56464ef6211f9ed59cccffe534129c77af" + +[[package]] +name = "icu_provider" +version = "2.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85962cf0ce02e1e0a629cc34e7ca3e373ce20dda4c4d7294bbd0bf1fdb59e614" +dependencies = [ + "displaydoc", + "icu_locale_core", + "writeable", + "yoke", + "zerofrom", + "zerotrie", + "zerovec", +] + +[[package]] +name = "id-arena" +version = "2.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3d3067d79b975e8844ca9eb072e16b31c3c1c36928edf9c6789548c524d0d954" + +[[package]] +name = "idna" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b0875f23caa03898994f6ddc501886a45c7d3d62d04d2d90788d47be1b1e4de" +dependencies = [ + "idna_adapter", + "smallvec", + "utf8_iter", +] + +[[package]] +name = "idna_adapter" +version = "1.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3acae9609540aa318d1bc588455225fb2085b9ed0c4f6bd0d9d5bcd86f1a0344" +dependencies = [ + "icu_normalizer", + "icu_properties", +] + +[[package]] +name = "ignore" +version = "0.4.25" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3d782a365a015e0f5c04902246139249abf769125006fbe7649e2ee88169b4a" +dependencies = [ + "crossbeam-deque", + "globset", + "log", + "memchr", + "regex-automata", + "same-file", + "walkdir", + "winapi-util", +] + +[[package]] +name = "indexmap" +version = "2.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7714e70437a7dc3ac8eb7e6f8df75fd8eb422675fc7678aff7364301092b1017" +dependencies = [ + "equivalent", + "hashbrown 0.16.1", + "serde", + "serde_core", +] + +[[package]] +name = "is_terminal_polyfill" +version = "1.70.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a6cb138bb79a146c1bd460005623e142ef0181e3d0219cb493e02f7d08a35695" + +[[package]] +name = "itoa" +version = "1.0.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "92ecc6618181def0457392ccd0ee51198e065e016d1d527a7ac1b6dc7c1f09d2" + +[[package]] +name = "js-sys" +version = "0.3.91" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b49715b7073f385ba4bc528e5747d02e66cb39c6146efb66b781f131f0fb399c" +dependencies = [ + "once_cell", + "wasm-bindgen", +] + +[[package]] +name = "lazy_static" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbd2bcb4c963f2ddae06a2efc7e9f3591312473c50c6685e1f298068316e66fe" +dependencies = [ + "spin", +] + +[[package]] +name = "leb128fmt" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09edd9e8b54e49e587e4f6295a7d29c3ea94d469cb40ab8ca70b288248a81db2" + +[[package]] +name = "libc" +version = "0.2.183" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b5b646652bf6661599e1da8901b3b9522896f01e736bad5f723fe7a3a27f899d" + +[[package]] +name = "libm" +version = "0.2.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6d2cec3eae94f9f509c767b45932f1ada8350c4bdb85af2fcab4a3c14807981" + +[[package]] +name = "libredox" +version = "0.1.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1744e39d1d6a9948f4f388969627434e31128196de472883b39f148769bfe30a" +dependencies = [ + "bitflags", + "libc", + "plain", + "redox_syscall 0.7.3", +] + +[[package]] +name = "libsqlite3-sys" +version = "0.30.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2e99fb7a497b1e3339bc746195567ed8d3e24945ecd636e3619d20b9de9e9149" +dependencies = [ + "pkg-config", + "vcpkg", +] + +[[package]] +name = "litemap" +version = "0.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6373607a59f0be73a39b6fe456b8192fcc3585f602af20751600e974dd455e77" + +[[package]] +name = "lock_api" +version = "0.4.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "224399e74b87b5f3557511d98dff8b14089b3dadafcab6bb93eab67d3aace965" +dependencies = [ + "scopeguard", +] + +[[package]] +name = "log" +version = "0.4.29" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5e5032e24019045c762d3c0f28f5b6b8bbf38563a65908389bf7978758920897" + +[[package]] +name = "matchers" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d1525a2a28c7f4fa0fc98bb91ae755d1e2d1505079e05539e35bc876b5d65ae9" +dependencies = [ + "regex-automata", +] + +[[package]] +name = "md-5" +version = "0.10.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d89e7ee0cfbedfc4da3340218492196241d89eefb6dab27de5df917a6d2e78cf" +dependencies = [ + "cfg-if", + "digest", +] + +[[package]] +name = "memchr" +version = "2.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f8ca58f447f06ed17d5fc4043ce1b10dd205e060fb3ce5b979b8ed8e59ff3f79" + +[[package]] +name = "mio" +version = "1.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a69bcab0ad47271a0234d9422b131806bf3968021e5dc9328caf2d4cd58557fc" +dependencies = [ + "libc", + "wasi", + "windows-sys 0.61.2", +] + +[[package]] +name = "nu-ansi-term" +version = "0.50.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7957b9740744892f114936ab4a57b3f487491bbeafaf8083688b16841a4240e5" +dependencies = [ + "windows-sys 0.61.2", +] + +[[package]] +name = "num-bigint-dig" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e661dda6640fad38e827a6d4a310ff4763082116fe217f279885c97f511bb0b7" +dependencies = [ + "lazy_static", + "libm", + "num-integer", + "num-iter", + "num-traits", + "rand", + "smallvec", + "zeroize", +] + +[[package]] +name = "num-integer" +version = "0.1.46" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7969661fd2958a5cb096e56c8e1ad0444ac2bbcd0061bd28660485a44879858f" +dependencies = [ + "num-traits", +] + +[[package]] +name = "num-iter" +version = "0.1.45" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1429034a0490724d0075ebb2bc9e875d6503c3cf69e235a8941aa757d83ef5bf" +dependencies = [ + "autocfg", + "num-integer", + "num-traits", +] + +[[package]] +name = "num-traits" +version = "0.2.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "071dfc062690e90b734c0b2273ce72ad0ffa95f0c74596bc250dcfd960262841" +dependencies = [ + "autocfg", + "libm", +] + +[[package]] +name = "once_cell" +version = "1.21.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9f7c3e4beb33f85d45ae3e3a1792185706c8e16d043238c593331cc7cd313b50" + +[[package]] +name = "once_cell_polyfill" +version = "1.70.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "384b8ab6d37215f3c5301a95a4accb5d64aa607f1fcb26a11b5303878451b4fe" + +[[package]] +name = "parking" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f38d5652c16fde515bb1ecef450ab0f6a219d619a7274976324d5e377f7dceba" + +[[package]] +name = "parking_lot" +version = "0.12.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93857453250e3077bd71ff98b6a65ea6621a19bb0f559a85248955ac12c45a1a" +dependencies = [ + "lock_api", + "parking_lot_core", +] + +[[package]] +name = "parking_lot_core" +version = "0.9.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2621685985a2ebf1c516881c026032ac7deafcda1a2c9b7850dc81e3dfcb64c1" +dependencies = [ + "cfg-if", + "libc", + "redox_syscall 0.5.18", + "smallvec", + "windows-link", +] + +[[package]] +name = "parse-zoneinfo" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1f2a05b18d44e2957b88f96ba460715e295bc1d7510468a2f3d3b44535d26c24" +dependencies = [ + "regex", +] + +[[package]] +name = "pem-rfc7468" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "88b39c9bfcfc231068454382784bb460aae594343fb030d46e9f50a645418412" +dependencies = [ + "base64ct", +] + +[[package]] +name = "percent-encoding" +version = "2.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9b4f627cb1b25917193a259e49bdad08f671f8d9708acfd5fe0a8c1455d87220" + +[[package]] +name = "pest" +version = "2.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e0848c601009d37dfa3430c4666e147e49cdcf1b92ecd3e63657d8a5f19da662" +dependencies = [ + "memchr", + "ucd-trie", +] + +[[package]] +name = "pest_derive" +version = "2.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "11f486f1ea21e6c10ed15d5a7c77165d0ee443402f0780849d1768e7d9d6fe77" +dependencies = [ + "pest", + "pest_generator", +] + +[[package]] +name = "pest_generator" +version = "2.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8040c4647b13b210a963c1ed407c1ff4fdfa01c31d6d2a098218702e6664f94f" +dependencies = [ + "pest", + "pest_meta", + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "pest_meta" +version = "2.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "89815c69d36021a140146f26659a81d6c2afa33d216d736dd4be5381a7362220" +dependencies = [ + "pest", + "sha2", +] + +[[package]] +name = "phf" +version = "0.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fd6780a80ae0c52cc120a26a1a42c1ae51b247a253e4e06113d23d2c2edd078" +dependencies = [ + "phf_shared", +] + +[[package]] +name = "phf_codegen" +version = "0.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aef8048c789fa5e851558d709946d6d79a8ff88c0440c587967f8e94bfb1216a" +dependencies = [ + "phf_generator", + "phf_shared", +] + +[[package]] +name = "phf_generator" +version = "0.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c80231409c20246a13fddb31776fb942c38553c51e871f8cbd687a4cfb5843d" +dependencies = [ + "phf_shared", + "rand", +] + +[[package]] +name = "phf_shared" +version = "0.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "67eabc2ef2a60eb7faa00097bd1ffdb5bd28e62bf39990626a582201b7a754e5" +dependencies = [ + "siphasher", +] + +[[package]] +name = "pin-project-lite" +version = "0.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a89322df9ebe1c1578d689c92318e070967d1042b512afbe49518723f4e6d5cd" + +[[package]] +name = "pkcs1" +version = "0.7.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c8ffb9f10fa047879315e6625af03c164b16962a5368d724ed16323b68ace47f" +dependencies = [ + "der", + "pkcs8", + "spki", +] + +[[package]] +name = "pkcs8" +version = "0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f950b2377845cebe5cf8b5165cb3cc1a5e0fa5cfa3e1f7f55707d8fd82e0a7b7" +dependencies = [ + "der", + "spki", +] + +[[package]] +name = "pkg-config" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7edddbd0b52d732b21ad9a5fab5c704c14cd949e5e9a1ec5929a24fded1b904c" + +[[package]] +name = "plain" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b4596b6d070b27117e987119b4dac604f3c58cfb0b191112e24771b2faeac1a6" + +[[package]] +name = "potential_utf" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b73949432f5e2a09657003c25bca5e19a0e9c84f8058ca374f49e0ebe605af77" +dependencies = [ + "zerovec", +] + +[[package]] +name = "ppv-lite86" +version = "0.2.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85eae3c4ed2f50dcfe72643da4befc30deadb458a9b590d720cde2f2b1e97da9" +dependencies = [ + "zerocopy", +] + +[[package]] +name = "prettyplease" +version = "0.2.37" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "479ca8adacdd7ce8f1fb39ce9ecccbfe93a3f1344b3d0d97f20bc0196208f62b" +dependencies = [ + "proc-macro2", + "syn", +] + +[[package]] +name = "proc-macro2" +version = "1.0.106" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8fd00f0bb2e90d81d1044c2b32617f68fcb9fa3bb7640c23e9c748e53fb30934" +dependencies = [ + "unicode-ident", +] + +[[package]] +name = "quote" +version = "1.0.45" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "41f2619966050689382d2b44f664f4bc593e129785a36d6ee376ddf37259b924" +dependencies = [ + "proc-macro2", +] + +[[package]] +name = "r-efi" +version = "6.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f8dcc9c7d52a811697d2151c701e0d08956f92b0e24136cf4cf27b57a6a0d9bf" + +[[package]] +name = "rand" +version = "0.8.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404" +dependencies = [ + "libc", + "rand_chacha", + "rand_core", +] + +[[package]] +name = "rand_chacha" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88" +dependencies = [ + "ppv-lite86", + "rand_core", +] + +[[package]] +name = "rand_core" +version = "0.6.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c" +dependencies = [ + "getrandom 0.2.17", +] + +[[package]] +name = "redox_syscall" +version = "0.5.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed2bf2547551a7053d6fdfafda3f938979645c44812fbfcda098faae3f1a362d" +dependencies = [ + "bitflags", +] + +[[package]] +name = "redox_syscall" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6ce70a74e890531977d37e532c34d45e9055d2409ed08ddba14529471ed0be16" +dependencies = [ + "bitflags", +] + +[[package]] +name = "regex" +version = "1.12.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e10754a14b9137dd7b1e3e5b0493cc9171fdd105e0ab477f51b72e7f3ac0e276" +dependencies = [ + "aho-corasick", + "memchr", + "regex-automata", + "regex-syntax", +] + +[[package]] +name = "regex-automata" +version = "0.4.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6e1dd4122fc1595e8162618945476892eefca7b88c52820e74af6262213cae8f" +dependencies = [ + "aho-corasick", + "memchr", + "regex-syntax", +] + +[[package]] +name = "regex-syntax" +version = "0.8.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc897dd8d9e8bd1ed8cdad82b5966c3e0ecae09fb1907d58efaa013543185d0a" + +[[package]] +name = "ring" +version = "0.17.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a4689e6c2294d81e88dc6261c768b63bc4fcdb852be6d1352498b114f61383b7" +dependencies = [ + "cc", + "cfg-if", + "getrandom 0.2.17", + "libc", + "untrusted", + "windows-sys 0.52.0", +] + +[[package]] +name = "rsa" +version = "0.9.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8573f03f5883dcaebdfcf4725caa1ecb9c15b2ef50c43a07b816e06799bb12d" +dependencies = [ + "const-oid", + "digest", + "num-bigint-dig", + "num-integer", + "num-traits", + "pkcs1", + "pkcs8", + "rand_core", + "signature", + "spki", + "subtle", + "zeroize", +] + +[[package]] +name = "rustls" +version = "0.23.37" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "758025cb5fccfd3bc2fd74708fd4682be41d99e5dff73c377c0646c6012c73a4" +dependencies = [ + "once_cell", + "ring", + "rustls-pki-types", + "rustls-webpki", + "subtle", + "zeroize", +] + +[[package]] +name = "rustls-pki-types" +version = "1.14.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "be040f8b0a225e40375822a563fa9524378b9d63112f53e19ffff34df5d33fdd" +dependencies = [ + "zeroize", +] + +[[package]] +name = "rustls-webpki" +version = "0.103.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d7df23109aa6c1567d1c575b9952556388da57401e4ace1d15f79eedad0d8f53" +dependencies = [ + "ring", + "rustls-pki-types", + "untrusted", +] + +[[package]] +name = "rustversion" +version = "1.0.22" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b39cdef0fa800fc44525c84ccb54a029961a8215f9619753635a9c0d2538d46d" + +[[package]] +name = "ryu" +version = "1.0.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9774ba4a74de5f7b1c1451ed6cd5285a32eddb5cccb8cc655a4e50009e06477f" + +[[package]] +name = "same-file" +version = "1.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93fc1dc3aaa9bfed95e02e6eadabb4baf7e3078b0bd1b4d7b6b0b68378900502" +dependencies = [ + "winapi-util", +] + +[[package]] +name = "scopeguard" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49" + +[[package]] +name = "semver" +version = "1.0.27" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d767eb0aabc880b29956c35734170f26ed551a859dbd361d140cdbeca61ab1e2" + +[[package]] +name = "serde" +version = "1.0.228" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9a8e94ea7f378bd32cbbd37198a4a91436180c5bb472411e48b5ec2e2124ae9e" +dependencies = [ + "serde_core", + "serde_derive", +] + +[[package]] +name = "serde_core" +version = "1.0.228" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "41d385c7d4ca58e59fc732af25c3983b67ac852c1a25000afe1175de458b67ad" +dependencies = [ + "serde_derive", +] + +[[package]] +name = "serde_derive" +version = "1.0.228" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d540f220d3187173da220f885ab66608367b6574e925011a9353e4badda91d79" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "serde_json" +version = "1.0.149" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "83fc039473c5595ace860d8c4fafa220ff474b3fc6bfdb4293327f1a37e94d86" +dependencies = [ + "itoa", + "memchr", + "serde", + "serde_core", + "zmij", +] + +[[package]] +name = "serde_urlencoded" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3491c14715ca2294c4d6a88f15e84739788c1d030eed8c110436aafdaa2f3fd" +dependencies = [ + "form_urlencoded", + "itoa", + "ryu", + "serde", +] + +[[package]] +name = "sha1" +version = "0.10.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3bf829a2d51ab4a5ddf1352d8470c140cadc8301b2ae1789db023f01cedd6ba" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest", +] + +[[package]] +name = "sha2" +version = "0.10.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a7507d819769d01a365ab707794a4084392c824f54a7a6a7862f8c3d0892b283" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest", +] + +[[package]] +name = "sharded-slab" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f40ca3c46823713e0d4209592e8d6e826aa57e928f09752619fc696c499637f6" +dependencies = [ + "lazy_static", +] + +[[package]] +name = "shlex" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64" + +[[package]] +name = "signal-hook-registry" +version = "1.4.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c4db69cba1110affc0e9f7bcd48bbf87b3f4fc7c61fc9155afd4c469eb3d6c1b" +dependencies = [ + "errno", + "libc", +] + +[[package]] +name = "signature" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "77549399552de45a898a580c1b41d445bf730df867cc44e6c0233bbc4b8329de" +dependencies = [ + "digest", + "rand_core", +] + +[[package]] +name = "siphasher" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b2aa850e253778c88a04c3d7323b043aeda9d3e30d5971937c1855769763678e" + +[[package]] +name = "slab" +version = "0.4.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0c790de23124f9ab44544d7ac05d60440adc586479ce501c1d6d7da3cd8c9cf5" + +[[package]] +name = "slug" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "882a80f72ee45de3cc9a5afeb2da0331d58df69e4e7d8eeb5d3c7784ae67e724" +dependencies = [ + "deunicode", + "wasm-bindgen", +] + +[[package]] +name = "smallvec" +version = "1.15.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "67b1b7a3b5fe4f1376887184045fcf45c69e92af734b7aaddc05fb777b6fbd03" +dependencies = [ + "serde", +] + +[[package]] +name = "socket2" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3a766e1110788c36f4fa1c2b71b387a7815aa65f88ce0229841826633d93723e" +dependencies = [ + "libc", + "windows-sys 0.61.2", +] + +[[package]] +name = "spin" +version = "0.9.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6980e8d7511241f8acf4aebddbb1ff938df5eebe98691418c4468d0b72a96a67" +dependencies = [ + "lock_api", +] + +[[package]] +name = "spki" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d91ed6c858b01f942cd56b37a94b3e0a1798290327d1236e4d9cf4eaca44d29d" +dependencies = [ + "base64ct", + "der", +] + +[[package]] +name = "sqlx" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fefb893899429669dcdd979aff487bd78f4064e5e7907e4269081e0ef7d97dc" +dependencies = [ + "sqlx-core", + "sqlx-macros", + "sqlx-mysql", + "sqlx-postgres", + "sqlx-sqlite", +] + +[[package]] +name = "sqlx-core" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ee6798b1838b6a0f69c007c133b8df5866302197e404e8b6ee8ed3e3a5e68dc6" +dependencies = [ + "base64", + "bytes", + "chrono", + "crc", + "crossbeam-queue", + "either", + "event-listener", + "futures-core", + "futures-intrusive", + "futures-io", + "futures-util", + "hashbrown 0.15.5", + "hashlink", + "indexmap", + "log", + "memchr", + "once_cell", + "percent-encoding", + "rustls", + "serde", + "serde_json", + "sha2", + "smallvec", + "thiserror", + "tokio", + "tokio-stream", + "tracing", + "url", + "uuid", + "webpki-roots 0.26.11", +] + +[[package]] +name = "sqlx-macros" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a2d452988ccaacfbf5e0bdbc348fb91d7c8af5bee192173ac3636b5fb6e6715d" +dependencies = [ + "proc-macro2", + "quote", + "sqlx-core", + "sqlx-macros-core", + "syn", +] + +[[package]] +name = "sqlx-macros-core" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "19a9c1841124ac5a61741f96e1d9e2ec77424bf323962dd894bdb93f37d5219b" +dependencies = [ + "dotenvy", + "either", + "heck", + "hex", + "once_cell", + "proc-macro2", + "quote", + "serde", + "serde_json", + "sha2", + "sqlx-core", + "sqlx-mysql", + "sqlx-postgres", + "sqlx-sqlite", + "syn", + "tokio", + "url", +] + +[[package]] +name = "sqlx-mysql" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aa003f0038df784eb8fecbbac13affe3da23b45194bd57dba231c8f48199c526" +dependencies = [ + "atoi", + "base64", + "bitflags", + "byteorder", + "bytes", + "chrono", + "crc", + "digest", + "dotenvy", + "either", + "futures-channel", + "futures-core", + "futures-io", + "futures-util", + "generic-array", + "hex", + "hkdf", + "hmac", + "itoa", + "log", + "md-5", + "memchr", + "once_cell", + "percent-encoding", + "rand", + "rsa", + "serde", + "sha1", + "sha2", + "smallvec", + "sqlx-core", + "stringprep", + "thiserror", + "tracing", + "uuid", + "whoami", +] + +[[package]] +name = "sqlx-postgres" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "db58fcd5a53cf07c184b154801ff91347e4c30d17a3562a635ff028ad5deda46" +dependencies = [ + "atoi", + "base64", + "bitflags", + "byteorder", + "chrono", + "crc", + "dotenvy", + "etcetera", + "futures-channel", + "futures-core", + "futures-util", + "hex", + "hkdf", + "hmac", + "home", + "itoa", + "log", + "md-5", + "memchr", + "once_cell", + "rand", + "serde", + "serde_json", + "sha2", + "smallvec", + "sqlx-core", + "stringprep", + "thiserror", + "tracing", + "uuid", + "whoami", +] + +[[package]] +name = "sqlx-sqlite" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c2d12fe70b2c1b4401038055f90f151b78208de1f9f89a7dbfd41587a10c3eea" +dependencies = [ + "atoi", + "chrono", + "flume", + "futures-channel", + "futures-core", + "futures-executor", + "futures-intrusive", + "futures-util", + "libsqlite3-sys", + "log", + "percent-encoding", + "serde", + "serde_urlencoded", + "sqlx-core", + "thiserror", + "tracing", + "url", + "uuid", +] + +[[package]] +name = "stable_deref_trait" +version = "1.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6ce2be8dc25455e1f91df71bfa12ad37d7af1092ae736f3a6cd0e37bc7810596" + +[[package]] +name = "stringprep" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7b4df3d392d81bd458a8a621b8bffbd2302a12ffe288a9d931670948749463b1" +dependencies = [ + "unicode-bidi", + "unicode-normalization", + "unicode-properties", +] + +[[package]] +name = "strsim" +version = "0.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7da8b5736845d9f2fcb837ea5d9e2628564b3b043a70948a3f0b778838c5fb4f" + +[[package]] +name = "subtle" +version = "2.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "13c2bddecc57b384dee18652358fb23172facb8a2c51ccc10d74c157bdea3292" + +[[package]] +name = "syn" +version = "2.0.117" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e665b8803e7b1d2a727f4023456bbbbe74da67099c585258af0ad9c5013b9b99" +dependencies = [ + "proc-macro2", + "quote", + "unicode-ident", +] + +[[package]] +name = "synops-render" +version = "0.1.0" +dependencies = [ + "chrono", + "clap", + "hex", + "serde", + "serde_json", + "sha2", + "sqlx", + "tera", + "tokio", + "tracing", + "tracing-subscriber", + "uuid", +] + +[[package]] +name = "synstructure" +version = "0.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "728a70f3dbaf5bab7f0c4b1ac8d7ae5ea60a4b5549c8a5914361c99147a709d2" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "tera" +version = "1.20.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e8004bca281f2d32df3bacd59bc67b312cb4c70cea46cbd79dbe8ac5ed206722" +dependencies = [ + "chrono", + "chrono-tz", + "globwalk", + "humansize", + "lazy_static", + "percent-encoding", + "pest", + "pest_derive", + "rand", + "regex", + "serde", + "serde_json", + "slug", + "unicode-segmentation", +] + +[[package]] +name = "thiserror" +version = "2.0.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4288b5bcbc7920c07a1149a35cf9590a2aa808e0bc1eafaade0b80947865fbc4" +dependencies = [ + "thiserror-impl", +] + +[[package]] +name = "thiserror-impl" +version = "2.0.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ebc4ee7f67670e9b64d05fa4253e753e016c6c95ff35b89b7941d6b856dec1d5" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "thread_local" +version = "1.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f60246a4944f24f6e018aa17cdeffb7818b76356965d03b07d6a9886e8962185" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "tinystr" +version = "0.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "42d3e9c45c09de15d06dd8acf5f4e0e399e85927b7f00711024eb7ae10fa4869" +dependencies = [ + "displaydoc", + "zerovec", +] + +[[package]] +name = "tinyvec" +version = "1.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3e61e67053d25a4e82c844e8424039d9745781b3fc4f32b8d55ed50f5f667ef3" +dependencies = [ + "tinyvec_macros", +] + +[[package]] +name = "tinyvec_macros" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" + +[[package]] +name = "tokio" +version = "1.50.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "27ad5e34374e03cfffefc301becb44e9dc3c17584f414349ebe29ed26661822d" +dependencies = [ + "bytes", + "libc", + "mio", + "parking_lot", + "pin-project-lite", + "signal-hook-registry", + "socket2", + "tokio-macros", + "windows-sys 0.61.2", +] + +[[package]] +name = "tokio-macros" +version = "2.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c55a2eff8b69ce66c84f85e1da1c233edc36ceb85a2058d11b0d6a3c7e7569c" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "tokio-stream" +version = "0.1.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "32da49809aab5c3bc678af03902d4ccddea2a87d028d86392a4b1560c6906c70" +dependencies = [ + "futures-core", + "pin-project-lite", + "tokio", +] + +[[package]] +name = "tracing" +version = "0.1.44" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "63e71662fa4b2a2c3a26f570f037eb95bb1f85397f3cd8076caed2f026a6d100" +dependencies = [ + "log", + "pin-project-lite", + "tracing-attributes", + "tracing-core", +] + +[[package]] +name = "tracing-attributes" +version = "0.1.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7490cfa5ec963746568740651ac6781f701c9c5ea257c58e057f3ba8cf69e8da" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "tracing-core" +version = "0.1.36" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "db97caf9d906fbde555dd62fa95ddba9eecfd14cb388e4f491a66d74cd5fb79a" +dependencies = [ + "once_cell", + "valuable", +] + +[[package]] +name = "tracing-log" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ee855f1f400bd0e5c02d150ae5de3840039a3f54b025156404e34c23c03f47c3" +dependencies = [ + "log", + "once_cell", + "tracing-core", +] + +[[package]] +name = "tracing-subscriber" +version = "0.3.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cb7f578e5945fb242538965c2d0b04418d38ec25c79d160cd279bf0731c8d319" +dependencies = [ + "matchers", + "nu-ansi-term", + "once_cell", + "regex-automata", + "sharded-slab", + "smallvec", + "thread_local", + "tracing", + "tracing-core", + "tracing-log", +] + +[[package]] +name = "typenum" +version = "1.19.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "562d481066bde0658276a35467c4af00bdc6ee726305698a55b86e61d7ad82bb" + +[[package]] +name = "ucd-trie" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2896d95c02a80c6d6a5d6e953d479f5ddf2dfdb6a244441010e373ac0fb88971" + +[[package]] +name = "unicode-bidi" +version = "0.3.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c1cb5db39152898a79168971543b1cb5020dff7fe43c8dc468b0885f5e29df5" + +[[package]] +name = "unicode-ident" +version = "1.0.24" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6e4313cd5fcd3dad5cafa179702e2b244f760991f45397d14d4ebf38247da75" + +[[package]] +name = "unicode-normalization" +version = "0.1.25" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5fd4f6878c9cb28d874b009da9e8d183b5abc80117c40bbd187a1fde336be6e8" +dependencies = [ + "tinyvec", +] + +[[package]] +name = "unicode-properties" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7df058c713841ad818f1dc5d3fd88063241cc61f49f5fbea4b951e8cf5a8d71d" + +[[package]] +name = "unicode-segmentation" +version = "1.12.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f6ccf251212114b54433ec949fd6a7841275f9ada20dddd2f29e9ceea4501493" + +[[package]] +name = "unicode-xid" +version = "0.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ebc1c04c71510c7f702b52b7c350734c9ff1295c464a03335b00bb84fc54f853" + +[[package]] +name = "untrusted" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8ecb6da28b8a351d773b68d5825ac39017e680750f980f3a1a85cd8dd28a47c1" + +[[package]] +name = "url" +version = "2.5.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ff67a8a4397373c3ef660812acab3268222035010ab8680ec4215f38ba3d0eed" +dependencies = [ + "form_urlencoded", + "idna", + "percent-encoding", + "serde", +] + +[[package]] +name = "utf8_iter" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6c140620e7ffbb22c2dee59cafe6084a59b5ffc27a8859a5f0d494b5d52b6be" + +[[package]] +name = "utf8parse" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "06abde3611657adf66d383f00b093d7faecc7fa57071cce2578660c9f1010821" + +[[package]] +name = "uuid" +version = "1.22.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a68d3c8f01c0cfa54a75291d83601161799e4a89a39e0929f4b0354d88757a37" +dependencies = [ + "getrandom 0.4.2", + "js-sys", + "serde_core", + "wasm-bindgen", +] + +[[package]] +name = "valuable" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba73ea9cf16a25df0c8caa16c51acb937d5712a8429db78a3ee29d5dcacd3a65" + +[[package]] +name = "vcpkg" +version = "0.2.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "accd4ea62f7bb7a82fe23066fb0957d48ef677f6eeb8215f372f52e48bb32426" + +[[package]] +name = "version_check" +version = "0.9.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b928f33d975fc6ad9f86c8f283853ad26bdd5b10b7f1542aa2fa15e2289105a" + +[[package]] +name = "walkdir" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "29790946404f91d9c5d06f9874efddea1dc06c5efe94541a7d6863108e3a5e4b" +dependencies = [ + "same-file", + "winapi-util", +] + +[[package]] +name = "wasi" +version = "0.11.1+wasi-snapshot-preview1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ccf3ec651a847eb01de73ccad15eb7d99f80485de043efb2f370cd654f4ea44b" + +[[package]] +name = "wasip2" +version = "1.0.2+wasi-0.2.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9517f9239f02c069db75e65f174b3da828fe5f5b945c4dd26bd25d89c03ebcf5" +dependencies = [ + "wit-bindgen", +] + +[[package]] +name = "wasip3" +version = "0.4.0+wasi-0.3.0-rc-2026-01-06" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5428f8bf88ea5ddc08faddef2ac4a67e390b88186c703ce6dbd955e1c145aca5" +dependencies = [ + "wit-bindgen", +] + +[[package]] +name = "wasite" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8dad83b4f25e74f184f64c43b150b91efe7647395b42289f38e50566d82855b" + +[[package]] +name = "wasm-bindgen" +version = "0.2.114" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6532f9a5c1ece3798cb1c2cfdba640b9b3ba884f5db45973a6f442510a87d38e" +dependencies = [ + "cfg-if", + "once_cell", + "rustversion", + "wasm-bindgen-macro", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-macro" +version = "0.2.114" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "18a2d50fcf105fb33bb15f00e7a77b772945a2ee45dcf454961fd843e74c18e6" +dependencies = [ + "quote", + "wasm-bindgen-macro-support", +] + +[[package]] +name = "wasm-bindgen-macro-support" +version = "0.2.114" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "03ce4caeaac547cdf713d280eda22a730824dd11e6b8c3ca9e42247b25c631e3" +dependencies = [ + "bumpalo", + "proc-macro2", + "quote", + "syn", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-shared" +version = "0.2.114" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "75a326b8c223ee17883a4251907455a2431acc2791c98c26279376490c378c16" +dependencies = [ + "unicode-ident", +] + +[[package]] +name = "wasm-encoder" +version = "0.244.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "990065f2fe63003fe337b932cfb5e3b80e0b4d0f5ff650e6985b1048f62c8319" +dependencies = [ + "leb128fmt", + "wasmparser", +] + +[[package]] +name = "wasm-metadata" +version = "0.244.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bb0e353e6a2fbdc176932bbaab493762eb1255a7900fe0fea1a2f96c296cc909" +dependencies = [ + "anyhow", + "indexmap", + "wasm-encoder", + "wasmparser", +] + +[[package]] +name = "wasmparser" +version = "0.244.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "47b807c72e1bac69382b3a6fb3dbe8ea4c0ed87ff5629b8685ae6b9a611028fe" +dependencies = [ + "bitflags", + "hashbrown 0.15.5", + "indexmap", + "semver", +] + +[[package]] +name = "webpki-roots" +version = "0.26.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "521bc38abb08001b01866da9f51eb7c5d647a19260e00054a8c7fd5f9e57f7a9" +dependencies = [ + "webpki-roots 1.0.6", +] + +[[package]] +name = "webpki-roots" +version = "1.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "22cfaf3c063993ff62e73cb4311efde4db1efb31ab78a3e5c457939ad5cc0bed" +dependencies = [ + "rustls-pki-types", +] + +[[package]] +name = "whoami" +version = "1.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d4a4db5077702ca3015d3d02d74974948aba2ad9e12ab7df718ee64ccd7e97d" +dependencies = [ + "libredox", + "wasite", +] + +[[package]] +name = "winapi-util" +version = "0.1.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c2a7b1c03c876122aa43f3020e6c3c3ee5c05081c9a00739faf7503aeba10d22" +dependencies = [ + "windows-sys 0.61.2", +] + +[[package]] +name = "windows-core" +version = "0.62.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8e83a14d34d0623b51dce9581199302a221863196a1dde71a7663a4c2be9deb" +dependencies = [ + "windows-implement", + "windows-interface", + "windows-link", + "windows-result", + "windows-strings", +] + +[[package]] +name = "windows-implement" +version = "0.60.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "053e2e040ab57b9dc951b72c264860db7eb3b0200ba345b4e4c3b14f67855ddf" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "windows-interface" +version = "0.59.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f316c4a2570ba26bbec722032c4099d8c8bc095efccdc15688708623367e358" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "windows-link" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f0805222e57f7521d6a62e36fa9163bc891acd422f971defe97d64e70d0a4fe5" + +[[package]] +name = "windows-result" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7781fa89eaf60850ac3d2da7af8e5242a5ea78d1a11c49bf2910bb5a73853eb5" +dependencies = [ + "windows-link", +] + +[[package]] +name = "windows-strings" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7837d08f69c77cf6b07689544538e017c1bfcf57e34b4c0ff58e6c2cd3b37091" +dependencies = [ + "windows-link", +] + +[[package]] +name = "windows-sys" +version = "0.48.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "677d2418bec65e3338edb076e806bc1ec15693c5d0104683f2efe857f61056a9" +dependencies = [ + "windows-targets 0.48.5", +] + +[[package]] +name = "windows-sys" +version = "0.52.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "282be5f36a8ce781fad8c8ae18fa3f9beff57ec1b52cb3de0789201425d9a33d" +dependencies = [ + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-sys" +version = "0.61.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ae137229bcbd6cdf0f7b80a31df61766145077ddf49416a728b02cb3921ff3fc" +dependencies = [ + "windows-link", +] + +[[package]] +name = "windows-targets" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9a2fa6e2155d7247be68c096456083145c183cbbbc2764150dda45a87197940c" +dependencies = [ + "windows_aarch64_gnullvm 0.48.5", + "windows_aarch64_msvc 0.48.5", + "windows_i686_gnu 0.48.5", + "windows_i686_msvc 0.48.5", + "windows_x86_64_gnu 0.48.5", + "windows_x86_64_gnullvm 0.48.5", + "windows_x86_64_msvc 0.48.5", +] + +[[package]] +name = "windows-targets" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9b724f72796e036ab90c1021d4780d4d3d648aca59e491e6b98e725b84e99973" +dependencies = [ + "windows_aarch64_gnullvm 0.52.6", + "windows_aarch64_msvc 0.52.6", + "windows_i686_gnu 0.52.6", + "windows_i686_gnullvm", + "windows_i686_msvc 0.52.6", + "windows_x86_64_gnu 0.52.6", + "windows_x86_64_gnullvm 0.52.6", + "windows_x86_64_msvc 0.52.6", +] + +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2b38e32f0abccf9987a4e3079dfb67dcd799fb61361e53e2882c3cbaf0d905d8" + +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "32a4622180e7a0ec044bb555404c800bc9fd9ec262ec147edd5989ccd0c02cd3" + +[[package]] +name = "windows_aarch64_msvc" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc35310971f3b2dbbf3f0690a219f40e2d9afcf64f9ab7cc1be722937c26b4bc" + +[[package]] +name = "windows_aarch64_msvc" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09ec2a7bb152e2252b53fa7803150007879548bc709c039df7627cabbd05d469" + +[[package]] +name = "windows_i686_gnu" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a75915e7def60c94dcef72200b9a8e58e5091744960da64ec734a6c6e9b3743e" + +[[package]] +name = "windows_i686_gnu" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e9b5ad5ab802e97eb8e295ac6720e509ee4c243f69d781394014ebfe8bbfa0b" + +[[package]] +name = "windows_i686_gnullvm" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0eee52d38c090b3caa76c563b86c3a4bd71ef1a819287c19d586d7334ae8ed66" + +[[package]] +name = "windows_i686_msvc" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f55c233f70c4b27f66c523580f78f1004e8b5a8b659e05a4eb49d4166cca406" + +[[package]] +name = "windows_i686_msvc" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "240948bc05c5e7c6dabba28bf89d89ffce3e303022809e73deaefe4f6ec56c66" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "53d40abd2583d23e4718fddf1ebec84dbff8381c07cae67ff7768bbf19c6718e" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "147a5c80aabfbf0c7d901cb5895d1de30ef2907eb21fbbab29ca94c5b08b1a78" + +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b7b52767868a23d5bab768e390dc5f5c55825b6d30b86c844ff2dc7414044cc" + +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "24d5b23dc417412679681396f2b49f3de8c1473deb516bd34410872eff51ed0d" + +[[package]] +name = "windows_x86_64_msvc" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed94fce61571a4006852b7389a063ab983c02eb1bb37b47f8272ce92d06d9538" + +[[package]] +name = "windows_x86_64_msvc" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec" + +[[package]] +name = "wit-bindgen" +version = "0.51.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d7249219f66ced02969388cf2bb044a09756a083d0fab1e566056b04d9fbcaa5" +dependencies = [ + "wit-bindgen-rust-macro", +] + +[[package]] +name = "wit-bindgen-core" +version = "0.51.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ea61de684c3ea68cb082b7a88508a8b27fcc8b797d738bfc99a82facf1d752dc" +dependencies = [ + "anyhow", + "heck", + "wit-parser", +] + +[[package]] +name = "wit-bindgen-rust" +version = "0.51.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b7c566e0f4b284dd6561c786d9cb0142da491f46a9fbed79ea69cdad5db17f21" +dependencies = [ + "anyhow", + "heck", + "indexmap", + "prettyplease", + "syn", + "wasm-metadata", + "wit-bindgen-core", + "wit-component", +] + +[[package]] +name = "wit-bindgen-rust-macro" +version = "0.51.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0c0f9bfd77e6a48eccf51359e3ae77140a7f50b1e2ebfe62422d8afdaffab17a" +dependencies = [ + "anyhow", + "prettyplease", + "proc-macro2", + "quote", + "syn", + "wit-bindgen-core", + "wit-bindgen-rust", +] + +[[package]] +name = "wit-component" +version = "0.244.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d66ea20e9553b30172b5e831994e35fbde2d165325bec84fc43dbf6f4eb9cb2" +dependencies = [ + "anyhow", + "bitflags", + "indexmap", + "log", + "serde", + "serde_derive", + "serde_json", + "wasm-encoder", + "wasm-metadata", + "wasmparser", + "wit-parser", +] + +[[package]] +name = "wit-parser" +version = "0.244.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ecc8ac4bc1dc3381b7f59c34f00b67e18f910c2c0f50015669dde7def656a736" +dependencies = [ + "anyhow", + "id-arena", + "indexmap", + "log", + "semver", + "serde", + "serde_derive", + "serde_json", + "unicode-xid", + "wasmparser", +] + +[[package]] +name = "writeable" +version = "0.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9edde0db4769d2dc68579893f2306b26c6ecfbe0ef499b013d731b7b9247e0b9" + +[[package]] +name = "yoke" +version = "0.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72d6e5c6afb84d73944e5cedb052c4680d5657337201555f9f2a16b7406d4954" +dependencies = [ + "stable_deref_trait", + "yoke-derive", + "zerofrom", +] + +[[package]] +name = "yoke-derive" +version = "0.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b659052874eb698efe5b9e8cf382204678a0086ebf46982b79d6ca3182927e5d" +dependencies = [ + "proc-macro2", + "quote", + "syn", + "synstructure", +] + +[[package]] +name = "zerocopy" +version = "0.8.42" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f2578b716f8a7a858b7f02d5bd870c14bf4ddbbcf3a4c05414ba6503640505e3" +dependencies = [ + "zerocopy-derive", +] + +[[package]] +name = "zerocopy-derive" +version = "0.8.42" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7e6cc098ea4d3bd6246687de65af3f920c430e236bee1e3bf2e441463f08a02f" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "zerofrom" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "50cc42e0333e05660c3587f3bf9d0478688e15d870fab3346451ce7f8c9fbea5" +dependencies = [ + "zerofrom-derive", +] + +[[package]] +name = "zerofrom-derive" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d71e5d6e06ab090c67b5e44993ec16b72dcbaabc526db883a360057678b48502" +dependencies = [ + "proc-macro2", + "quote", + "syn", + "synstructure", +] + +[[package]] +name = "zeroize" +version = "1.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b97154e67e32c85465826e8bcc1c59429aaaf107c1e4a9e53c8d8ccd5eff88d0" + +[[package]] +name = "zerotrie" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2a59c17a5562d507e4b54960e8569ebee33bee890c70aa3fe7b97e85a9fd7851" +dependencies = [ + "displaydoc", + "yoke", + "zerofrom", +] + +[[package]] +name = "zerovec" +version = "0.11.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c28719294829477f525be0186d13efa9a3c602f7ec202ca9e353d310fb9a002" +dependencies = [ + "yoke", + "zerofrom", + "zerovec-derive", +] + +[[package]] +name = "zerovec-derive" +version = "0.11.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eadce39539ca5cb3985590102671f2567e659fca9666581ad3411d59207951f3" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "zmij" +version = "1.0.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8848ee67ecc8aedbaf3e4122217aff892639231befc6a1b58d29fff4c2cabaa" diff --git a/tools/synops-render/Cargo.toml b/tools/synops-render/Cargo.toml new file mode 100644 index 0000000..5c495b8 --- /dev/null +++ b/tools/synops-render/Cargo.toml @@ -0,0 +1,22 @@ +[package] +name = "synops-render" +version = "0.1.0" +edition = "2024" + +[[bin]] +name = "synops-render" +path = "src/main.rs" + +[dependencies] +clap = { version = "4", features = ["derive"] } +tokio = { version = "1", features = ["full"] } +sqlx = { version = "0.8", features = ["runtime-tokio", "tls-rustls", "postgres", "uuid", "chrono", "json"] } +serde = { version = "1", features = ["derive"] } +serde_json = "1" +uuid = { version = "1", features = ["v7", "serde"] } +chrono = { version = "0.4", features = ["serde"] } +tera = "1" +sha2 = "0.10" +hex = "0.4" +tracing = "0.1" +tracing-subscriber = { version = "0.3", features = ["env-filter"] } diff --git a/tools/synops-render/src/main.rs b/tools/synops-render/src/main.rs new file mode 100644 index 0000000..6b978cc --- /dev/null +++ b/tools/synops-render/src/main.rs @@ -0,0 +1,1238 @@ +// synops-render — Tera HTML-rendering til CAS. +// +// Rendrer artikler, forsider og om-sider til HTML via Tera-templates, +// lagrer i CAS (content-addressable store), og oppdaterer node metadata. +// Erstatter rendering-logikken i maskinrommet/src/publishing.rs. +// +// Miljøvariabler: +// DATABASE_URL — PostgreSQL-tilkobling (påkrevd med --write) +// CAS_ROOT — Rot for content-addressable store (default: /srv/synops/media/cas) +// +// Ref: docs/retninger/unix_filosofi.md, docs/concepts/publisering.md + +use chrono::{DateTime, Utc}; +use clap::Parser; +use serde::{Deserialize, Serialize}; +use sha2::{Digest, Sha256}; +use std::path::PathBuf; +use std::process; +use tera::{Context, Tera}; +use uuid::Uuid; + +/// Renderer-versjon. Økes ved mal-/template-endringer. +const RENDERER_VERSION: i64 = 2; + +// ============================================================================= +// CLI +// ============================================================================= + +/// Render artikkel/forside/om-side til HTML og lagre i CAS. +#[derive(Parser)] +#[command(name = "synops-render", about = "Tera HTML-rendering til CAS")] +struct Cli { + /// Node-ID å rendere (artikkel eller samling) + #[arg(long)] + node_id: Uuid, + + /// Tema (avis, magasin, blogg, tidsskrift). Overstyrer samlingens konfig. + #[arg(long)] + theme: Option, + + /// Samlings-ID (påkrevd for artikkel-rendering, brukes for tema/slug-oppslag) + #[arg(long)] + collection_id: Option, + + /// Render-type: article, index, about (default: article) + #[arg(long, default_value = "article")] + render_type: String, + + /// Bruker-ID som utløste renderingen (for ressurslogging) + #[arg(long)] + requested_by: Option, + + /// Skriv resultater til database (uten dette flagget: kun stdout) + #[arg(long)] + write: bool, +} + +// ============================================================================= +// Datamodeller +// ============================================================================= + +#[derive(Deserialize, Default, Debug)] +struct PublishingConfig { + slug: Option, + theme: Option, + #[serde(default)] + theme_config: ThemeConfig, + custom_domain: Option, + featured_max: Option, + stream_page_size: Option, +} + +#[derive(Deserialize, Default, Debug, Clone, Serialize)] +struct ThemeConfig { + #[serde(default)] + colors: ColorConfig, + #[serde(default)] + typography: TypographyConfig, + #[serde(default)] + layout: LayoutConfig, + logo_hash: Option, +} + +#[derive(Deserialize, Default, Debug, Clone, Serialize)] +struct ColorConfig { + primary: Option, + accent: Option, + background: Option, + text: Option, + muted: Option, +} + +#[derive(Deserialize, Default, Debug, Clone, Serialize)] +struct TypographyConfig { + heading_font: Option, + body_font: Option, +} + +#[derive(Deserialize, Default, Debug, Clone, Serialize)] +struct LayoutConfig { + max_width: Option, +} + +#[derive(Serialize, Clone)] +struct ArticleData { + id: String, + short_id: String, + title: String, + subtitle: Option, + content: String, + summary: Option, + og_image: Option, + published_at: String, + published_at_short: String, +} + +#[derive(Serialize)] +struct IndexData { + title: String, + description: Option, + hero: Option, + featured: Vec, + stream: Vec, +} + +#[derive(Serialize, Clone)] +struct SeoData { + og_title: String, + description: String, + canonical_url: String, + og_image: Option, + json_ld: String, +} + +#[derive(Serialize)] +struct RenderResult { + html_hash: String, + size: u64, + renderer_version: i64, + render_type: String, +} + +// ============================================================================= +// Presentasjonselementer +// ============================================================================= + +struct PresEl { + title: Option, + content: Option, + metadata: serde_json::Value, + edge_metadata: serde_json::Value, +} + +impl PresEl { + fn ab_status(&self) -> &str { + self.edge_metadata + .get("ab_status") + .and_then(|v| v.as_str()) + .unwrap_or("") + } +} + +struct PresentationElements { + titles: Vec, + subtitles: Vec, + summaries: Vec, + og_images: Vec, +} + +impl PresentationElements { + fn best_of(elements: &[PresEl]) -> Option<&PresEl> { + if let Some(el) = elements.iter().find(|e| e.ab_status() == "winner") { + return Some(el); + } + elements.iter().find(|e| e.ab_status() != "retired") + } + + fn best_title(&self) -> Option { + Self::best_of(&self.titles) + .and_then(|el| el.title.clone().or(el.content.clone())) + } + + fn best_subtitle(&self) -> Option { + Self::best_of(&self.subtitles) + .and_then(|el| el.title.clone().or(el.content.clone())) + } + + fn best_summary(&self) -> Option { + Self::best_of(&self.summaries) + .and_then(|el| el.content.clone().or(el.title.clone())) + } + + fn best_og_image(&self) -> Option { + Self::best_of(&self.og_images) + .and_then(|el| el.metadata.get("cas_hash").and_then(|h| h.as_str()).map(|s| s.to_string())) + } +} + +// ============================================================================= +// Tema-defaults og CSS-variabler +// ============================================================================= + +struct ThemeDefaults { + primary: &'static str, + accent: &'static str, + background: &'static str, + text: &'static str, + muted: &'static str, + heading_font: &'static str, + body_font: &'static str, + max_width: &'static str, +} + +fn theme_defaults(theme: &str) -> ThemeDefaults { + match theme { + "avis" => ThemeDefaults { + primary: "#1a1a2e", + accent: "#e94560", + background: "#ffffff", + text: "#1a1a2e", + muted: "#6b7280", + heading_font: "'Georgia', 'Times New Roman', serif", + body_font: "'Charter', 'Georgia', serif", + max_width: "1200px", + }, + "magasin" => ThemeDefaults { + primary: "#2d3436", + accent: "#0984e3", + background: "#fafafa", + text: "#2d3436", + muted: "#636e72", + heading_font: "'Playfair Display', 'Georgia', serif", + body_font: "system-ui, -apple-system, sans-serif", + max_width: "1100px", + }, + "blogg" => ThemeDefaults { + primary: "#2c3e50", + accent: "#3498db", + background: "#ffffff", + text: "#333333", + muted: "#7f8c8d", + heading_font: "system-ui, -apple-system, sans-serif", + body_font: "system-ui, -apple-system, sans-serif", + max_width: "720px", + }, + "tidsskrift" => ThemeDefaults { + primary: "#1a1a1a", + accent: "#8b0000", + background: "#fffff8", + text: "#1a1a1a", + muted: "#555555", + heading_font: "'Georgia', 'Times New Roman', serif", + body_font: "'Georgia', 'Times New Roman', serif", + max_width: "680px", + }, + _ => theme_defaults("blogg"), + } +} + +fn build_css_variables(theme: &str, config: &ThemeConfig) -> String { + let defaults = theme_defaults(theme); + format!( + r#":root {{ + --color-primary: {primary}; + --color-accent: {accent}; + --color-background: {background}; + --color-text: {text}; + --color-muted: {muted}; + --font-heading: {heading_font}; + --font-body: {body_font}; + --layout-max-width: {max_width}; +}}"#, + primary = config.colors.primary.as_deref().unwrap_or(defaults.primary), + accent = config.colors.accent.as_deref().unwrap_or(defaults.accent), + background = config.colors.background.as_deref().unwrap_or(defaults.background), + text = config.colors.text.as_deref().unwrap_or(defaults.text), + muted = config.colors.muted.as_deref().unwrap_or(defaults.muted), + heading_font = config.typography.heading_font.as_deref().unwrap_or(defaults.heading_font), + body_font = config.typography.body_font.as_deref().unwrap_or(defaults.body_font), + max_width = config.layout.max_width.as_deref().unwrap_or(defaults.max_width), + ) +} + +// ============================================================================= +// Tera-templates (innebygde) +// ============================================================================= + +fn build_tera() -> Tera { + let mut tera = Tera::default(); + + tera.add_raw_template("base.html", include_str!("templates/base.html")) + .expect("Feil i base.html template"); + + tera.add_raw_template("avis/article.html", include_str!("templates/avis/article.html")) + .expect("Feil i avis/article.html"); + tera.add_raw_template("avis/index.html", include_str!("templates/avis/index.html")) + .expect("Feil i avis/index.html"); + + tera.add_raw_template("magasin/article.html", include_str!("templates/magasin/article.html")) + .expect("Feil i magasin/article.html"); + tera.add_raw_template("magasin/index.html", include_str!("templates/magasin/index.html")) + .expect("Feil i magasin/index.html"); + + tera.add_raw_template("blogg/article.html", include_str!("templates/blogg/article.html")) + .expect("Feil i blogg/article.html"); + tera.add_raw_template("blogg/index.html", include_str!("templates/blogg/index.html")) + .expect("Feil i blogg/index.html"); + + tera.add_raw_template("tidsskrift/article.html", include_str!("templates/tidsskrift/article.html")) + .expect("Feil i tidsskrift/article.html"); + tera.add_raw_template("tidsskrift/index.html", include_str!("templates/tidsskrift/index.html")) + .expect("Feil i tidsskrift/index.html"); + + tera.add_raw_template("category.html", include_str!("templates/category.html")) + .expect("Feil i category.html"); + tera.add_raw_template("archive.html", include_str!("templates/archive.html")) + .expect("Feil i archive.html"); + tera.add_raw_template("search.html", include_str!("templates/search.html")) + .expect("Feil i search.html"); + tera.add_raw_template("about.html", include_str!("templates/about.html")) + .expect("Feil i about.html"); + + tera +} + +// ============================================================================= +// SEO +// ============================================================================= + +fn build_seo_data( + article: &ArticleData, + collection_title: &str, + canonical_url: &str, +) -> SeoData { + let description = article.summary.as_deref().unwrap_or("").to_string(); + let json_ld = build_json_ld(article, collection_title, canonical_url); + let og_image = article.og_image.as_ref().map(|hash| format!("/cas/{hash}")); + + SeoData { + og_title: article.title.clone(), + description, + canonical_url: canonical_url.to_string(), + og_image, + json_ld, + } +} + +fn build_json_ld( + article: &ArticleData, + publisher_name: &str, + canonical_url: &str, +) -> String { + let ld = serde_json::json!({ + "@context": "https://schema.org", + "@type": "Article", + "headline": article.title, + "datePublished": article.published_at, + "url": canonical_url, + "publisher": { + "@type": "Organization", + "name": publisher_name + }, + "description": article.summary.as_deref().unwrap_or("") + }); + ld.to_string() +} + +// ============================================================================= +// TipTap JSON → HTML +// ============================================================================= + +mod tiptap; + +// ============================================================================= +// CAS-lagring +// ============================================================================= + +fn cas_path(root: &str, hash: &str) -> PathBuf { + let (p1, rest) = hash.split_at(2.min(hash.len())); + let (p2, _) = rest.split_at(2.min(rest.len())); + PathBuf::from(root).join(p1).join(p2).join(hash) +} + +async fn store_in_cas(root: &str, data: &[u8]) -> Result<(String, u64, bool), String> { + let mut hasher = Sha256::new(); + hasher.update(data); + let hash = hex::encode(hasher.finalize()); + let size = data.len() as u64; + let path = cas_path(root, &hash); + + if path.exists() { + return Ok((hash, size, true)); + } + + if let Some(parent) = path.parent() { + tokio::fs::create_dir_all(parent) + .await + .map_err(|e| format!("Kunne ikke opprette CAS-katalog: {e}"))?; + } + + // Skriv via temp-fil for atomisk operasjon + let tmp_dir = PathBuf::from(root).join("tmp"); + tokio::fs::create_dir_all(&tmp_dir) + .await + .map_err(|e| format!("Kunne ikke opprette CAS tmp-katalog: {e}"))?; + let tmp_path = tmp_dir.join(format!("{hash}.tmp")); + + tokio::fs::write(&tmp_path, data) + .await + .map_err(|e| format!("Kunne ikke skrive til CAS tmp-fil: {e}"))?; + + tokio::fs::rename(&tmp_path, &path) + .await + .map_err(|e| format!("Atomisk rename feilet: {e}"))?; + + Ok((hash, size, false)) +} + +// ============================================================================= +// Hjelpefunksjoner +// ============================================================================= + +fn truncate(s: &str, max: usize) -> String { + if s.len() <= max { + return s.to_string(); + } + match s[..max].rfind(' ') { + Some(pos) => format!("{}…", &s[..pos]), + None => format!("{}…", &s[..max]), + } +} + +// ============================================================================= +// Render-logikk +// ============================================================================= + +fn render_article_html( + tera: &Tera, + theme: &str, + config: &ThemeConfig, + article: &ArticleData, + collection_title: &str, + base_url: &str, + seo: &SeoData, + has_rss: bool, +) -> Result { + let css_vars = build_css_variables(theme, config); + let template_name = format!("{theme}/article.html"); + + let mut ctx = Context::new(); + ctx.insert("css_variables", &css_vars); + ctx.insert("theme", theme); + ctx.insert("article", article); + ctx.insert("collection_title", collection_title); + ctx.insert("base_url", base_url); + ctx.insert("logo_hash", &config.logo_hash); + ctx.insert("seo", seo); + ctx.insert("has_rss", &has_rss); + + tera.render(&template_name, &ctx) + .map_err(|e| format!("Tera render-feil: {e}")) +} + +fn render_index_html( + tera: &Tera, + theme: &str, + config: &ThemeConfig, + index: &IndexData, + base_url: &str, + has_rss: bool, +) -> Result { + let css_vars = build_css_variables(theme, config); + let template_name = format!("{theme}/index.html"); + + let mut ctx = Context::new(); + ctx.insert("css_variables", &css_vars); + ctx.insert("theme", theme); + ctx.insert("index", index); + ctx.insert("collection_title", &index.title); + ctx.insert("base_url", base_url); + ctx.insert("logo_hash", &config.logo_hash); + ctx.insert("has_rss", &has_rss); + + tera.render(&template_name, &ctx) + .map_err(|e| format!("Tera render-feil (index): {e}")) +} + +// ============================================================================= +// Database-oppslag +// ============================================================================= + +async fn fetch_collection_config( + db: &sqlx::PgPool, + collection_id: Uuid, +) -> Result<(String, PublishingConfig, bool), String> { + let row: Option<(Option, serde_json::Value)> = sqlx::query_as( + r#" + SELECT title, metadata + FROM nodes + WHERE id = $1 AND node_kind = 'collection' + "#, + ) + .bind(collection_id) + .fetch_optional(db) + .await + .map_err(|e| format!("Feil ved henting av samling: {e}"))?; + + let Some((title_opt, metadata)) = row else { + return Err(format!("Samling {collection_id} finnes ikke")); + }; + + let traits = metadata.get("traits"); + + let config: PublishingConfig = traits + .and_then(|t| t.get("publishing")) + .cloned() + .map(|v| serde_json::from_value(v).unwrap_or_default()) + .unwrap_or_default(); + + let has_rss = traits.and_then(|t| t.get("rss")).is_some(); + + let slug = config.slug.as_deref().unwrap_or("unknown"); + let title = title_opt.unwrap_or_else(|| slug.to_string()); + + Ok((title, config, has_rss)) +} + +/// Finn samlings-ID for en artikkel via belongs_to-edge. +async fn find_collection_for_article( + db: &sqlx::PgPool, + node_id: Uuid, +) -> Result, String> { + let row: Option<(Uuid,)> = sqlx::query_as( + r#" + SELECT e.target_id + FROM edges e + JOIN nodes n ON n.id = e.target_id AND n.node_kind = 'collection' + WHERE e.source_id = $1 + AND e.edge_type = 'belongs_to' + AND n.metadata->'traits' ? 'publishing' + LIMIT 1 + "#, + ) + .bind(node_id) + .fetch_optional(db) + .await + .map_err(|e| format!("Feil ved oppslag av samling for artikkel: {e}"))?; + + Ok(row.map(|(id,)| id)) +} + +async fn fetch_presentation_elements( + db: &sqlx::PgPool, + article_id: Uuid, +) -> Result { + let rows: Vec<(String, Option, Option, serde_json::Value, serde_json::Value)> = sqlx::query_as( + r#" + SELECT e.edge_type, n.title, n.content, n.metadata, e.metadata AS edge_metadata + FROM edges e + JOIN nodes n ON n.id = e.source_id + WHERE e.target_id = $1 + AND e.edge_type IN ('title', 'subtitle', 'summary', 'og_image') + ORDER BY e.created_at + "#, + ) + .bind(article_id) + .fetch_all(db) + .await + .map_err(|e| format!("Feil ved henting av presentasjonselementer: {e}"))?; + + let mut titles = vec![]; + let mut subtitles = vec![]; + let mut summaries = vec![]; + let mut og_images = vec![]; + + for (edge_type, title, content, metadata, edge_metadata) in rows { + let el = PresEl { title, content, metadata, edge_metadata }; + match edge_type.as_str() { + "title" => titles.push(el), + "subtitle" => subtitles.push(el), + "summary" => summaries.push(el), + "og_image" => og_images.push(el), + _ => {} + } + } + + Ok(PresentationElements { titles, subtitles, summaries, og_images }) +} + +// ============================================================================= +// Artikkel-rendering +// ============================================================================= + +async fn render_article_to_cas( + db: &sqlx::PgPool, + cas_root: &str, + node_id: Uuid, + collection_id: Uuid, + theme_override: Option<&str>, + write: bool, +) -> Result { + let (collection_title, pub_config, has_rss) = + fetch_collection_config(db, collection_id).await?; + + let theme = theme_override + .or(pub_config.theme.as_deref()) + .unwrap_or("blogg"); + let config = &pub_config.theme_config; + let slug = pub_config.slug.as_deref().unwrap_or("unknown"); + + // Hent artikkeldata + let article_row: Option<(Uuid, Option, Option, serde_json::Value, DateTime)> = sqlx::query_as( + "SELECT id, title, content, metadata, created_at FROM nodes WHERE id = $1", + ) + .bind(node_id) + .fetch_optional(db) + .await + .map_err(|e| format!("Feil ved henting av artikkel: {e}"))?; + + let Some((id, title, content, metadata, created_at)) = article_row else { + return Err(format!("Artikkel {node_id} finnes ikke")); + }; + + // Hent publish_at fra edge-metadata + let edge_meta: Option<(Option,)> = sqlx::query_as( + r#" + SELECT metadata FROM edges + WHERE source_id = $1 AND target_id = $2 AND edge_type = 'belongs_to' + LIMIT 1 + "#, + ) + .bind(node_id) + .bind(collection_id) + .fetch_optional(db) + .await + .map_err(|e| format!("Feil ved henting av edge: {e}"))?; + + let publish_at = edge_meta + .as_ref() + .and_then(|(m,)| m.as_ref()) + .and_then(|m| m.get("publish_at")) + .and_then(|v| v.as_str()) + .and_then(|s| s.parse::>().ok()) + .unwrap_or(created_at); + + // Konverter TipTap JSON → HTML + let article_html = if let Some(doc) = metadata.get("document") { + let html = tiptap::document_to_html(doc); + if html.is_empty() { + content.unwrap_or_default() + } else { + html + } + } else { + content.unwrap_or_default() + }; + + let short_id = id.to_string()[..8].to_string(); + + // Hent presentasjonselementer + let pres = fetch_presentation_elements(db, node_id).await?; + + let article_title = pres.best_title() + .unwrap_or_else(|| title.unwrap_or_else(|| "Uten tittel".to_string())); + + let summary_text = pres.best_summary() + .unwrap_or_else(|| truncate(&article_html.replace("

", "").replace("

", " ").replace('\n', " "), 200)); + + let article_data = ArticleData { + id: id.to_string(), + short_id: short_id.clone(), + title: article_title, + subtitle: pres.best_subtitle(), + content: article_html, + summary: Some(summary_text), + og_image: pres.best_og_image(), + published_at: publish_at.to_rfc3339(), + published_at_short: publish_at.format("%e. %B %Y").to_string(), + }; + + // SEO + render + let base_url = pub_config + .custom_domain + .as_deref() + .map(|d| format!("https://{d}")) + .unwrap_or_else(|| format!("/pub/{slug}")); + + let canonical_url = format!("{base_url}/{short_id}"); + let seo = build_seo_data(&article_data, &collection_title, &canonical_url); + + let tera = build_tera(); + let html = render_article_html(&tera, theme, config, &article_data, &collection_title, &base_url, &seo, has_rss)?; + + // Lagre i CAS + let (hash, size, deduplicated) = store_in_cas(cas_root, html.as_bytes()).await?; + + tracing::info!( + node_id = %node_id, + hash = %hash, + size = size, + deduplicated = deduplicated, + "Artikkel rendret og lagret i CAS" + ); + + // Oppdater metadata i PG + if write { + let now = Utc::now(); + sqlx::query( + r#" + UPDATE nodes + SET metadata = jsonb_set( + jsonb_set( + jsonb_set( + CASE WHEN metadata ? 'rendered' + THEN metadata + ELSE jsonb_set(metadata, '{rendered}', '{}'::jsonb) + END, + '{rendered,html_hash}', + to_jsonb($2::text) + ), + '{rendered,rendered_at}', + to_jsonb($3::text) + ), + '{rendered,renderer_version}', + to_jsonb($4::bigint) + ) + WHERE id = $1 + "#, + ) + .bind(node_id) + .bind(&hash) + .bind(now.to_rfc3339()) + .bind(RENDERER_VERSION) + .execute(db) + .await + .map_err(|e| format!("Feil ved oppdatering av metadata.rendered: {e}"))?; + + tracing::info!(node_id = %node_id, html_hash = %hash, "metadata.rendered oppdatert"); + + // Ressurslogging + log_resource_usage(db, node_id, size, "render_article").await; + } + + Ok(RenderResult { + html_hash: hash, + size, + renderer_version: RENDERER_VERSION, + render_type: "article".to_string(), + }) +} + +// ============================================================================= +// Index-rendering +// ============================================================================= + +async fn render_index_to_cas( + db: &sqlx::PgPool, + cas_root: &str, + collection_id: Uuid, + theme_override: Option<&str>, + write: bool, +) -> Result { + let (collection_title, pub_config, has_rss) = + fetch_collection_config(db, collection_id).await?; + + let theme = theme_override + .or(pub_config.theme.as_deref()) + .unwrap_or("blogg"); + let config = &pub_config.theme_config; + let slug = pub_config.slug.as_deref().unwrap_or("unknown"); + let featured_max = pub_config.featured_max.unwrap_or(4); + let stream_page_size = pub_config.stream_page_size.unwrap_or(20); + + let base_url = pub_config + .custom_domain + .as_deref() + .map(|d| format!("https://{d}")) + .unwrap_or_else(|| format!("/pub/{slug}")); + + // Hent artikler for forsiden + let articles = fetch_index_articles(db, collection_id, featured_max, stream_page_size).await?; + + let (hero, featured, stream) = categorize_articles(articles); + + let index_data = IndexData { + title: collection_title, + description: None, + hero, + featured, + stream, + }; + + let tera = build_tera(); + let html = render_index_html(&tera, theme, config, &index_data, &base_url, has_rss)?; + + let (hash, size, deduplicated) = store_in_cas(cas_root, html.as_bytes()).await?; + + tracing::info!( + collection_id = %collection_id, + hash = %hash, + size = size, + deduplicated = deduplicated, + "Forside rendret og lagret i CAS" + ); + + if write { + let now = Utc::now(); + sqlx::query( + r#" + UPDATE nodes + SET metadata = jsonb_set( + jsonb_set( + jsonb_set( + CASE WHEN metadata ? 'rendered_index' + THEN metadata + ELSE jsonb_set(metadata, '{rendered_index}', '{}'::jsonb) + END, + '{rendered_index,index_hash}', + to_jsonb($2::text) + ), + '{rendered_index,rendered_at}', + to_jsonb($3::text) + ), + '{rendered_index,renderer_version}', + to_jsonb($4::bigint) + ) + WHERE id = $1 + "#, + ) + .bind(collection_id) + .bind(&hash) + .bind(now.to_rfc3339()) + .bind(RENDERER_VERSION) + .execute(db) + .await + .map_err(|e| format!("Feil ved oppdatering av metadata.rendered_index: {e}"))?; + + tracing::info!(collection_id = %collection_id, index_hash = %hash, "metadata.rendered_index oppdatert"); + + log_resource_usage(db, collection_id, size, "render_index").await; + } + + Ok(RenderResult { + html_hash: hash, + size, + renderer_version: RENDERER_VERSION, + render_type: "index".to_string(), + }) +} + +/// Hent publiserte artikler for en samling, sortert etter publish_at desc. +async fn fetch_index_articles( + db: &sqlx::PgPool, + collection_id: Uuid, + featured_max: i64, + stream_page_size: i64, +) -> Result)>, String> { + // Hent artikler med slot-metadata fra edge + let limit = 1 + featured_max + stream_page_size; + let rows: Vec<(Uuid, Option, Option, serde_json::Value, DateTime, Option)> = sqlx::query_as( + r#" + SELECT n.id, n.title, n.content, n.metadata, n.created_at, e.metadata AS edge_metadata + FROM edges e + JOIN nodes n ON n.id = e.source_id + WHERE e.target_id = $1 + AND e.edge_type = 'belongs_to' + AND n.node_kind IN ('article', 'note') + ORDER BY + COALESCE((e.metadata->>'publish_at')::timestamptz, n.created_at) DESC + LIMIT $2 + "#, + ) + .bind(collection_id) + .bind(limit) + .fetch_all(db) + .await + .map_err(|e| format!("Feil ved henting av forsideartikler: {e}"))?; + + let mut articles = Vec::new(); + for (id, title, content, metadata, created_at, edge_metadata) in rows { + let publish_at = edge_metadata + .as_ref() + .and_then(|m| m.get("publish_at")) + .and_then(|v| v.as_str()) + .and_then(|s| s.parse::>().ok()) + .unwrap_or(created_at); + + let slot = edge_metadata + .as_ref() + .and_then(|m| m.get("slot")) + .and_then(|v| v.as_str()) + .map(|s| s.to_string()); + + let article_html = if let Some(doc) = metadata.get("document") { + let html = tiptap::document_to_html(doc); + if html.is_empty() { content.unwrap_or_default() } else { html } + } else { + content.unwrap_or_default() + }; + + let short_id = id.to_string()[..8].to_string(); + let summary_text = truncate( + &article_html.replace("

", "").replace("

", " ").replace('\n', " "), + 200, + ); + + let article_data = ArticleData { + id: id.to_string(), + short_id, + title: title.unwrap_or_else(|| "Uten tittel".to_string()), + subtitle: None, + content: article_html, + summary: Some(summary_text), + og_image: None, + published_at: publish_at.to_rfc3339(), + published_at_short: publish_at.format("%e. %B %Y").to_string(), + }; + + articles.push((article_data, slot)); + } + + Ok(articles) +} + +/// Fordel artikler i hero/featured/stream basert på slot-metadata. +fn categorize_articles( + articles: Vec<(ArticleData, Option)>, +) -> (Option, Vec, Vec) { + let mut hero: Option = None; + let mut featured = Vec::new(); + let mut stream = Vec::new(); + + for (article, slot) in articles { + match slot.as_deref() { + Some("hero") if hero.is_none() => hero = Some(article), + Some("featured") => featured.push(article), + _ => stream.push(article), + } + } + + // Hvis ingen hero er satt, bruk første stream-artikkel + if hero.is_none() && !stream.is_empty() { + hero = Some(stream.remove(0)); + } + + (hero, featured, stream) +} + +// ============================================================================= +// Ressurslogging +// ============================================================================= + +async fn log_resource_usage(db: &sqlx::PgPool, node_id: Uuid, bytes: u64, operation: &str) { + let _ = sqlx::query( + r#" + INSERT INTO resource_usage_log (node_id, resource_type, amount, unit, metadata) + VALUES ($1, $2, $3, 'bytes', $4) + "#, + ) + .bind(node_id) + .bind(operation) + .bind(bytes as i64) + .bind(serde_json::json!({"renderer_version": RENDERER_VERSION})) + .execute(db) + .await; +} + +// ============================================================================= +// main +// ============================================================================= + +#[tokio::main] +async fn main() { + tracing_subscriber::fmt() + .with_env_filter( + tracing_subscriber::EnvFilter::try_from_default_env() + .unwrap_or_else(|_| tracing_subscriber::EnvFilter::new("info")), + ) + .with_target(false) + .with_writer(std::io::stderr) + .init(); + + let cli = Cli::parse(); + + // Valider render_type + if !["article", "index"].contains(&cli.render_type.as_str()) { + eprintln!("Ugyldig render-type: {}. Bruk: article, index", cli.render_type); + process::exit(1); + } + + // Koble til database + let db_url = std::env::var("DATABASE_URL").unwrap_or_else(|_| { + if cli.write { + eprintln!("DATABASE_URL er påkrevd med --write"); + process::exit(1); + } + // Uten --write trenger vi fortsatt DB for oppslag + eprintln!("DATABASE_URL mangler — påkrevd for databaseoppslag"); + process::exit(1); + }); + + let db = sqlx::postgres::PgPoolOptions::new() + .max_connections(2) + .connect(&db_url) + .await + .unwrap_or_else(|e| { + eprintln!("Kunne ikke koble til database: {e}"); + process::exit(1); + }); + + let cas_root = std::env::var("CAS_ROOT").unwrap_or_else(|_| "/srv/synops/media/cas".into()); + + let result = match cli.render_type.as_str() { + "article" => { + // Finn collection_id: brukt direkte, eller oppslag via belongs_to + let collection_id = match cli.collection_id { + Some(id) => id, + None => { + match find_collection_for_article(&db, cli.node_id).await { + Ok(Some(id)) => id, + Ok(None) => { + eprintln!("Ingen publishing-samling funnet for node {}. Bruk --collection-id.", cli.node_id); + process::exit(1); + } + Err(e) => { + eprintln!("{e}"); + process::exit(1); + } + } + } + }; + + render_article_to_cas( + &db, + &cas_root, + cli.node_id, + collection_id, + cli.theme.as_deref(), + cli.write, + ) + .await + } + "index" => { + render_index_to_cas( + &db, + &cas_root, + cli.node_id, + cli.theme.as_deref(), + cli.write, + ) + .await + } + _ => unreachable!(), + }; + + match result { + Ok(res) => { + let json = serde_json::to_string_pretty(&res).unwrap(); + println!("{json}"); + } + Err(e) => { + eprintln!("Render feilet: {e}"); + process::exit(1); + } + } +} + +// ============================================================================= +// Tester +// ============================================================================= + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn test_build_css_variables_defaults() { + let config = ThemeConfig::default(); + let css = build_css_variables("blogg", &config); + assert!(css.contains("--color-primary: #2c3e50")); + assert!(css.contains("--color-accent: #3498db")); + assert!(css.contains("--layout-max-width: 720px")); + } + + #[test] + fn test_build_css_variables_override() { + let config = ThemeConfig { + colors: ColorConfig { + primary: Some("#ff0000".to_string()), + ..Default::default() + }, + ..Default::default() + }; + let css = build_css_variables("blogg", &config); + assert!(css.contains("--color-primary: #ff0000")); + // Andre verdier bruker defaults + assert!(css.contains("--color-accent: #3498db")); + } + + #[test] + fn test_theme_defaults_fallback() { + let defaults = theme_defaults("ukjent_tema"); + // Skal falle tilbake til blogg + assert_eq!(defaults.primary, "#2c3e50"); + } + + #[test] + fn test_truncate() { + assert_eq!(truncate("kort", 10), "kort"); + assert_eq!(truncate("dette er en lang tekst som skal kuttes", 20), "dette er en lang…"); + } + + #[test] + fn test_cas_path() { + let path = cas_path("/srv/synops/media/cas", "b94d27b9934d3e08"); + assert!(path.to_string_lossy().contains("/b9/4d/")); + } + + #[test] + fn test_build_seo_data() { + let article = ArticleData { + id: "test-id".to_string(), + short_id: "test-sho".to_string(), + title: "Testittel".to_string(), + subtitle: None, + content: "

Innhold

".to_string(), + summary: Some("Sammendrag".to_string()), + og_image: Some("abc123".to_string()), + published_at: "2024-01-01T00:00:00Z".to_string(), + published_at_short: " 1. January 2024".to_string(), + }; + let seo = build_seo_data(&article, "Testpub", "/pub/test/test-sho"); + assert_eq!(seo.og_title, "Testittel"); + assert_eq!(seo.description, "Sammendrag"); + assert!(seo.json_ld.contains("\"@type\":\"Article\"")); + assert_eq!(seo.og_image, Some("/cas/abc123".to_string())); + } + + #[test] + fn test_categorize_articles() { + let hero_art = ArticleData { + id: "1".into(), short_id: "1".into(), title: "Hero".into(), + subtitle: None, content: "".into(), summary: None, og_image: None, + published_at: "".into(), published_at_short: "".into(), + }; + let feat_art = ArticleData { + id: "2".into(), short_id: "2".into(), title: "Featured".into(), + subtitle: None, content: "".into(), summary: None, og_image: None, + published_at: "".into(), published_at_short: "".into(), + }; + let stream_art = ArticleData { + id: "3".into(), short_id: "3".into(), title: "Stream".into(), + subtitle: None, content: "".into(), summary: None, og_image: None, + published_at: "".into(), published_at_short: "".into(), + }; + + let articles = vec![ + (hero_art, Some("hero".to_string())), + (feat_art, Some("featured".to_string())), + (stream_art, None), + ]; + + let (hero, featured, stream) = categorize_articles(articles); + assert!(hero.is_some()); + assert_eq!(hero.unwrap().title, "Hero"); + assert_eq!(featured.len(), 1); + assert_eq!(stream.len(), 1); + } + + #[test] + fn test_categorize_articles_auto_hero() { + let art = ArticleData { + id: "1".into(), short_id: "1".into(), title: "Auto hero".into(), + subtitle: None, content: "".into(), summary: None, og_image: None, + published_at: "".into(), published_at_short: "".into(), + }; + let articles = vec![(art, None)]; + let (hero, _, stream) = categorize_articles(articles); + assert!(hero.is_some()); + assert_eq!(hero.unwrap().title, "Auto hero"); + assert!(stream.is_empty()); + } + + #[test] + fn test_render_article_html() { + let tera = build_tera(); + let config = ThemeConfig::default(); + let article = ArticleData { + id: "test".into(), short_id: "test".into(), title: "Test".into(), + subtitle: None, content: "

Hello

".into(), + summary: Some("Sum".into()), og_image: None, + published_at: "2024-01-01T00:00:00Z".into(), + published_at_short: "1. jan 2024".into(), + }; + let seo = build_seo_data(&article, "Pub", "/pub/test/test"); + let html = render_article_html(&tera, "blogg", &config, &article, "Pub", "/pub/test", &seo, false); + assert!(html.is_ok()); + let html = html.unwrap(); + assert!(html.contains("Test")); // tittel + assert!(html.contains("

Hello

")); // innhold + assert!(html.contains("--color-primary")); // CSS-variabler + } + + #[test] + fn test_render_index_html() { + let tera = build_tera(); + let config = ThemeConfig::default(); + let index = IndexData { + title: "Forside".to_string(), + description: None, + hero: Some(ArticleData { + id: "1".into(), short_id: "1".into(), title: "Hero".into(), + subtitle: None, content: "

Hero content

".into(), + summary: Some("Hero sum".into()), og_image: None, + published_at: "2024-01-01T00:00:00Z".into(), + published_at_short: "1. jan".into(), + }), + featured: vec![], + stream: vec![], + }; + let html = render_index_html(&tera, "blogg", &config, &index, "/pub/test", false); + assert!(html.is_ok()); + let html = html.unwrap(); + assert!(html.contains("Hero")); + assert!(html.contains("Forside")); + } + + #[test] + fn test_presentation_elements_empty() { + let pres = PresentationElements { + titles: vec![], subtitles: vec![], summaries: vec![], og_images: vec![], + }; + assert!(pres.best_title().is_none()); + assert!(pres.best_subtitle().is_none()); + assert!(pres.best_summary().is_none()); + assert!(pres.best_og_image().is_none()); + } +} diff --git a/tools/synops-render/src/templates/about.html b/tools/synops-render/src/templates/about.html new file mode 100644 index 0000000..84cd961 --- /dev/null +++ b/tools/synops-render/src/templates/about.html @@ -0,0 +1,54 @@ +{% extends "base.html" %} + +{% block title %}Om — {{ collection_title }}{% endblock %} + +{% block extra_css %} +.about-page { + max-width: var(--layout-max-width); + margin: 2rem auto; + padding: 0 1rem; +} +.about-page__content { + font-size: 1.05rem; + line-height: 1.8; +} +.about-page__content h1, +.about-page__content h2, +.about-page__content h3 { + font-family: var(--font-heading); + color: var(--color-primary); + margin-top: 2rem; + margin-bottom: 0.75rem; + line-height: 1.3; +} +.about-page__content h1 { font-size: 2rem; } +.about-page__content h2 { font-size: 1.5rem; } +.about-page__content h3 { font-size: 1.25rem; } +.about-page__content p { margin-bottom: 1rem; } +.about-page__content blockquote { + border-left: 3px solid var(--color-accent); + padding-left: 1rem; + margin: 1.5rem 0; + color: var(--color-muted); + font-style: italic; +} +.about-page__content ul, .about-page__content ol { + margin: 1rem 0; + padding-left: 1.5rem; +} +.about-page__content li { margin-bottom: 0.5rem; } +.about-page__content img { margin: 1.5rem 0; border-radius: 4px; } +@media (max-width: 768px) { + .about-page__content h1 { font-size: 1.5rem; } + .about-page__content h2 { font-size: 1.25rem; } + .about-page__content { font-size: 1rem; } +} +{% endblock %} + +{% block content %} +
+
+ {{ about_html | safe }} +
+
+{% endblock %} diff --git a/tools/synops-render/src/templates/archive.html b/tools/synops-render/src/templates/archive.html new file mode 100644 index 0000000..129e7af --- /dev/null +++ b/tools/synops-render/src/templates/archive.html @@ -0,0 +1,126 @@ +{% extends "base.html" %} + +{% block title %}Arkiv — {{ collection_title }}{% endblock %} + +{% block extra_css %} +.dynamic-page { + max-width: var(--layout-max-width); + margin: 2rem auto; + padding: 0 1rem; +} +.dynamic-page__header { + margin-bottom: 2rem; + padding-bottom: 1rem; + border-bottom: 2px solid var(--color-accent); +} +.dynamic-page__title { + font-family: var(--font-heading); + font-size: 2rem; + color: var(--color-primary); +} +.dynamic-page__subtitle { + color: var(--color-muted); + margin-top: 0.25rem; +} +.month-group { + margin-bottom: 2rem; +} +.month-group__heading { + font-family: var(--font-heading); + font-size: 1.25rem; + color: var(--color-primary); + padding-bottom: 0.5rem; + border-bottom: 1px solid #e0e0e0; + margin-bottom: 0.75rem; +} +.article-list { list-style: none; } +.article-list__item { + padding: 1rem 0; + border-bottom: 1px solid #f5f5f5; +} +.article-list__item:last-child { border-bottom: none; } +.article-list__title { + font-family: var(--font-heading); + font-size: 1.2rem; + color: var(--color-primary); + line-height: 1.3; + margin-bottom: 0.15rem; +} +.article-list__meta { + font-size: 0.8rem; + color: var(--color-muted); +} +.pagination { + display: flex; + gap: 0.5rem; + justify-content: center; + margin-top: 2rem; + padding-top: 1rem; + border-top: 1px solid #f0f0f0; +} +.pagination a, .pagination span { + padding: 0.5rem 1rem; + border: 1px solid #e0e0e0; + border-radius: 4px; + font-size: 0.9rem; +} +.pagination .current { + background: var(--color-accent); + color: #fff; + border-color: var(--color-accent); +} +.empty-state { + text-align: center; + padding: 3rem 1rem; + color: var(--color-muted); +} +@media (max-width: 768px) { + .dynamic-page__title { font-size: 1.5rem; } + .article-list__title { font-size: 1.05rem; } +} +{% endblock %} + +{% block content %} +
+
+

Arkiv

+

{{ total_articles }} artikler totalt

+
+ + {% if month_groups | length > 0 %} + {% for group in month_groups %} +
+

{{ group.label }}

+
    + {% for item in group.articles %} +
  • +

    {{ item.title }}

    +
    {{ item.published_at_short }}
    +
  • + {% endfor %} +
+
+ {% endfor %} + + {% if total_pages > 1 %} + + {% endif %} + {% else %} +
Ingen publiserte artikler ennå.
+ {% endif %} +
+{% endblock %} diff --git a/tools/synops-render/src/templates/avis/article.html b/tools/synops-render/src/templates/avis/article.html new file mode 100644 index 0000000..b4f0701 --- /dev/null +++ b/tools/synops-render/src/templates/avis/article.html @@ -0,0 +1,82 @@ +{% extends "base.html" %} + +{% block title %}{{ article.title }} — {{ collection_title }}{% endblock %} + +{% block seo %} + + + + + + + + {% if seo.og_image %}{% endif %} + + + +{% endblock %} + +{% block extra_css %} +.article { + max-width: var(--layout-max-width); + margin: 2rem auto; + padding: 0 1rem; + display: grid; + grid-template-columns: 1fr 300px; + gap: 2rem; +} +.article__main { min-width: 0; } +.article__sidebar { + border-left: 2px solid var(--color-accent); + padding-left: 1.5rem; +} +.article__title { + font-family: var(--font-heading); + font-size: 2.5rem; + line-height: 1.15; + margin-bottom: 0.5rem; + color: var(--color-primary); +} +.article__meta { + color: var(--color-muted); + font-size: 0.875rem; + margin-bottom: 1.5rem; + padding-bottom: 1rem; + border-bottom: 1px solid var(--color-muted); +} +.article__content { + font-size: 1.05rem; + line-height: 1.75; +} +.article__content p { margin-bottom: 1em; } +.article__back { + display: inline-block; + margin-top: 2rem; + font-size: 0.9rem; +} + +@media (max-width: 768px) { + .article { grid-template-columns: 1fr; } + .article__sidebar { + border-left: none; + border-top: 2px solid var(--color-accent); + padding-left: 0; + padding-top: 1.5rem; + } +} +{% endblock %} + +{% block content %} +
+
+ {% if article.og_image %}{{ article.title }}{% endif %} +

{{ article.title }}

+ {% if article.subtitle %}

{{ article.subtitle }}

{% endif %} + +
{{ article.content | safe }}
+ ← Tilbake til forsiden +
+ +
+{% endblock %} diff --git a/tools/synops-render/src/templates/avis/index.html b/tools/synops-render/src/templates/avis/index.html new file mode 100644 index 0000000..fbe32e8 --- /dev/null +++ b/tools/synops-render/src/templates/avis/index.html @@ -0,0 +1,159 @@ +{% extends "base.html" %} + +{% block title %}{{ index.title }}{% endblock %} + +{% block extra_css %} +.avis-layout { + max-width: var(--layout-max-width); + margin: 1.5rem auto; + padding: 0 1rem; +} + +/* Hero */ +.hero { + border-bottom: 3px solid var(--color-primary); + padding-bottom: 1.5rem; + margin-bottom: 1.5rem; +} +.hero__title { + font-family: var(--font-heading); + font-size: 2.5rem; + line-height: 1.1; + color: var(--color-primary); + margin-bottom: 0.5rem; +} +.hero__summary { + font-size: 1.1rem; + color: var(--color-muted); + max-width: 60ch; +} +.hero__meta { + font-size: 0.8rem; + color: var(--color-muted); + margin-top: 0.5rem; +} + +/* Featured + sidebar grid */ +.avis-grid { + display: grid; + grid-template-columns: 2fr 1fr; + gap: 1.5rem; + margin-bottom: 2rem; +} +.featured-list { display: flex; flex-direction: column; gap: 1rem; } +.featured-item { + border-bottom: 1px solid #e5e7eb; + padding-bottom: 1rem; +} +.featured-item__title { + font-family: var(--font-heading); + font-size: 1.25rem; + color: var(--color-primary); + margin-bottom: 0.25rem; +} +.featured-item__summary { + font-size: 0.9rem; + color: var(--color-muted); +} + +/* Sidebar */ +.sidebar { + border-left: 2px solid var(--color-accent); + padding-left: 1rem; +} +.sidebar__heading { + font-family: var(--font-heading); + font-size: 0.85rem; + text-transform: uppercase; + letter-spacing: 0.05em; + color: var(--color-muted); + margin-bottom: 1rem; +} + +/* Stream */ +.stream { border-top: 2px solid var(--color-primary); padding-top: 1rem; } +.stream__heading { + font-family: var(--font-heading); + font-size: 0.85rem; + text-transform: uppercase; + letter-spacing: 0.05em; + color: var(--color-muted); + margin-bottom: 1rem; +} +.stream-grid { + display: grid; + grid-template-columns: repeat(auto-fill, minmax(250px, 1fr)); + gap: 1rem; +} +.stream-item__title { + font-family: var(--font-heading); + font-size: 1rem; + color: var(--color-primary); +} +.stream-item__meta { + font-size: 0.8rem; + color: var(--color-muted); +} + +@media (max-width: 768px) { + .avis-grid { grid-template-columns: 1fr; } + .sidebar { border-left: none; border-top: 2px solid var(--color-accent); padding-left: 0; padding-top: 1rem; } + .hero__title { font-size: 1.75rem; } +} +{% endblock %} + +{% block content %} +
+ {% if index.hero %} +
+

{{ index.hero.title }}

+ {% if index.hero.summary %} +

{{ index.hero.summary }}

+ {% endif %} +
{{ index.hero.published_at_short }}
+
+ {% endif %} + + {% if index.featured | length > 0 or index.stream | length > 0 %} +
+ + +
+ {% endif %} + + {% if index.stream | length > 5 %} +
+
Flere saker
+
+ {% for item in index.stream %} + {% if loop.index > 5 %} +
+

{{ item.title }}

+
{{ item.published_at_short }}
+
+ {% endif %} + {% endfor %} +
+
+ {% endif %} +
+{% endblock %} diff --git a/tools/synops-render/src/templates/base.html b/tools/synops-render/src/templates/base.html new file mode 100644 index 0000000..7032900 --- /dev/null +++ b/tools/synops-render/src/templates/base.html @@ -0,0 +1,94 @@ + + + + + + {% block title %}{{ collection_title | default(value="Synops") }}{% endblock %} + {% if has_rss %}{% endif %} + {% block seo %}{% endblock %} + + + + + +
+ {% block content %}{% endblock %} +
+ +
+
+ Drevet av Synops +
+
+ + diff --git a/tools/synops-render/src/templates/blogg/article.html b/tools/synops-render/src/templates/blogg/article.html new file mode 100644 index 0000000..53f4910 --- /dev/null +++ b/tools/synops-render/src/templates/blogg/article.html @@ -0,0 +1,60 @@ +{% extends "base.html" %} + +{% block title %}{{ article.title }} — {{ collection_title }}{% endblock %} + +{% block seo %} + + + + + + + + {% if seo.og_image %}{% endif %} + + + +{% endblock %} + +{% block extra_css %} +.blog-article { + max-width: var(--layout-max-width); + margin: 2rem auto; + padding: 0 1rem; +} +.blog-article__title { + font-family: var(--font-heading); + font-size: 2rem; + line-height: 1.2; + color: var(--color-primary); + margin-bottom: 0.5rem; +} +.blog-article__meta { + color: var(--color-muted); + font-size: 0.875rem; + margin-bottom: 2rem; +} +.blog-article__content { + font-size: 1.05rem; + line-height: 1.75; +} +.blog-article__content p { margin-bottom: 1em; } +.blog-article__back { + display: inline-block; + margin-top: 2rem; + font-size: 0.9rem; +} +{% endblock %} + +{% block content %} +
+ {% if article.og_image %}{{ article.title }}{% endif %} +

{{ article.title }}

+ {% if article.subtitle %}

{{ article.subtitle }}

{% endif %} + +
+ {{ article.content | safe }} +
+ ← Tilbake +
+{% endblock %} diff --git a/tools/synops-render/src/templates/blogg/index.html b/tools/synops-render/src/templates/blogg/index.html new file mode 100644 index 0000000..05ce5d8 --- /dev/null +++ b/tools/synops-render/src/templates/blogg/index.html @@ -0,0 +1,93 @@ +{% extends "base.html" %} + +{% block title %}{{ index.title }}{% endblock %} + +{% block extra_css %} +.blog-layout { + max-width: var(--layout-max-width); + margin: 2rem auto; + padding: 0 1rem; +} +.blog-list { list-style: none; } +.blog-item { + padding: 1.5rem 0; + border-bottom: 1px solid #f3f4f6; +} +.blog-item:first-child { padding-top: 0; } +.blog-item__title { + font-family: var(--font-heading); + font-size: 1.5rem; + color: var(--color-primary); + margin-bottom: 0.25rem; + line-height: 1.3; +} +.blog-item__meta { + font-size: 0.8rem; + color: var(--color-muted); + margin-bottom: 0.5rem; +} +.blog-item__summary { + font-size: 0.95rem; + color: var(--color-text); + line-height: 1.5; +} + +/* Pinned hero-artikkel */ +.blog-pinned { + background: #f8f9fa; + border-radius: 8px; + padding: 1.5rem; + margin-bottom: 2rem; + border-left: 4px solid var(--color-accent); +} +.blog-pinned__label { + font-size: 0.75rem; + text-transform: uppercase; + letter-spacing: 0.05em; + color: var(--color-accent); + margin-bottom: 0.5rem; +} + +@media (max-width: 768px) { + .blog-item__title { font-size: 1.25rem; } +} +{% endblock %} + +{% block content %} +
+ {% if index.hero %} +
+
Fremhevet
+

{{ index.hero.title }}

+ {% if index.hero.summary %} +

{{ index.hero.summary }}

+ {% endif %} +
{{ index.hero.published_at_short }}
+
+ {% endif %} + + {% if index.featured | length > 0 %} + {% for item in index.featured %} +
+

{{ item.title }}

+ {% if item.summary %} +

{{ item.summary }}

+ {% endif %} +
{{ item.published_at_short }}
+
+ {% endfor %} + {% endif %} + +
    + {% for item in index.stream %} +
  • +

    {{ item.title }}

    +
    {{ item.published_at_short }}
    + {% if item.summary %} +

    {{ item.summary }}

    + {% endif %} +
  • + {% endfor %} +
+
+{% endblock %} diff --git a/tools/synops-render/src/templates/category.html b/tools/synops-render/src/templates/category.html new file mode 100644 index 0000000..6941945 --- /dev/null +++ b/tools/synops-render/src/templates/category.html @@ -0,0 +1,133 @@ +{% extends "base.html" %} + +{% block title %}{{ tag_name }} — {{ collection_title }}{% endblock %} + +{% block extra_css %} +.dynamic-page { + max-width: var(--layout-max-width); + margin: 2rem auto; + padding: 0 1rem; +} +.dynamic-page__header { + margin-bottom: 2rem; + padding-bottom: 1rem; + border-bottom: 2px solid var(--color-accent); +} +.dynamic-page__title { + font-family: var(--font-heading); + font-size: 2rem; + color: var(--color-primary); +} +.dynamic-page__subtitle { + color: var(--color-muted); + margin-top: 0.25rem; +} +.article-list { list-style: none; } +.article-list__item { + padding: 1.25rem 0; + border-bottom: 1px solid #f0f0f0; +} +.article-list__item:first-child { padding-top: 0; } +.article-list__title { + font-family: var(--font-heading); + font-size: 1.35rem; + color: var(--color-primary); + line-height: 1.3; + margin-bottom: 0.25rem; +} +.article-list__meta { + font-size: 0.8rem; + color: var(--color-muted); + margin-bottom: 0.4rem; +} +.article-list__summary { + font-size: 0.95rem; + line-height: 1.5; +} +.pagination { + display: flex; + gap: 0.5rem; + justify-content: center; + margin-top: 2rem; + padding-top: 1rem; + border-top: 1px solid #f0f0f0; +} +.pagination a, .pagination span { + padding: 0.5rem 1rem; + border: 1px solid #e0e0e0; + border-radius: 4px; + font-size: 0.9rem; +} +.pagination .current { + background: var(--color-accent); + color: #fff; + border-color: var(--color-accent); +} +.tag-list { + display: flex; + flex-wrap: wrap; + gap: 0.5rem; + margin-top: 1rem; +} +.tag-link { + display: inline-block; + padding: 0.25rem 0.75rem; + background: #f3f4f6; + border-radius: 1rem; + font-size: 0.85rem; + color: var(--color-text); +} +.tag-link:hover { background: var(--color-accent); color: #fff; text-decoration: none; } +.empty-state { + text-align: center; + padding: 3rem 1rem; + color: var(--color-muted); +} +@media (max-width: 768px) { + .dynamic-page__title { font-size: 1.5rem; } + .article-list__title { font-size: 1.15rem; } +} +{% endblock %} + +{% block content %} +
+
+

{{ tag_name }}

+

{{ article_count }} {% if article_count == 1 %}artikkel{% else %}artikler{% endif %}

+
+ + {% if articles | length > 0 %} +
    + {% for item in articles %} +
  • +

    {{ item.title }}

    +
    {{ item.published_at_short }}
    + {% if item.summary %} +

    {{ item.summary }}

    + {% endif %} +
  • + {% endfor %} +
+ + {% if total_pages > 1 %} + + {% endif %} + {% else %} +
Ingen artikler i denne kategorien.
+ {% endif %} +
+{% endblock %} diff --git a/tools/synops-render/src/templates/magasin/article.html b/tools/synops-render/src/templates/magasin/article.html new file mode 100644 index 0000000..faa1ea5 --- /dev/null +++ b/tools/synops-render/src/templates/magasin/article.html @@ -0,0 +1,75 @@ +{% extends "base.html" %} + +{% block title %}{{ article.title }} — {{ collection_title }}{% endblock %} + +{% block seo %} + + + + + + + + {% if seo.og_image %}{% endif %} + + + +{% endblock %} + +{% block extra_css %} +.mag-article { + max-width: var(--layout-max-width); + margin: 0 auto; + padding: 0 1rem; +} +.mag-article__header { + text-align: center; + padding: 3rem 0 2rem; + max-width: 720px; + margin: 0 auto; +} +.mag-article__title { + font-family: var(--font-heading); + font-size: 3rem; + line-height: 1.1; + color: var(--color-primary); + margin-bottom: 0.75rem; +} +.mag-article__meta { + color: var(--color-muted); + font-size: 0.9rem; +} +.mag-article__content { + max-width: 680px; + margin: 0 auto; + font-size: 1.1rem; + line-height: 1.8; + padding-bottom: 3rem; +} +.mag-article__content p { margin-bottom: 1.25em; } +.mag-article__back { + display: inline-block; + margin-top: 2rem; + font-size: 0.9rem; +} + +@media (max-width: 768px) { + .mag-article__title { font-size: 2rem; } + .mag-article__header { padding: 2rem 0 1.5rem; } +} +{% endblock %} + +{% block content %} +
+ {% if article.og_image %}{{ article.title }}{% endif %} +
+

{{ article.title }}

+ {% if article.subtitle %}

{{ article.subtitle }}

{% endif %} + +
+
+ {{ article.content | safe }} + ← Tilbake +
+
+{% endblock %} diff --git a/tools/synops-render/src/templates/magasin/index.html b/tools/synops-render/src/templates/magasin/index.html new file mode 100644 index 0000000..ade075d --- /dev/null +++ b/tools/synops-render/src/templates/magasin/index.html @@ -0,0 +1,132 @@ +{% extends "base.html" %} + +{% block title %}{{ index.title }}{% endblock %} + +{% block extra_css %} +.mag-layout { + max-width: var(--layout-max-width); + margin: 0 auto; + padding: 0 1rem; +} + +/* Hero — fullbredde */ +.mag-hero { + padding: 3rem 0; + text-align: center; + border-bottom: 1px solid #e5e7eb; + margin-bottom: 2rem; +} +.mag-hero__title { + font-family: var(--font-heading); + font-size: 3rem; + line-height: 1.1; + color: var(--color-primary); + margin-bottom: 0.75rem; +} +.mag-hero__summary { + font-size: 1.15rem; + color: var(--color-muted); + max-width: 50ch; + margin: 0 auto; +} +.mag-hero__meta { + font-size: 0.85rem; + color: var(--color-muted); + margin-top: 0.75rem; +} + +/* Featured cards */ +.mag-cards { + display: grid; + grid-template-columns: repeat(auto-fill, minmax(280px, 1fr)); + gap: 2rem; + margin-bottom: 3rem; +} +.mag-card { + border: 1px solid #e5e7eb; + border-radius: 8px; + padding: 1.5rem; + transition: box-shadow 0.2s; +} +.mag-card:hover { box-shadow: 0 4px 12px rgba(0,0,0,0.08); } +.mag-card__title { + font-family: var(--font-heading); + font-size: 1.35rem; + color: var(--color-primary); + margin-bottom: 0.5rem; + line-height: 1.25; +} +.mag-card__summary { + font-size: 0.95rem; + color: var(--color-muted); + line-height: 1.5; +} +.mag-card__meta { + font-size: 0.8rem; + color: var(--color-muted); + margin-top: 0.75rem; +} + +/* Kronologisk strøm */ +.mag-stream { border-top: 2px solid var(--color-primary); padding-top: 1.5rem; } +.mag-stream__heading { + font-family: var(--font-heading); + font-size: 1.25rem; + color: var(--color-primary); + margin-bottom: 1rem; +} +.mag-stream-item { + display: flex; + justify-content: space-between; + align-items: baseline; + padding: 0.75rem 0; + border-bottom: 1px solid #f3f4f6; +} +.mag-stream-item__title { font-size: 1rem; } +.mag-stream-item__date { font-size: 0.8rem; color: var(--color-muted); white-space: nowrap; margin-left: 1rem; } + +@media (max-width: 768px) { + .mag-hero__title { font-size: 2rem; } + .mag-hero { padding: 2rem 0; } +} +{% endblock %} + +{% block content %} +
+ {% if index.hero %} +
+

{{ index.hero.title }}

+ {% if index.hero.summary %} +

{{ index.hero.summary }}

+ {% endif %} +
{{ index.hero.published_at_short }}
+
+ {% endif %} + + {% if index.featured | length > 0 %} +
+ {% for item in index.featured %} +
+

{{ item.title }}

+ {% if item.summary %} +

{{ item.summary }}

+ {% endif %} +
{{ item.published_at_short }}
+
+ {% endfor %} +
+ {% endif %} + + {% if index.stream | length > 0 %} +
+

Alle artikler

+ {% for item in index.stream %} +
+ {{ item.title }} + {{ item.published_at_short }} +
+ {% endfor %} +
+ {% endif %} +
+{% endblock %} diff --git a/tools/synops-render/src/templates/search.html b/tools/synops-render/src/templates/search.html new file mode 100644 index 0000000..147a346 --- /dev/null +++ b/tools/synops-render/src/templates/search.html @@ -0,0 +1,162 @@ +{% extends "base.html" %} + +{% block title %}{% if query %}Søk: {{ query }} — {% endif %}{{ collection_title }}{% endblock %} + +{% block extra_css %} +.dynamic-page { + max-width: var(--layout-max-width); + margin: 2rem auto; + padding: 0 1rem; +} +.dynamic-page__header { + margin-bottom: 2rem; + padding-bottom: 1rem; + border-bottom: 2px solid var(--color-accent); +} +.dynamic-page__title { + font-family: var(--font-heading); + font-size: 2rem; + color: var(--color-primary); +} +.search-form { + margin-top: 1rem; + display: flex; + gap: 0.5rem; +} +.search-form__input { + flex: 1; + padding: 0.75rem 1rem; + border: 2px solid #e0e0e0; + border-radius: 6px; + font-size: 1rem; + font-family: var(--font-body); + color: var(--color-text); + background: var(--color-background); +} +.search-form__input:focus { + border-color: var(--color-accent); + outline: none; +} +.search-form__button { + padding: 0.75rem 1.5rem; + background: var(--color-accent); + color: #fff; + border: none; + border-radius: 6px; + font-size: 1rem; + cursor: pointer; +} +.search-form__button:hover { opacity: 0.9; } +.search-results__info { + color: var(--color-muted); + margin-bottom: 1rem; + font-size: 0.9rem; +} +.article-list { list-style: none; } +.article-list__item { + padding: 1.25rem 0; + border-bottom: 1px solid #f0f0f0; +} +.article-list__item:first-child { padding-top: 0; } +.article-list__title { + font-family: var(--font-heading); + font-size: 1.35rem; + color: var(--color-primary); + line-height: 1.3; + margin-bottom: 0.25rem; +} +.article-list__meta { + font-size: 0.8rem; + color: var(--color-muted); + margin-bottom: 0.4rem; +} +.article-list__summary { + font-size: 0.95rem; + line-height: 1.5; +} +.article-list__highlight { + background: rgba(233, 69, 96, 0.1); + padding: 0 0.15rem; + border-radius: 2px; +} +.pagination { + display: flex; + gap: 0.5rem; + justify-content: center; + margin-top: 2rem; + padding-top: 1rem; + border-top: 1px solid #f0f0f0; +} +.pagination a, .pagination span { + padding: 0.5rem 1rem; + border: 1px solid #e0e0e0; + border-radius: 4px; + font-size: 0.9rem; +} +.pagination .current { + background: var(--color-accent); + color: #fff; + border-color: var(--color-accent); +} +.empty-state { + text-align: center; + padding: 3rem 1rem; + color: var(--color-muted); +} +@media (max-width: 768px) { + .dynamic-page__title { font-size: 1.5rem; } + .article-list__title { font-size: 1.15rem; } + .search-form { flex-direction: column; } +} +{% endblock %} + +{% block content %} +
+
+

Søk

+
+ + +
+
+ + {% if query %} + {% if articles | length > 0 %} +

{{ result_count }} treff for «{{ query }}»

+
    + {% for item in articles %} +
  • +

    {{ item.title }}

    +
    {{ item.published_at_short }}
    + {% if item.summary %} +

    {{ item.summary }}

    + {% endif %} +
  • + {% endfor %} +
+ + {% if total_pages > 1 %} + + {% endif %} + {% else %} +
Ingen treff for «{{ query }}».
+ {% endif %} + {% else %} +
Skriv inn et søkeord for å søke i artiklene.
+ {% endif %} +
+{% endblock %} diff --git a/tools/synops-render/src/templates/tidsskrift/article.html b/tools/synops-render/src/templates/tidsskrift/article.html new file mode 100644 index 0000000..6744df0 --- /dev/null +++ b/tools/synops-render/src/templates/tidsskrift/article.html @@ -0,0 +1,67 @@ +{% extends "base.html" %} + +{% block title %}{{ article.title }} — {{ collection_title }}{% endblock %} + +{% block seo %} + + + + + + + + {% if seo.og_image %}{% endif %} + + + +{% endblock %} + +{% block extra_css %} +.journal-article { + max-width: var(--layout-max-width); + margin: 3rem auto; + padding: 0 1rem; +} +.journal-article__title { + font-family: var(--font-heading); + font-size: 1.75rem; + line-height: 1.3; + color: var(--color-primary); + margin-bottom: 0.25rem; + text-align: center; +} +.journal-article__meta { + color: var(--color-muted); + font-size: 0.85rem; + text-align: center; + margin-bottom: 2rem; + padding-bottom: 1.5rem; + border-bottom: 1px solid var(--color-muted); +} +.journal-article__content { + font-size: 1rem; + line-height: 1.8; + text-align: justify; + hyphens: auto; +} +.journal-article__content p { margin-bottom: 1em; text-indent: 1.5em; } +.journal-article__content p:first-child { text-indent: 0; } +.journal-article__back { + display: inline-block; + margin-top: 2rem; + font-size: 0.85rem; +} +{% endblock %} + +{% block content %} +
+ {% if article.og_image %}
{{ article.title }}
{% endif %} +

{{ article.title }}

+ {% if article.subtitle %}

{{ article.subtitle }}

{% endif %} + +
+ {{ article.content | safe }} +
+ ← Tilbake til innholdsfortegnelse +
+{% endblock %} diff --git a/tools/synops-render/src/templates/tidsskrift/index.html b/tools/synops-render/src/templates/tidsskrift/index.html new file mode 100644 index 0000000..747e9a3 --- /dev/null +++ b/tools/synops-render/src/templates/tidsskrift/index.html @@ -0,0 +1,133 @@ +{% extends "base.html" %} + +{% block title %}{{ index.title }}{% endblock %} + +{% block extra_css %} +.journal-layout { + max-width: var(--layout-max-width); + margin: 3rem auto; + padding: 0 1rem; +} +.journal-header { + text-align: center; + margin-bottom: 2rem; + padding-bottom: 1.5rem; + border-bottom: 2px solid var(--color-primary); +} +.journal-header__title { + font-family: var(--font-heading); + font-size: 2rem; + color: var(--color-primary); +} +.journal-header__desc { + font-size: 0.95rem; + color: var(--color-muted); + font-style: italic; + margin-top: 0.5rem; +} + +/* Nummerert innholdsfortegnelse */ +.journal-toc { + counter-reset: article-counter; + list-style: none; +} +.journal-toc__item { + counter-increment: article-counter; + padding: 1rem 0; + border-bottom: 1px solid #eee; + display: flex; + align-items: baseline; + gap: 1rem; +} +.journal-toc__item::before { + content: counter(article-counter) "."; + font-family: var(--font-heading); + font-size: 1.1rem; + color: var(--color-muted); + min-width: 2rem; + text-align: right; +} +.journal-toc__title { + font-family: var(--font-heading); + font-size: 1.1rem; + color: var(--color-primary); + line-height: 1.3; +} +.journal-toc__meta { + font-size: 0.8rem; + color: var(--color-muted); + margin-top: 0.25rem; +} + +/* Fremhevet (hero/featured) */ +.journal-featured { + margin-bottom: 2rem; + padding-bottom: 1.5rem; + border-bottom: 1px solid var(--color-primary); +} +.journal-featured__label { + font-size: 0.75rem; + text-transform: uppercase; + letter-spacing: 0.1em; + color: var(--color-accent); + margin-bottom: 0.5rem; +} +.journal-featured__title { + font-family: var(--font-heading); + font-size: 1.5rem; + color: var(--color-primary); + line-height: 1.3; +} +.journal-featured__summary { + font-size: 0.95rem; + color: var(--color-text); + margin-top: 0.5rem; + line-height: 1.5; +} +{% endblock %} + +{% block content %} +
+
+

{{ index.title }}

+ {% if index.description %} +

{{ index.description }}

+ {% endif %} +
+ + {% if index.hero %} + + {% endif %} + + {% if index.featured | length > 0 %} + {% for item in index.featured %} + + {% endfor %} + {% endif %} + + {% if index.stream | length > 0 %} +

Innholdsfortegnelse

+
    + {% for item in index.stream %} +
  1. +
    + +
    {{ item.published_at_short }}
    +
    +
  2. + {% endfor %} +
+ {% endif %} +
+{% endblock %} diff --git a/tools/synops-render/src/tiptap.rs b/tools/synops-render/src/tiptap.rs new file mode 100644 index 0000000..bfc48ab --- /dev/null +++ b/tools/synops-render/src/tiptap.rs @@ -0,0 +1,291 @@ +//! TipTap/ProseMirror JSON → HTML-konvertering. +//! +//! Konverterer `metadata.document` (TipTap JSON) til HTML-streng. +//! Identisk med maskinrommet/src/tiptap.rs — delt logikk. +//! Ref: oppgave 21.16 (synops-common) vil samle dette i felles crate. + +use serde_json::Value; + +/// Konverter et TipTap/ProseMirror-dokument (JSON) til HTML. +/// Returnerer tom streng hvis dokumentet er ugyldig. +pub fn document_to_html(doc: &Value) -> String { + let Some(content) = doc.get("content").and_then(|c| c.as_array()) else { + return String::new(); + }; + + let mut html = String::new(); + for node in content { + render_node(node, &mut html); + } + html +} + +fn render_node(node: &Value, out: &mut String) { + let node_type = node.get("type").and_then(|t| t.as_str()).unwrap_or(""); + + match node_type { + "paragraph" => { + out.push_str("

"); + render_inline_content(node, out); + out.push_str("

\n"); + } + "heading" => { + let level = node + .get("attrs") + .and_then(|a| a.get("level")) + .and_then(|l| l.as_u64()) + .unwrap_or(2) + .min(6); + out.push_str(&format!("")); + render_inline_content(node, out); + out.push_str(&format!("\n")); + } + "blockquote" => { + out.push_str("
\n"); + render_children(node, out); + out.push_str("
\n"); + } + "bulletList" | "bullet_list" => { + out.push_str("
    \n"); + render_children(node, out); + out.push_str("
\n"); + } + "orderedList" | "ordered_list" => { + let start = node + .get("attrs") + .and_then(|a| a.get("start")) + .and_then(|s| s.as_u64()) + .unwrap_or(1); + if start == 1 { + out.push_str("
    \n"); + } else { + out.push_str(&format!("
      \n")); + } + render_children(node, out); + out.push_str("
    \n"); + } + "listItem" | "list_item" => { + out.push_str("
  1. "); + render_children(node, out); + out.push_str("
  2. \n"); + } + "codeBlock" | "code_block" => { + let lang = node + .get("attrs") + .and_then(|a| a.get("language")) + .and_then(|l| l.as_str()) + .unwrap_or(""); + if lang.is_empty() { + out.push_str("
    ");
    +            } else {
    +                out.push_str(&format!("
    ", escape_html(lang)));
    +            }
    +            render_inline_content(node, out);
    +            out.push_str("
    \n"); + } + "horizontalRule" | "horizontal_rule" => { + out.push_str("
    \n"); + } + "image" => { + let attrs = node.get("attrs"); + let src = attrs + .and_then(|a| a.get("src")) + .and_then(|s| s.as_str()) + .unwrap_or(""); + let alt = attrs + .and_then(|a| a.get("alt")) + .and_then(|s| s.as_str()) + .unwrap_or(""); + let title = attrs + .and_then(|a| a.get("title")) + .and_then(|s| s.as_str()); + out.push_str(&format!( + "\"{}\"",\n"); + } + "hardBreak" | "hard_break" => { + out.push_str("
    "); + } + _ => { + render_children(node, out); + } + } +} + +fn render_children(node: &Value, out: &mut String) { + if let Some(content) = node.get("content").and_then(|c| c.as_array()) { + for child in content { + render_node(child, out); + } + } +} + +fn render_inline_content(node: &Value, out: &mut String) { + let Some(content) = node.get("content").and_then(|c| c.as_array()) else { + return; + }; + + for child in content { + let child_type = child.get("type").and_then(|t| t.as_str()).unwrap_or(""); + match child_type { + "text" => { + let text = child.get("text").and_then(|t| t.as_str()).unwrap_or(""); + let marks = child.get("marks").and_then(|m| m.as_array()); + render_text_with_marks(text, marks, out); + } + "hardBreak" | "hard_break" => { + out.push_str("
    "); + } + "image" => { + render_node(child, out); + } + _ => { + render_node(child, out); + } + } + } +} + +fn render_text_with_marks(text: &str, marks: Option<&Vec>, out: &mut String) { + let Some(marks) = marks else { + out.push_str(&escape_html(text)); + return; + }; + + let mut close_tags: Vec<&str> = Vec::new(); + for mark in marks { + let mark_type = mark.get("type").and_then(|t| t.as_str()).unwrap_or(""); + match mark_type { + "bold" | "strong" => { + out.push_str(""); + close_tags.push(""); + } + "italic" | "em" => { + out.push_str(""); + close_tags.push(""); + } + "strike" | "strikethrough" => { + out.push_str(""); + close_tags.push(""); + } + "code" => { + out.push_str(""); + close_tags.push(""); + } + "underline" => { + out.push_str(""); + close_tags.push(""); + } + "link" => { + let href = mark + .get("attrs") + .and_then(|a| a.get("href")) + .and_then(|h| h.as_str()) + .unwrap_or("#"); + let target = mark + .get("attrs") + .and_then(|a| a.get("target")) + .and_then(|t| t.as_str()); + out.push_str(&format!(""); + close_tags.push(""); + } + _ => {} + } + } + + out.push_str(&escape_html(text)); + + for tag in close_tags.iter().rev() { + out.push_str(tag); + } +} + +fn escape_html(s: &str) -> String { + s.replace('&', "&") + .replace('<', "<") + .replace('>', ">") +} + +fn escape_attr(s: &str) -> String { + s.replace('&', "&") + .replace('<', "<") + .replace('>', ">") + .replace('"', """) +} + +#[cfg(test)] +mod tests { + use super::*; + use serde_json::json; + + #[test] + fn simple_paragraph() { + let doc = json!({ + "type": "doc", + "content": [{ + "type": "paragraph", + "content": [{ "type": "text", "text": "Hello world" }] + }] + }); + assert_eq!(document_to_html(&doc), "

    Hello world

    \n"); + } + + #[test] + fn heading_levels() { + let doc = json!({ + "type": "doc", + "content": [{ + "type": "heading", + "attrs": { "level": 2 }, + "content": [{ "type": "text", "text": "Title" }] + }] + }); + assert_eq!(document_to_html(&doc), "

    Title

    \n"); + } + + #[test] + fn bold_mark() { + let doc = json!({ + "type": "doc", + "content": [{ + "type": "paragraph", + "content": [{ + "type": "text", + "text": "bold", + "marks": [{ "type": "bold" }] + }] + }] + }); + assert_eq!(document_to_html(&doc), "

    bold

    \n"); + } + + #[test] + fn html_escaping() { + let doc = json!({ + "type": "doc", + "content": [{ + "type": "paragraph", + "content": [{ "type": "text", "text": "" }] + }] + }); + let html = document_to_html(&doc); + assert!(!html.contains("