@jant/core 0.3.32 → 0.3.33
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/client/client.css +1 -1
- package/dist/client/client.js +1442 -989
- package/dist/index.js +1429 -1055
- package/package.json +1 -1
- package/src/__tests__/helpers/app.ts +6 -3
- package/src/__tests__/helpers/db.ts +3 -0
- package/src/client.ts +2 -1
- package/src/db/migrations/0011_add_path_registry.sql +23 -0
- package/src/db/schema.ts +12 -1
- package/src/i18n/locales/en.po +225 -91
- package/src/i18n/locales/en.ts +1 -1
- package/src/i18n/locales/zh-Hans.po +201 -152
- package/src/i18n/locales/zh-Hans.ts +1 -1
- package/src/i18n/locales/zh-Hant.po +201 -152
- package/src/i18n/locales/zh-Hant.ts +1 -1
- package/src/lib/__tests__/excerpt.test.ts +25 -0
- package/src/lib/__tests__/resolve-config.test.ts +26 -2
- package/src/lib/__tests__/timeline.test.ts +2 -1
- package/src/lib/compose-bridge.ts +30 -1
- package/src/lib/excerpt.ts +16 -7
- package/src/lib/nav-manager-bridge.ts +54 -0
- package/src/lib/navigation.ts +7 -4
- package/src/lib/render.tsx +5 -2
- package/src/lib/resolve-config.ts +7 -0
- package/src/lib/view.ts +42 -10
- package/src/middleware/error-handler.ts +16 -0
- package/src/routes/api/__tests__/posts.test.ts +80 -0
- package/src/routes/api/__tests__/settings.test.ts +1 -1
- package/src/routes/api/posts.ts +6 -29
- package/src/routes/api/upload.ts +2 -14
- package/src/routes/auth/__tests__/setup.test.ts +2 -1
- package/src/routes/compose.tsx +13 -5
- package/src/routes/dash/__tests__/pages.test.ts +2 -1
- package/src/routes/dash/__tests__/settings-avatar.test.ts +151 -33
- package/src/routes/dash/appearance.tsx +71 -4
- package/src/routes/dash/collections.tsx +15 -21
- package/src/routes/dash/media.tsx +1 -13
- package/src/routes/dash/pages.tsx +5 -150
- package/src/routes/dash/posts.tsx +25 -32
- package/src/routes/dash/redirects.tsx +9 -11
- package/src/routes/dash/settings.tsx +29 -111
- package/src/routes/feed/__tests__/rss.test.ts +5 -1
- package/src/routes/pages/__tests__/collections.test.ts +2 -1
- package/src/routes/pages/__tests__/featured.test.ts +2 -1
- package/src/routes/pages/page.tsx +20 -25
- package/src/services/__tests__/collection.test.ts +2 -1
- package/src/services/__tests__/media.test.ts +78 -1
- package/src/services/__tests__/navigation.test.ts +2 -1
- package/src/services/__tests__/page.test.ts +78 -1
- package/src/services/__tests__/path-registry.test.ts +165 -0
- package/src/services/__tests__/post-timeline.test.ts +2 -1
- package/src/services/__tests__/post.test.ts +103 -1
- package/src/services/__tests__/redirect.test.ts +53 -4
- package/src/services/__tests__/search.test.ts +2 -1
- package/src/services/__tests__/settings.test.ts +153 -0
- package/src/services/index.ts +12 -4
- package/src/services/media.ts +72 -4
- package/src/services/page.ts +64 -17
- package/src/services/path-registry.ts +160 -0
- package/src/services/post.ts +119 -24
- package/src/services/redirect.ts +23 -3
- package/src/services/settings.ts +181 -0
- package/src/styles/components.css +135 -0
- package/src/styles/tokens.css +6 -1
- package/src/styles/ui.css +70 -26
- package/src/types/bindings.ts +1 -0
- package/src/types/config.ts +7 -2
- package/src/types/constants.ts +9 -1
- package/src/types/sortablejs.d.ts +8 -2
- package/src/types/views.ts +1 -1
- package/src/ui/color-themes.ts +31 -31
- package/src/ui/components/__tests__/jant-settings-avatar.test.ts +0 -3
- package/src/ui/components/__tests__/jant-settings-general.test.ts +2 -6
- package/src/ui/components/jant-compose-dialog.ts +3 -2
- package/src/ui/components/jant-compose-editor.ts +17 -2
- package/src/ui/components/jant-nav-manager.ts +1067 -0
- package/src/ui/components/jant-settings-general.ts +2 -35
- package/src/ui/components/nav-manager-types.ts +72 -0
- package/src/ui/components/settings-types.ts +0 -3
- package/src/ui/compose/ComposePrompt.tsx +3 -11
- package/src/ui/dash/appearance/AdvancedContent.tsx +0 -3
- package/src/ui/dash/appearance/AppearanceNav.tsx +12 -8
- package/src/ui/dash/appearance/ColorThemeContent.tsx +1 -4
- package/src/ui/dash/appearance/FontThemeContent.tsx +0 -3
- package/src/ui/dash/appearance/NavigationContent.tsx +302 -0
- package/src/ui/dash/pages/PagesContent.tsx +74 -0
- package/src/ui/dash/settings/AccountContent.tsx +0 -3
- package/src/ui/dash/settings/GeneralContent.tsx +1 -19
- package/src/ui/dash/settings/SettingsNav.tsx +2 -6
- package/src/ui/feed/NoteCard.tsx +2 -2
- package/src/ui/layouts/DashLayout.tsx +83 -86
- package/src/ui/layouts/SiteLayout.tsx +82 -21
- package/src/lib/nav-reorder.ts +0 -26
- package/src/ui/dash/pages/LinkFormContent.tsx +0 -119
- package/src/ui/dash/pages/UnifiedPagesContent.tsx +0 -203
|
@@ -1 +1 @@
|
|
|
1
|
-
/*eslint-disable*/import type{Messages}from"@lingui/core";export const messages=JSON.parse("{\"+7Wr2a\":[\"Edit: \",[\"title\"]],\"+AXdXp\":[\"標籤和網址是必填的\"],\"+MACwa\":[\"尚未有任何收藏。\"],\"+MH6k9\":[\"加入導航\"],\"+bHzpy\":[\"顯示連結的文字\"],\"+owNNn\":[\"文章\"],\"+zy2Nq\":[\"類型\"],\"/0D1Xp\":[\"編輯收藏集\"],\"/DFKdU\":[\"輸入引用...\"],\"/R/sGB\":[\"密碼已成功更改。\"],\"/Rj5P4\":[\"您的姓名\"],\"/rkqRV\":[\"所有頁面都在您的導航中。\"],\"07Epll\":[\"這將為您的網站和儀表板設置主題。所有顏色主題都支持深色模式。\"],\"0JkyS7\":[\"Create your first page\"],\"0a6MpL\":[\"新重定向\"],\"0ieXE7\":[\"最高評價\"],\"0yIy82\":[\"尚未有精選文章。\"],\"1CU1Td\":[\"URL-safe identifier (lowercase, numbers, hyphens)\"],\"1DBGsz\":[\"筆記\"],\"1Oj1sI\":[\"已保存順序\"],\"1o+wgo\":[\"e.g. The Verge, John Doe\"],\"2N0qpv\":[\"文章標題...\"],\"2cFU6q\":[\"網站頁腳\"],\"2fUwEY\":[\"選擇媒體\"],\"2q/Q7x\":[\"Visibility\"],\"2rJGtU\":[\"頁面標題...\"],\"3SAro+\":[\"Choose a font for your site. All options use system fonts for fast loading.\"],\"3Siwmw\":[\"More options\"],\"3Yvsaz\":[\"302(臨時)\"],\"4/SFQS\":[\"查看網站\"],\"40TVQj\":[\"Custom Path (optional)\"],\"4KzVT6\":[\"刪除頁面\"],\"4Ml90q\":[\"SEO\"],\"4b3oEV\":[\"內容\"],\"4mDPGp\":[\"此頁面的 URL 路徑。使用小寫字母、數字和連字符。\"],\"6C8dEg\":[\"附加文本\"],\"6WdDG7\":[\"頁面\"],\"6YtxFj\":[\"名稱\"],\"6tU2jr\":[\"No collections found.\"],\"71Xwww\":[\"無效的請求\"],\"7G4SBz\":[\"頁面內容(支持Markdown)...\"],\"7Mk+/h\":[\"更新收藏集\"],\"7Q1KKN\":[\"來源路徑\"],\"7aECQB\":[\"無效或已過期的連結\"],\"7nGhhM\":[\"你在想什麼?\"],\"7p5kLi\":[\"儀表板\"],\"7vhWI8\":[\"新密碼\"],\"87a/t/\":[\"標籤\"],\"8HgKQc\":[\"SEO 設定已成功儲存。\"],\"8WX0J+\":[\"您的想法(可選)\"],\"8WtVZw\":[\"無法保存帖子。請再試一次。\"],\"8ZsakT\":[\"密碼\"],\"8qX8Jl\":[\"選擇一個字體搭配以供您的網站使用。所有選項均使用系統字體以加快加載速度。\"],\"8tM8+a\":[\"儲存為草稿\"],\"9+vGLh\":[\"自訂 CSS\"],\"90Luob\":[[\"count\"],\" replies\"],\"A1taO8\":[\"搜尋\"],\"AeXO77\":[\"帳戶\"],\"AyHO4m\":[\"這個收藏是關於什麼的?\"],\"B373X+\":[\"編輯文章\"],\"B495Gs\":[\"檔案館\"],\"BjF0Jv\":[\"僅限小寫字母、數字和連字符\"],\"Cl55aD\":[\"當前密碼不正確。\"],\"D3uuEX\":[\"尚未選擇任何媒體。\"],\"D9Oea+\":[\"永久鏈接\"],\"DCKkhU\":[\"當前密碼\"],\"DHhJ7s\":[\"上一頁\"],\"DPfwMq\":[\"完成\"],\"DoJzLz\":[\"收藏夾\"],\"E80cJw\":[\"刪除此媒體將永久從存儲中移除它。\"],\"EEYbdt\":[\"發佈\"],\"EGwzOK\":[\"完成設置\"],\"EdQY6l\":[\"None\"],\"EkH9pt\":[\"更新\"],\"Eq6YVV\":[\"分數\"],\"FESYvt\":[\"Describe this for people with visual impairments...\"],\"FGrimz\":[\"新帖子\"],\"FkMol5\":[\"精選\"],\"Fxf4jq\":[\"描述(可選)\"],\"GA5A5H\":[\"Delete Collection\"],\"GBDTf5\":[\"身份驗證未配置\"],\"GHg6h/\":[\"post\"],\"GTPbOX\":[\"您的網站導航\"],\"GX2VMa\":[\"建立您的管理員帳戶。\"],\"GbVAnd\":[\"此密碼重設連結無效或已過期。請生成一個新的連結。\"],\"GorKul\":[\"歡迎來到 Jant\"],\"GrZ6fH\":[\"新頁面\"],\"GxkJXS\":[\"上傳中...\"],\"H29JXm\":[\"+ ALT\"],\"HfyyXl\":[\"My Blog\"],\"HiETwV\":[\"Quiet (normal)\"],\"Hzi9AA\":[\"未找到任何帖子。\"],\"I6gXOa\":[\"Path\"],\"I8hDlV\":[\"At least 1 image required for image posts.\"],\"IUwGEM\":[\"保存更改\"],\"IagCbF\":[\"網址\"],\"J4FNfC\":[\"此集合中沒有帖子。\"],\"JIBC/T\":[\"Supported formats: JPEG, PNG, GIF, WebP, SVG. Max size: 10MB.\"],\"Jed1wB\":[\"需要幫助嗎?請訪問<0>文檔</0>。\"],\"JiP4aa\":[\"Published pages are accessible via their path. Drafts are not visible.\"],\"K0r7TC\":[\"What's new?\"],\"K9NcLu\":[\"使用此 URL 將媒體嵌入到您的帖子中。\"],\"KbS2K9\":[\"重設密碼\"],\"KdSsVl\":[\"作者(可選)\"],\"KiJn9B\":[\"備註\"],\"KmGXnO\":[\"您確定要刪除這篇文章嗎?這個操作無法撤銷。\"],\"KsIZ3c\":[\"頁腳已成功保存。\"],\"KuCcWu\":[\"Displayed at the bottom of all posts and pages. Markdown supported.\"],\"L85WcV\":[\"縮略名\"],\"LdyooL\":[\"鏈接\"],\"LkA8jz\":[\"Add alt text\"],\"LkvLQe\":[\"No pages yet.\"],\"M1RvTd\":[\"點擊圖片以查看完整大小\"],\"M2kIWU\":[\"字型主題\"],\"M8kJqa\":[\"草稿\"],\"M9xgHy\":[\"重定向\"],\"MHrjPM\":[\"標題\"],\"MLSRl9\":[\"引用文本\"],\"MWBOxm\":[\"收藏(可選)\"],\"MZbQHL\":[\"未找到結果。\"],\"MdMyne\":[\"來源連結(選填)\"],\"Mhf/H/\":[\"建立重定向\"],\"MnbH31\":[\"頁面\"],\"MqghUt\":[\"搜尋帖子...\"],\"N0APCr\":[\"這會顯示在您預設首頁的部落格文章上方。這也用於您首頁的 meta 描述。\"],\"N40H+G\":[\"所有\"],\"NU2Fqi\":[\"儲存 CSS\"],\"Naqg3G\":[\"未提供檔案\"],\"O3oNi5\":[\"電子郵件\"],\"OCNZaU\":[\"重定向來源的路徑\"],\"ODiSoW\":[\"尚未有帖子。\"],\"ONWvwQ\":[\"上傳\"],\"OVSkIF\":[\"敏捷的棕色狐狸跳過懶惰的狗。\"],\"P/XNX0\":[\"This is used for your favicon.\"],\"PZ7HJ8\":[\"部落格頭像\"],\"Pbm2/N\":[\"創建收藏夾\"],\"QEbNBb\":[\"路徑(例如 /archive)或完整網址(例如 https://example.com)\"],\"QLkhbH\":[\"被引用的文本...\"],\"Qjlym2\":[\"無法創建帳戶\"],\"RDjuBN\":[\"Setup\"],\"Rj01Fz\":[\"連結\"],\"RwGhWy\":[\"包含 \",[\"count\"],\" 則帖子的主題\"],\"RxsRD6\":[\"時區\"],\"SGJDS5\":[\"儲存未配置\"],\"SJmfuf\":[\"網站名稱\"],\"ST+lN2\":[\"尚未上傳任何媒體。\"],\"T0bsor\":[\"設置已成功保存。\"],\"TNFigk\":[\"預設首頁視圖\"],\"Tt5T6+\":[\"Articles\"],\"TxE+Mj\":[\"1 reply\"],\"Tz0i8g\":[\"設定\"],\"U5v6Gh\":[\"編輯頁面\"],\"UDMjsP\":[\"快速操作\"],\"UGT5vp\":[\"Save Settings\"],\"UeOiKl\":[\"無效的輸入\"],\"Ui5/i3\":[\"允許搜尋引擎索引我的網站是可以的\"],\"Uj/btJ\":[\"在我的網站標頭中顯示頭像\"],\"UxKoFf\":[\"Navigation\"],\"V4WsyL\":[\"新增連結\"],\"VUSy8D\":[\"Search failed. Please try again.\"],\"VgozPa\":[\"頭像顯示設置已成功保存。\"],\"VhMDMg\":[\"更改密碼\"],\"WDcQq9\":[\"Unlisted\"],\"Weq9zb\":[\"一般設定\"],\"WmZ/rP\":[\"到路徑\"],\"XrnWzN\":[\"Published!\"],\"Y+7JGK\":[\"創建頁面\"],\"Y75ho6\":[\"其他頁面\"],\"YIix5Y\":[\"Search...\"],\"Z2lfX1\":[\"選擇圖示\"],\"Z3FXyt\":[\"載入中...\"],\"Z6NwTi\":[\"儲存為草稿\"],\"ZQKLI1\":[\"危險區域\"],\"ZUpE9/\":[\"這會顯示在您所有的文章和頁面的底部。支持 Markdown。\"],\"ZhhOwV\":[\"引用\"],\"aAIQg2\":[\"外觀\"],\"aHTB7P\":[\"附加在您帖子上的補充內容\"],\"aT4jc4\":[\"無效的電子郵件或密碼\"],\"aaGV/9\":[\"新連結\"],\"ajBsih\":[\"發佈文章成功。\"],\"alKG0+\":[\"Font Theme\"],\"an5hVd\":[\"Images\"],\"anibOb\":[\"關於這個部落格\"],\"b+/jO6\":[\"301(永久)\"],\"b4VwHs\":[\"未提供檔案。\"],\"bG4pfW\":[\"Displayed above your blog posts on the home page. Also used as the meta description. Markdown supported.\"],\"bHYIks\":[\"登出\"],\"bcE7Mx\":[\"無法更新個人資料。\"],\"biOepV\":[\"← Back to home\"],\"bzSI52\":[\"丟棄\"],\"cTUByn\":[\"最新的在前\"],\"cnGeoo\":[\"刪除\"],\"dEgA5A\":[\"取消\"],\"dmCcPs\":[\"這是用於您的網站圖標和蘋果觸控圖標。為了獲得最佳效果,請上傳至少 180x180 像素的正方形圖片。\"],\"dtQNkT\":[\"選擇的字體主題無效。\"],\"e6Jr7Q\":[\"← 返回收藏夾\"],\"ePK91l\":[\"編輯\"],\"eWLklq\":[\"引用\"],\"eneWvv\":[\"草稿\"],\"er8+x7\":[\"示範帳戶已預填。只需點擊登入。\"],\"f/bxrN\":[\"名稱是必填的。\"],\"f6e0Ry\":[\"Article\"],\"fDGOiR\":[\"密碼不匹配。\"],\"fG7BxZ\":[\"Upload images via the API: POST /api/upload with a file form field.\"],\"fttd2R\":[\"我的收藏\"],\"g3mKmM\":[\"取消導航\"],\"gDx5MG\":[\"編輯連結\"],\"gJH6Bs\":[\"Alt text improves accessibility\"],\"gOwwEy\":[\"儲存空間未配置。\"],\"hDRU5q\":[\"找到 \",[\"0\"],\" 個結果\"],\"hG89Ed\":[\"Image\"],\"hWOZIv\":[\"請輸入您的新密碼。\"],\"hXzOVo\":[\"下一頁\"],\"he3ygx\":[\"複製\"],\"heSQoS\":[\"粘貼一個網址...\"],\"hmXTCY\":[\"選擇的主題無效。\"],\"hrL0Be\":[\"圖示(可選)\"],\"i0vDGK\":[\"排序順序\"],\"iBc+/N\":[\"Custom URL path. Leave empty to use default /p/ID format.\"],\"iDAqU6\":[\"網址(可選)\"],\"iEqmSU\":[\"自訂 CSS 已成功儲存。\"],\"iH8pgl\":[\"返回\"],\"iPHeYN\":[\"Posting...\"],\"ig4hg2\":[\"Let's set up your site.\"],\"jSRrXo\":[\"已發佈的頁面可以通過其標識符訪問。草稿不可見。\"],\"jUV7CU\":[\"上傳頭像\"],\"jVUmOK\":[\"支援Markdown\"],\"jpctdh\":[\"查看\"],\"jt/Ow/\":[\"posts\"],\"jvyYZG\":[\"你在想什麼...\"],\"k1ifdL\":[\"處理中...\"],\"kI1qVD\":[\"格式\"],\"kL1h6U\":[\"移除分隔線\"],\"kNiQp6\":[\"置頂\"],\"kPMIr+\":[\"給它一個標題...\"],\"kd7eBB\":[\"建立連結\"],\"kj6ppi\":[\"條目\"],\"kyNTQ2\":[\"Emoji or icon name\"],\"l/6JHD\":[\"搜尋圖示...\"],\"l6ANt9\":[\"最低評價\"],\"lO1Oow\":[\"上傳成功!\"],\"m16xKo\":[\"Add\"],\"mTOYla\":[\"View all posts →\"],\"mnkknn\":[\"Collection (optional)\"],\"n1ekoW\":[\"登入\"],\"n6QD94\":[\"最舊的在前\"],\"o21Y+P\":[\"條目\"],\"o4dofa\":[\"AUTH_SECRET 未配置\"],\"oJFOZk\":[\"Source Name (optional)\"],\"oKOOsY\":[\"Color Theme\"],\"oYPBa0\":[\"更新頁面\"],\"p2/GCq\":[\"確認密碼\"],\"pB0OKE\":[\"新分隔線\"],\"pRhYH2\":[\"收藏中的帖子 (\",[\"count\"],\")\"],\"pZq3aX\":[\"上傳失敗。請再試一次。\"],\"pnve/d\":[\"個人資料已成功儲存。\"],\"q+hNag\":[\"集合\"],\"qMyM2u\":[\"Source URL (optional)\"],\"qdxmd4\":[\"Attached text\"],\"qiXmlF\":[\"添加媒體\"],\"qt89I8\":[\"草稿已保存。\"],\"quFPTj\":[\"Custom Slug (optional)\"],\"r1MpXi\":[\"Quiet\"],\"rFmBG3\":[\"顏色主題\"],\"rdUucN\":[\"預覽\"],\"rzNUSl\":[\"包含 1 則貼文的主題\"],\"sGajR7\":[\"線程開始\"],\"smzF8S\":[\"顯示 \",[\"remainingCount\"],\" 個更多 \",[\"0\"]],\"ssqvZi\":[\"保存個人資料\"],\"sxkWRg\":[\"Advanced\"],\"t/YqKh\":[\"移除\"],\"tfDRzk\":[\"保存\"],\"tfrt7B\":[\"未配置任何重定向。\"],\"tiq7kl\":[\"頁面 \",[\"page\"]],\"u2f7vd\":[\"Site Description\"],\"u3wRF+\":[\"已發佈\"],\"u6Hp4N\":[\"Markdown\"],\"uAQUqI\":[\"狀態\"],\"vERlcd\":[\"個人資料\"],\"vXIe7J\":[\"語言\"],\"vh0C9b\":[\"尚未有導航連結。請將頁面添加到導航或創建連結。\"],\"vmQmHx\":[\"添加自定義 CSS 以覆蓋任何樣式。使用數據屬性,如 [data-page]、[data-post]、[data-format] 來針對特定元素。\"],\"vzU4k9\":[\"新收藏集\"],\"wEF6Ix\":[\"目的地路徑或 URL\"],\"wK4OTM\":[\"標題(選填)\"],\"wL3cK8\":[\"最新\"],\"wM5UXj\":[\"刪除媒體\"],\"wRR604\":[\"頁面\"],\"wc+17X\":[\"/* 您的自訂 CSS 在這裡 */\"],\"wdGjkd\":[\"No navigation links configured.\"],\"wja8aL\":[\"無標題\"],\"x+doid\":[\"圖片會自動優化:調整大小至最大 1920 像素,轉換為 WebP 格式,並去除元數據。\"],\"x0mzE0\":[\"創建你的第一篇帖子\"],\"x4RuFo\":[\"Back to home\"],\"xYilR2\":[\"媒體\"],\"y0R9F0\":[\"帖子已成功更新。\"],\"y28hnO\":[\"文章\"],\"yQ2kGp\":[\"載入更多\"],\"yjkELF\":[\"確認新密碼\"],\"yz7wBu\":[\"關閉\"],\"yzF66j\":[\"連結\"],\"z1U/Fh\":[\"Rating\"],\"z8ajIE\":[\"找到 1 個結果\"],\"zBFr9G\":[\"粘貼一篇長文章、AI 回應或任何文本...\\n\\nMarkdown 格式將被保留。\"],\"zH6KqE\":[\"Found \",[\"count\"],\" results\"],\"zennIg\":[\"URL安全識別碼(小寫、數字、連字符)。對於CJK標題,slug將在伺服器上自動生成。\"]}")as Messages;
|
|
1
|
+
/*eslint-disable*/import type{Messages}from"@lingui/core";export const messages=JSON.parse("{\"+7Wr2a\":[\"Edit: \",[\"title\"]],\"+AXdXp\":[\"標籤和網址是必填的\"],\"+MACwa\":[\"尚未有任何收藏。\"],\"+MH6k9\":[\"Add to nav\"],\"+bHzpy\":[\"Display text for the link\"],\"+owNNn\":[\"文章\"],\"+zy2Nq\":[\"類型\"],\"/0D1Xp\":[\"編輯收藏集\"],\"/DFKdU\":[\"輸入引用...\"],\"/R/sGB\":[\"密碼已成功更改。\"],\"/Rj5P4\":[\"您的姓名\"],\"/rkqRV\":[\"All pages are in your navigation.\"],\"07Epll\":[\"這將為您的網站和儀表板設置主題。所有顏色主題都支持深色模式。\"],\"0JkyS7\":[\"Create your first page\"],\"0a6MpL\":[\"新重定向\"],\"0ieXE7\":[\"最高評價\"],\"0yIy82\":[\"尚未有精選文章。\"],\"1CU1Td\":[\"URL-safe identifier (lowercase, numbers, hyphens)\"],\"1DBGsz\":[\"筆記\"],\"1Oj1sI\":[\"已保存順序\"],\"1o+wgo\":[\"e.g. The Verge, John Doe\"],\"2N0qpv\":[\"文章標題...\"],\"2cFU6q\":[\"網站頁腳\"],\"2fUwEY\":[\"選擇媒體\"],\"2q/Q7x\":[\"Visibility\"],\"2rJGtU\":[\"頁面標題...\"],\"3SAro+\":[\"Choose a font for your site. All options use system fonts for fast loading.\"],\"3Siwmw\":[\"More options\"],\"3Yvsaz\":[\"302(臨時)\"],\"3uSoGn\":[\"Header Nav Links\"],\"4/SFQS\":[\"查看網站\"],\"40TVQj\":[\"Custom Path (optional)\"],\"4JBD+x\":[\"保存失敗。請再試一次。\"],\"4KzVT6\":[\"刪除頁面\"],\"4Ml90q\":[\"SEO\"],\"4b3oEV\":[\"內容\"],\"4mDPGp\":[\"此頁面的 URL 路徑。使用小寫字母、數字和連字符。\"],\"538Vy5\":[\"尚未有導航項目。請在下方添加頁面、鏈接或啟用系統項目。\"],\"6C8dEg\":[\"附加文本\"],\"6WdDG7\":[\"頁面\"],\"6YtxFj\":[\"名稱\"],\"6tU2jr\":[\"找不到任何收藏。\"],\"71Xwww\":[\"Invalid request\"],\"7G4SBz\":[\"頁面內容(支持Markdown)...\"],\"7Mk+/h\":[\"更新收藏集\"],\"7Q1KKN\":[\"來源路徑\"],\"7aECQB\":[\"無效或已過期的連結\"],\"7aYVPs\":[\"所有頁面都在導航中\"],\"7nGhhM\":[\"你在想什麼?\"],\"7p5kLi\":[\"儀表板\"],\"7vhWI8\":[\"新密碼\"],\"87a/t/\":[\"標籤\"],\"8HgKQc\":[\"SEO 設定已成功儲存。\"],\"8WX0J+\":[\"您的想法(可選)\"],\"8WtVZw\":[\"無法保存帖子。請再試一次。\"],\"8ZsakT\":[\"密碼\"],\"8qX8Jl\":[\"選擇一個字體搭配以供您的網站使用。所有選項均使用系統字體以加快加載速度。\"],\"8tM8+a\":[\"儲存為草稿\"],\"8xE385\":[\"添加到導航\"],\"9+vGLh\":[\"自訂 CSS\"],\"90Luob\":[[\"count\"],\" replies\"],\"A1taO8\":[\"搜尋\"],\"AeXO77\":[\"帳戶\"],\"AyHO4m\":[\"這個收藏是關於什麼的?\"],\"Az4JB1\":[\"Use Featured as default home view\"],\"B373X+\":[\"編輯文章\"],\"B495Gs\":[\"檔案館\"],\"BjF0Jv\":[\"僅限小寫字母、數字和連字符\"],\"Cl55aD\":[\"當前密碼不正確。\"],\"D3uuEX\":[\"尚未選擇任何媒體。\"],\"D9Oea+\":[\"永久鏈接\"],\"DCKkhU\":[\"當前密碼\"],\"DHhJ7s\":[\"上一頁\"],\"DPfwMq\":[\"完成\"],\"DVljCN\":[\"Choose a page…\"],\"DoJzLz\":[\"收藏夾\"],\"E80cJw\":[\"刪除此媒體將永久從存儲中移除它。\"],\"EEYbdt\":[\"發佈\"],\"EGwzOK\":[\"完成設置\"],\"EdQY6l\":[\"None\"],\"EkH9pt\":[\"更新\"],\"Eq6YVV\":[\"分數\"],\"FESYvt\":[\"為視障人士描述這個...\"],\"FGrimz\":[\"新帖子\"],\"FkMol5\":[\"精選\"],\"Fxf4jq\":[\"描述(可選)\"],\"GA5A5H\":[\"Delete Collection\"],\"GBDTf5\":[\"身份驗證未配置\"],\"GHg6h/\":[\"post\"],\"GTPbOX\":[\"Your site navigation\"],\"GX2VMa\":[\"建立您的管理員帳戶。\"],\"GbVAnd\":[\"此密碼重設連結無效或已過期。請生成一個新的連結。\"],\"GlEzsR\":[\"為搜尋引擎和訂閱閱讀器提供的簡短介紹。僅限純文字。\"],\"GorKul\":[\"歡迎來到 Jant\"],\"GrZ6fH\":[\"新頁面\"],\"GxkJXS\":[\"上傳中...\"],\"H29JXm\":[\"+ ALT\"],\"HfyyXl\":[\"My Blog\"],\"HiETwV\":[\"Quiet (normal)\"],\"Hzi9AA\":[\"未找到任何帖子。\"],\"I6gXOa\":[\"Path\"],\"I8hDlV\":[\"At least 1 image required for image posts.\"],\"IUwGEM\":[\"Save Changes\"],\"IagCbF\":[\"網址\"],\"J4FNfC\":[\"此集合中沒有帖子。\"],\"J6bLeg\":[\"添加自定義連結到任何 URL\"],\"JIBC/T\":[\"Supported formats: JPEG, PNG, GIF, WebP, SVG. Max size: 10MB.\"],\"Jed1wB\":[\"需要幫助嗎?請訪問<0>文檔</0>。\"],\"JiP4aa\":[\"Published pages are accessible via their path. Drafts are not visible.\"],\"K0r7TC\":[\"What's new?\"],\"K9NcLu\":[\"使用此 URL 將媒體嵌入到您的帖子中。\"],\"KbS2K9\":[\"重設密碼\"],\"KdSsVl\":[\"作者(可選)\"],\"KiJn9B\":[\"備註\"],\"KmGXnO\":[\"您確定要刪除這篇文章嗎?這個操作無法撤銷。\"],\"KsIZ3c\":[\"Footer saved successfully.\"],\"KuCcWu\":[\"顯示在所有文章和頁面的底部。支持Markdown。\"],\"L85WcV\":[\"縮略名\"],\"LdyooL\":[\"鏈接\"],\"LkA8jz\":[\"添加替代文字\"],\"LkvLQe\":[\"No pages yet.\"],\"M1RvTd\":[\"點擊圖片以查看完整大小\"],\"M2kIWU\":[\"字型主題\"],\"M6CbAU\":[\"切換編輯面板\"],\"M8kJqa\":[\"草稿\"],\"M8lheL\":[\"最大可見導航連結\"],\"M9xgHy\":[\"重定向\"],\"MHrjPM\":[\"標題\"],\"MLSRl9\":[\"引用文本\"],\"MWBOxm\":[\"收藏(可選)\"],\"MZbQHL\":[\"未找到結果。\"],\"MdMyne\":[\"來源連結(選填)\"],\"Mhf/H/\":[\"建立重定向\"],\"MnbH31\":[\"頁面\"],\"MqghUt\":[\"搜尋帖子...\"],\"N0APCr\":[\"This is displayed above your blog posts on your default home page. This is also used for the meta description on your home page.\"],\"N40H+G\":[\"所有\"],\"NU2Fqi\":[\"儲存 CSS\"],\"Naqg3G\":[\"未提供檔案\"],\"O3oNi5\":[\"電子郵件\"],\"OCNZaU\":[\"重定向來源的路徑\"],\"ODiSoW\":[\"尚未有帖子。\"],\"ONWvwQ\":[\"上傳\"],\"OVSkIF\":[\"敏捷的棕色狐狸跳過懶惰的狗。\"],\"OeUWA7\":[\"新增頁面\"],\"P/XNX0\":[\"This is used for your favicon.\"],\"PJnyHS\":[\"Max visible links saved\"],\"PZ7HJ8\":[\"部落格頭像\"],\"Pbm2/N\":[\"創建收藏夾\"],\"QEbNBb\":[\"Path (e.g. /archive) or full URL (e.g. https://example.com)\"],\"QLkhbH\":[\"被引用的文本...\"],\"Qjlym2\":[\"無法創建帳戶\"],\"RDjuBN\":[\"Setup\"],\"Rj01Fz\":[\"連結\"],\"RwGhWy\":[\"包含 \",[\"count\"],\" 則帖子的主題\"],\"RxsRD6\":[\"時區\"],\"SGJDS5\":[\"儲存未配置\"],\"SJmfuf\":[\"網站名稱\"],\"ST+lN2\":[\"尚未上傳任何媒體。\"],\"T0bsor\":[\"設置已成功保存。\"],\"TNFigk\":[\"預設首頁視圖\"],\"Tt5T6+\":[\"Articles\"],\"TxE+Mj\":[\"1 reply\"],\"Tz0i8g\":[\"設定\"],\"U5v6Gh\":[\"編輯頁面\"],\"UDMjsP\":[\"快速操作\"],\"UGT5vp\":[\"Save Settings\"],\"UeOiKl\":[\"無效的輸入\"],\"Ui5/i3\":[\"允許搜尋引擎索引我的網站是可以的\"],\"Uj/btJ\":[\"在我的網站標頭中顯示頭像\"],\"UxKoFf\":[\"導航\"],\"UzGRD9\":[\"Home view saved\"],\"V4WsyL\":[\"新增連結\"],\"VUSy8D\":[\"Search failed. Please try again.\"],\"VgozPa\":[\"頭像顯示設置已成功保存。\"],\"VhMDMg\":[\"更改密碼\"],\"Vn3jYy\":[\"導航項目\"],\"WDcQq9\":[\"Unlisted\"],\"Weq9zb\":[\"一般設定\"],\"WmZ/rP\":[\"到路徑\"],\"XrnWzN\":[\"已發佈!\"],\"Y+7JGK\":[\"創建頁面\"],\"Y75ho6\":[\"Other pages\"],\"YIix5Y\":[\"搜尋...\"],\"Z2lfX1\":[\"選擇圖示\"],\"Z3FXyt\":[\"載入中...\"],\"Z6NwTi\":[\"儲存為草稿\"],\"ZQKLI1\":[\"危險區域\"],\"ZUpE9/\":[\"This is displayed at the bottom of all of your posts and pages. Markdown is supported.\"],\"ZhhOwV\":[\"引用\"],\"ZmUkwN\":[\"Add custom link to navigation\"],\"aAIQg2\":[\"外觀\"],\"aHTB7P\":[\"附加在您帖子上的補充內容\"],\"aT4jc4\":[\"無效的電子郵件或密碼\"],\"aaGV/9\":[\"New Link\"],\"ajBsih\":[\"發佈文章成功。\"],\"alKG0+\":[\"字型主題\"],\"an5hVd\":[\"Images\"],\"anibOb\":[\"關於這個部落格\"],\"b+/jO6\":[\"301(永久)\"],\"b+FyBD\":[\"Add page to navigation\"],\"b+JhJf\":[\"Max visible links\"],\"b4VwHs\":[\"未提供檔案。\"],\"bDqhXY\":[\"刪除失敗。請再試一次。\"],\"bG4pfW\":[\"Displayed above your blog posts on the home page. Also used as the meta description. Markdown supported.\"],\"bHYIks\":[\"登出\"],\"bcE7Mx\":[\"無法更新個人資料。\"],\"biOepV\":[\"← Back to home\"],\"bzSI52\":[\"丟棄\"],\"cTUByn\":[\"最新的在前\"],\"ccaIM9\":[\"更多連結\"],\"cnGeoo\":[\"刪除\"],\"dEgA5A\":[\"取消\"],\"dStw5E\":[\"將現有頁面添加到您的導航中\"],\"dmCcPs\":[\"這是用於您的網站圖標和蘋果觸控圖標。為了獲得最佳效果,請上傳至少 180x180 像素的正方形圖片。\"],\"dtQNkT\":[\"選擇的字體主題無效。\"],\"e6Jr7Q\":[\"← 返回收藏夾\"],\"ePK91l\":[\"編輯\"],\"eWLklq\":[\"引用\"],\"eneWvv\":[\"草稿\"],\"er8+x7\":[\"示範帳戶已預填。只需點擊登入。\"],\"f/bxrN\":[\"名稱是必填的。\"],\"f6e0Ry\":[\"Article\"],\"fDGOiR\":[\"密碼不匹配。\"],\"fG7BxZ\":[\"Upload images via the API: POST /api/upload with a file form field.\"],\"fttd2R\":[\"我的收藏\"],\"g3mKmM\":[\"Un-nav\"],\"gDx5MG\":[\"Edit Link\"],\"gJH6Bs\":[\"替代文字改善可及性\"],\"gOwwEy\":[\"儲存空間未配置。\"],\"hDRU5q\":[\"找到 \",[\"0\"],\" 個結果\"],\"hG89Ed\":[\"Image\"],\"hQAbqI\":[\"尚未有頁面。創建您的第一個頁面以開始使用。\"],\"hWOZIv\":[\"請輸入您的新密碼。\"],\"hXzOVo\":[\"下一頁\"],\"he3ygx\":[\"複製\"],\"heSQoS\":[\"粘貼一個網址...\"],\"hmXTCY\":[\"選擇的主題無效。\"],\"hrL0Be\":[\"圖示(可選)\"],\"i0vDGK\":[\"排序順序\"],\"iBc+/N\":[\"Custom URL path. Leave empty to use default /p/ID format.\"],\"iDAqU6\":[\"網址(可選)\"],\"iEUzMn\":[\"系統\"],\"iEqmSU\":[\"自訂 CSS 已成功儲存。\"],\"iH8pgl\":[\"返回\"],\"iPHeYN\":[\"Posting...\"],\"ig4hg2\":[\"Let's set up your site.\"],\"jSRrXo\":[\"已發佈的頁面可以通過其標識符訪問。草稿不可見。\"],\"jUV7CU\":[\"上傳頭像\"],\"jVUmOK\":[\"支援Markdown\"],\"jpctdh\":[\"查看\"],\"jt/Ow/\":[\"posts\"],\"jvyYZG\":[\"你在想什麼...\"],\"k1ifdL\":[\"處理中...\"],\"kI1qVD\":[\"格式\"],\"kL1h6U\":[\"移除分隔線\"],\"kNiQp6\":[\"置頂\"],\"kPMIr+\":[\"給它一個標題...\"],\"kd7eBB\":[\"Create Link\"],\"kj6ppi\":[\"條目\"],\"kyNTQ2\":[\"Emoji or icon name\"],\"l/6JHD\":[\"搜尋圖示...\"],\"l6ANt9\":[\"最低評價\"],\"lO1Oow\":[\"上傳成功!\"],\"m16xKo\":[\"新增\"],\"mO5HMZ\":[\"All pages are already in navigation.\"],\"mTOYla\":[\"View all posts →\"],\"mnkknn\":[\"Collection (optional)\"],\"n1ekoW\":[\"登入\"],\"n6QD94\":[\"最舊的在前\"],\"o21Y+P\":[\"條目\"],\"o4dofa\":[\"AUTH_SECRET 未配置\"],\"oJFOZk\":[\"Source Name (optional)\"],\"oKOOsY\":[\"顏色主題\"],\"oSiRP0\":[\"系統連結\"],\"oYPBa0\":[\"更新頁面\"],\"p2/GCq\":[\"確認密碼\"],\"pB0OKE\":[\"新分隔線\"],\"pI2MWS\":[\"Search pages…\"],\"pRhYH2\":[\"收藏中的帖子 (\",[\"count\"],\")\"],\"pZq3aX\":[\"上傳失敗。請再試一次。\"],\"pnve/d\":[\"個人資料已成功儲存。\"],\"q+hNag\":[\"集合\"],\"qMyM2u\":[\"Source URL (optional)\"],\"qdxmd4\":[\"Attached text\"],\"qiXmlF\":[\"添加媒體\"],\"qt89I8\":[\"草稿已保存。\"],\"quFPTj\":[\"Custom Slug (optional)\"],\"r1MpXi\":[\"Quiet\"],\"rFmBG3\":[\"顏色主題\"],\"rdUucN\":[\"預覽\"],\"rzNUSl\":[\"包含 1 則貼文的主題\"],\"sDGoxy\":[\"切換內建導航項目。啟用的項目會與頁面和連結一起顯示在您的導航中。\"],\"sGajR7\":[\"線程開始\"],\"smzF8S\":[\"顯示 \",[\"remainingCount\"],\" 個更多 \",[\"0\"]],\"ssqvZi\":[\"保存個人資料\"],\"sxkWRg\":[\"進階\"],\"t/YqKh\":[\"移除\"],\"tfDRzk\":[\"保存\"],\"tfQNeI\":[\"No pages found.\"],\"tfrt7B\":[\"未配置任何重定向。\"],\"tiq7kl\":[\"頁面 \",[\"page\"]],\"u2f7vd\":[\"Site Description\"],\"u3wRF+\":[\"已發佈\"],\"u6Hp4N\":[\"Markdown\"],\"uAQUqI\":[\"狀態\"],\"vERlcd\":[\"個人資料\"],\"vXIe7J\":[\"語言\"],\"vh0C9b\":[\"No navigation links yet. Add pages to navigation or create links.\"],\"vmQmHx\":[\"添加自定義 CSS 以覆蓋任何樣式。使用數據屬性,如 [data-page]、[data-post]、[data-format] 來針對特定元素。\"],\"vzU4k9\":[\"新收藏集\"],\"w8Rv8T\":[\"標籤是必需的\"],\"wEF6Ix\":[\"目的地路徑或 URL\"],\"wK4OTM\":[\"標題(選填)\"],\"wL3cK8\":[\"最新\"],\"wM5UXj\":[\"刪除媒體\"],\"wRR604\":[\"頁面\"],\"wc+17X\":[\"/* 您的自訂 CSS 在這裡 */\"],\"wdGjkd\":[\"No navigation links configured.\"],\"wja8aL\":[\"無標題\"],\"x+doid\":[\"圖片會自動優化:調整大小至最大 1920 像素,轉換為 WebP 格式,並去除元數據。\"],\"x0mzE0\":[\"創建你的第一篇帖子\"],\"x4RuFo\":[\"Back to home\"],\"xYilR2\":[\"媒體\"],\"y0R9F0\":[\"帖子已成功更新。\"],\"y28hnO\":[\"文章\"],\"yQ2kGp\":[\"載入更多\"],\"yjkELF\":[\"確認新密碼\"],\"yz7wBu\":[\"關閉\"],\"yzF66j\":[\"連結\"],\"z1U/Fh\":[\"Rating\"],\"z8ajIE\":[\"找到 1 個結果\"],\"zBFr9G\":[\"粘貼一篇長文章、AI 回應或任何文本...\\n\\nMarkdown 格式將被保留。\"],\"zH6KqE\":[\"Found \",[\"count\"],\" results\"],\"zennIg\":[\"URL安全識別碼(小寫、數字、連字符)。對於CJK標題,slug將在伺服器上自動生成。\"],\"zucql+\":[\"菜單\"]}")as Messages;
|
|
@@ -111,6 +111,31 @@ describe("getHtmlExcerpt", () => {
|
|
|
111
111
|
expect(result.hasMore).toBe(true);
|
|
112
112
|
});
|
|
113
113
|
|
|
114
|
+
it("stops at 5 paragraphs even if under 500 chars", () => {
|
|
115
|
+
const paras = Array.from(
|
|
116
|
+
{ length: 8 },
|
|
117
|
+
(_, i) => `<p>Paragraph ${i + 1}</p>`,
|
|
118
|
+
);
|
|
119
|
+
const html = paras.join("");
|
|
120
|
+
|
|
121
|
+
const result = getHtmlExcerpt(html);
|
|
122
|
+
expect(result.excerpt).toBe(paras.slice(0, 5).join(""));
|
|
123
|
+
expect(result.hasMore).toBe(true);
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
it("stops at char limit before reaching 5 paragraphs", () => {
|
|
127
|
+
const p1 = `<p>${"A".repeat(300)}</p>`;
|
|
128
|
+
const p2 = `<p>${"B".repeat(300)}</p>`;
|
|
129
|
+
const p3 = `<p>${"C".repeat(50)}</p>`;
|
|
130
|
+
const html = p1 + p2 + p3;
|
|
131
|
+
|
|
132
|
+
const result = getHtmlExcerpt(html);
|
|
133
|
+
// 300 + 300 > 500, so only first paragraph fits (second would exceed)
|
|
134
|
+
// Wait: first iteration charCount=0+300=300 < 500, added. Second: 300+300=600 > 500, break.
|
|
135
|
+
expect(result.excerpt).toBe(p1);
|
|
136
|
+
expect(result.hasMore).toBe(true);
|
|
137
|
+
});
|
|
138
|
+
|
|
114
139
|
it("handles paragraphs with nested HTML elements", () => {
|
|
115
140
|
const p1 = `<p>Hello <strong>bold</strong> and <em>italic</em> text here.</p>`;
|
|
116
141
|
const p2 = `<p>${"B".repeat(500)}</p>`;
|
|
@@ -21,7 +21,9 @@ describe("resolveConfig", () => {
|
|
|
21
21
|
const config = resolveConfig(makeEnv(), {});
|
|
22
22
|
|
|
23
23
|
expect(config.siteName).toBe("Jant");
|
|
24
|
-
expect(config.siteDescription).toBe(
|
|
24
|
+
expect(config.siteDescription).toBe(
|
|
25
|
+
"Thoughts, links, and quotes — one post at a time",
|
|
26
|
+
);
|
|
25
27
|
expect(config.siteLanguage).toBe("en");
|
|
26
28
|
expect(config.homeDefaultView).toBe("latest");
|
|
27
29
|
expect(config.timeZone).toBe("UTC");
|
|
@@ -170,6 +172,28 @@ describe("resolveConfig", () => {
|
|
|
170
172
|
expect(config.customCSS).toBe("body { color: red; }");
|
|
171
173
|
});
|
|
172
174
|
|
|
175
|
+
it("resolves headerNavMaxVisible with default, DB override, and clamping", () => {
|
|
176
|
+
// Default is 3
|
|
177
|
+
const config1 = resolveConfig(makeEnv(), {});
|
|
178
|
+
expect(config1.headerNavMaxVisible).toBe(3);
|
|
179
|
+
|
|
180
|
+
// DB override works
|
|
181
|
+
const config2 = resolveConfig(makeEnv(), { HEADER_NAV_MAX_VISIBLE: "5" });
|
|
182
|
+
expect(config2.headerNavMaxVisible).toBe(5);
|
|
183
|
+
|
|
184
|
+
// Clamped to 0 minimum
|
|
185
|
+
const config3 = resolveConfig(makeEnv(), { HEADER_NAV_MAX_VISIBLE: "-1" });
|
|
186
|
+
expect(config3.headerNavMaxVisible).toBe(0);
|
|
187
|
+
|
|
188
|
+
// Clamped to 5 maximum
|
|
189
|
+
const config4 = resolveConfig(makeEnv(), { HEADER_NAV_MAX_VISIBLE: "10" });
|
|
190
|
+
expect(config4.headerNavMaxVisible).toBe(5);
|
|
191
|
+
|
|
192
|
+
// Zero is valid
|
|
193
|
+
const config5 = resolveConfig(makeEnv(), { HEADER_NAV_MAX_VISIBLE: "0" });
|
|
194
|
+
expect(config5.headerNavMaxVisible).toBe(0);
|
|
195
|
+
});
|
|
196
|
+
|
|
173
197
|
it("resolves defaultThemeId from env", () => {
|
|
174
198
|
const config = resolveConfig(
|
|
175
199
|
makeEnv({ DEFAULT_THEME: "dark" } as Partial<Bindings>),
|
|
@@ -179,6 +203,6 @@ describe("resolveConfig", () => {
|
|
|
179
203
|
|
|
180
204
|
// Falls back to default
|
|
181
205
|
const config2 = resolveConfig(makeEnv(), {});
|
|
182
|
-
expect(config2.defaultThemeId).toBe("
|
|
206
|
+
expect(config2.defaultThemeId).toBe("notepad");
|
|
183
207
|
});
|
|
184
208
|
});
|
|
@@ -11,6 +11,7 @@ import { describe, it, expect, beforeEach } from "vitest";
|
|
|
11
11
|
import { createTestDatabase } from "../../__tests__/helpers/db.js";
|
|
12
12
|
import { createPostService } from "../../services/post.js";
|
|
13
13
|
import { createMediaService } from "../../services/media.js";
|
|
14
|
+
import { createPathRegistryService } from "../../services/path-registry.js";
|
|
14
15
|
import { buildMediaMap } from "../media-helpers.js";
|
|
15
16
|
import type { Database } from "../../db/index.js";
|
|
16
17
|
import type { PostWithMedia } from "../../types.js";
|
|
@@ -23,7 +24,7 @@ describe("Timeline data assembly", () => {
|
|
|
23
24
|
beforeEach(() => {
|
|
24
25
|
const testDb = createTestDatabase();
|
|
25
26
|
db = testDb.db as unknown as Database;
|
|
26
|
-
postService = createPostService(db);
|
|
27
|
+
postService = createPostService(db, createPathRegistryService(db));
|
|
27
28
|
mediaService = createMediaService(db);
|
|
28
29
|
});
|
|
29
30
|
|
|
@@ -21,6 +21,9 @@ import {
|
|
|
21
21
|
/** Track in-flight upload promises keyed by clientId */
|
|
22
22
|
const uploadPromises = new Map<string, Promise<string | null>>();
|
|
23
23
|
|
|
24
|
+
/** Track attachments removed while their upload is still in flight */
|
|
25
|
+
const removedClientIds = new Set<string>();
|
|
26
|
+
|
|
24
27
|
/**
|
|
25
28
|
* Upload a single file: process with ImageProcessor, then POST to /api/upload.
|
|
26
29
|
* Returns the mediaId on success, null on failure.
|
|
@@ -67,6 +70,22 @@ function getEditor(): JantComposeEditor | null {
|
|
|
67
70
|
return document.querySelector("jant-compose-editor");
|
|
68
71
|
}
|
|
69
72
|
|
|
73
|
+
// ── Attachment removal handler ───────────────────────────────────────
|
|
74
|
+
|
|
75
|
+
document.addEventListener("jant:attachment-removed", (e: Event) => {
|
|
76
|
+
const { clientId, mediaId } = (
|
|
77
|
+
e as CustomEvent<{ clientId: string; mediaId: string | null }>
|
|
78
|
+
).detail;
|
|
79
|
+
|
|
80
|
+
if (mediaId) {
|
|
81
|
+
// Upload already finished — fire-and-forget delete
|
|
82
|
+
fetch(`/api/upload/${mediaId}`, { method: "DELETE" }).catch(() => {});
|
|
83
|
+
} else {
|
|
84
|
+
// Upload still in flight — mark for cleanup after it finishes
|
|
85
|
+
removedClientIds.add(clientId);
|
|
86
|
+
}
|
|
87
|
+
});
|
|
88
|
+
|
|
70
89
|
// ── File selection handler ──────────────────────────────────────────
|
|
71
90
|
|
|
72
91
|
document.addEventListener("jant:files-selected", (e: Event) => {
|
|
@@ -76,7 +95,17 @@ document.addEventListener("jant:files-selected", (e: Event) => {
|
|
|
76
95
|
const editor = getEditor();
|
|
77
96
|
|
|
78
97
|
for (const { file, clientId } of event.detail.files) {
|
|
79
|
-
const promise = uploadFile(file, clientId, editor)
|
|
98
|
+
const promise = uploadFile(file, clientId, editor).then((mediaId) => {
|
|
99
|
+
// If the attachment was removed while uploading, delete it immediately
|
|
100
|
+
if (removedClientIds.has(clientId)) {
|
|
101
|
+
removedClientIds.delete(clientId);
|
|
102
|
+
if (mediaId) {
|
|
103
|
+
fetch(`/api/upload/${mediaId}`, { method: "DELETE" }).catch(() => {});
|
|
104
|
+
}
|
|
105
|
+
return null;
|
|
106
|
+
}
|
|
107
|
+
return mediaId;
|
|
108
|
+
});
|
|
80
109
|
uploadPromises.set(clientId, promise);
|
|
81
110
|
promise.finally(() => uploadPromises.delete(clientId));
|
|
82
111
|
}
|
package/src/lib/excerpt.ts
CHANGED
|
@@ -28,14 +28,17 @@ export interface HtmlExcerpt {
|
|
|
28
28
|
excerpt: string;
|
|
29
29
|
/** Whether the original content has more text beyond the excerpt */
|
|
30
30
|
hasMore: boolean;
|
|
31
|
+
/** Character offset in the original HTML where the excerpt ends */
|
|
32
|
+
excerptEnd: number;
|
|
31
33
|
}
|
|
32
34
|
|
|
33
35
|
/**
|
|
34
36
|
* Extracts a paragraph-aware HTML excerpt from body HTML.
|
|
35
37
|
*
|
|
36
|
-
* Uses a greedy algorithm: accumulates paragraphs until
|
|
37
|
-
* plain-text length exceeds 500 characters
|
|
38
|
-
*
|
|
38
|
+
* Uses a greedy algorithm: accumulates paragraphs until either
|
|
39
|
+
* the total plain-text length exceeds 500 characters or 5 paragraphs
|
|
40
|
+
* have been collected, whichever comes first. At least one paragraph
|
|
41
|
+
* is always included.
|
|
39
42
|
*
|
|
40
43
|
* If the content contains a `<!--more-->` marker, the content before
|
|
41
44
|
* the marker is used as the excerpt instead.
|
|
@@ -62,26 +65,32 @@ export function getHtmlExcerpt(bodyHtml: string): HtmlExcerpt {
|
|
|
62
65
|
// Honor manual <!--more--> marker
|
|
63
66
|
if (bodyHtml.includes("<!--more-->")) {
|
|
64
67
|
const excerpt = bodyHtml.split("<!--more-->")[0] ?? "";
|
|
65
|
-
return {
|
|
68
|
+
return {
|
|
69
|
+
excerpt,
|
|
70
|
+
hasMore: true,
|
|
71
|
+
excerptEnd: excerpt.length + "<!--more-->".length,
|
|
72
|
+
};
|
|
66
73
|
}
|
|
67
74
|
|
|
68
75
|
const paragraphs = bodyHtml.match(/<p>[\s\S]*?<\/p>/g) || [];
|
|
69
76
|
|
|
70
77
|
// No paragraphs found — return full content
|
|
71
78
|
if (paragraphs.length === 0) {
|
|
72
|
-
return { excerpt: bodyHtml, hasMore: false };
|
|
79
|
+
return { excerpt: bodyHtml, hasMore: false, excerptEnd: bodyHtml.length };
|
|
73
80
|
}
|
|
74
81
|
|
|
75
82
|
let excerpt = "";
|
|
76
83
|
let charCount = 0;
|
|
84
|
+
let paraCount = 0;
|
|
77
85
|
|
|
78
86
|
for (const p of paragraphs) {
|
|
79
87
|
const textLen = stripHtml(p).length;
|
|
80
|
-
if (charCount + textLen > 500 && excerpt) break;
|
|
88
|
+
if ((charCount + textLen > 500 || paraCount >= 5) && excerpt) break;
|
|
81
89
|
excerpt += p;
|
|
82
90
|
charCount += textLen;
|
|
91
|
+
paraCount++;
|
|
83
92
|
}
|
|
84
93
|
|
|
85
94
|
const hasMore = excerpt.length < bodyHtml.length;
|
|
86
|
-
return { excerpt, hasMore };
|
|
95
|
+
return { excerpt, hasMore, excerptEnd: excerpt.length };
|
|
87
96
|
}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Nav Manager Bridge
|
|
3
|
+
*
|
|
4
|
+
* Handles communication between <jant-nav-manager> and the server.
|
|
5
|
+
* Listens for `jant:nav-update` and `jant:nav-delete`, calls API endpoints,
|
|
6
|
+
* and reloads the page on success.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import type {
|
|
10
|
+
NavManagerUpdateDetail,
|
|
11
|
+
NavManagerDeleteDetail,
|
|
12
|
+
} from "../ui/components/nav-manager-types.js";
|
|
13
|
+
import { showToast } from "./toast.js";
|
|
14
|
+
|
|
15
|
+
document.addEventListener("jant:nav-update", async (event: Event) => {
|
|
16
|
+
const { detail } = event as CustomEvent<NavManagerUpdateDetail>;
|
|
17
|
+
if (!detail?.id) return;
|
|
18
|
+
|
|
19
|
+
try {
|
|
20
|
+
const res = await fetch(`/api/nav-items/${detail.id}`, {
|
|
21
|
+
method: "PUT",
|
|
22
|
+
headers: {
|
|
23
|
+
"Content-Type": "application/json",
|
|
24
|
+
Accept: "application/json",
|
|
25
|
+
},
|
|
26
|
+
body: JSON.stringify({
|
|
27
|
+
label: detail.label,
|
|
28
|
+
...(detail.url !== undefined && { url: detail.url }),
|
|
29
|
+
}),
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
if (!res.ok) throw new Error(`HTTP ${res.status}`);
|
|
33
|
+
window.location.reload();
|
|
34
|
+
} catch {
|
|
35
|
+
showToast("Failed to save. Please try again.", "error");
|
|
36
|
+
}
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
document.addEventListener("jant:nav-delete", async (event: Event) => {
|
|
40
|
+
const { detail } = event as CustomEvent<NavManagerDeleteDetail>;
|
|
41
|
+
if (!detail?.id) return;
|
|
42
|
+
|
|
43
|
+
try {
|
|
44
|
+
const res = await fetch(`/api/nav-items/${detail.id}`, {
|
|
45
|
+
method: "DELETE",
|
|
46
|
+
headers: { Accept: "application/json" },
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
if (!res.ok) throw new Error(`HTTP ${res.status}`);
|
|
50
|
+
window.location.reload();
|
|
51
|
+
} catch {
|
|
52
|
+
showToast("Failed to delete. Please try again.", "error");
|
|
53
|
+
}
|
|
54
|
+
});
|
package/src/lib/navigation.ts
CHANGED
|
@@ -10,16 +10,18 @@ import { toNavItemViews } from "./view.js";
|
|
|
10
10
|
import { render as renderMarkdown } from "./markdown.js";
|
|
11
11
|
|
|
12
12
|
/**
|
|
13
|
-
* Navigation data needed by
|
|
13
|
+
* Navigation data needed by public page rendering
|
|
14
14
|
*/
|
|
15
15
|
export interface NavigationData {
|
|
16
16
|
links: NavItemView[];
|
|
17
17
|
currentPath: string;
|
|
18
18
|
siteName: string;
|
|
19
|
+
/** Used as meta description fallback and in RSS/Atom feeds */
|
|
19
20
|
siteDescription: string;
|
|
20
21
|
isAuthenticated: boolean;
|
|
21
22
|
collections: Collection[];
|
|
22
23
|
homeDefaultView: string;
|
|
24
|
+
headerNavMaxVisible: number;
|
|
23
25
|
siteAvatarUrl?: string;
|
|
24
26
|
showHeaderAvatar?: boolean;
|
|
25
27
|
siteFooterHtml?: string;
|
|
@@ -65,9 +67,7 @@ export async function getNavigationData(c: Context): Promise<NavigationData> {
|
|
|
65
67
|
// Render footer markdown
|
|
66
68
|
const siteFooterHtml = siteFooter ? renderMarkdown(siteFooter) : undefined;
|
|
67
69
|
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
// Check auth status for compose button
|
|
70
|
+
// Check auth status (needed for compose button and system nav items)
|
|
71
71
|
let isAuthenticated = false;
|
|
72
72
|
let collections: Collection[] = [];
|
|
73
73
|
try {
|
|
@@ -79,6 +79,8 @@ export async function getNavigationData(c: Context): Promise<NavigationData> {
|
|
|
79
79
|
// Not authenticated
|
|
80
80
|
}
|
|
81
81
|
|
|
82
|
+
const links = toNavItemViews(items, currentPath, isAuthenticated);
|
|
83
|
+
|
|
82
84
|
// Only load collections when authenticated (for compose dialog)
|
|
83
85
|
if (isAuthenticated) {
|
|
84
86
|
collections = await c.var.services.collections.list();
|
|
@@ -92,6 +94,7 @@ export async function getNavigationData(c: Context): Promise<NavigationData> {
|
|
|
92
94
|
isAuthenticated,
|
|
93
95
|
collections,
|
|
94
96
|
homeDefaultView,
|
|
97
|
+
headerNavMaxVisible: appConfig.headerNavMaxVisible,
|
|
95
98
|
siteAvatarUrl,
|
|
96
99
|
showHeaderAvatar: showHeaderAvatar && !!siteAvatarUrl,
|
|
97
100
|
siteFooterHtml,
|
package/src/lib/render.tsx
CHANGED
|
@@ -45,14 +45,17 @@ export interface RenderPublicPageOptions {
|
|
|
45
45
|
export function renderPublicPage(c: Context, options: RenderPublicPageOptions) {
|
|
46
46
|
const { title, description, navData, content, sidebar } = options;
|
|
47
47
|
|
|
48
|
+
// Use siteDescription as meta description fallback when not explicitly provided
|
|
49
|
+
const metaDescription = description || navData.siteDescription || undefined;
|
|
50
|
+
|
|
48
51
|
const layoutProps: SiteLayoutProps = {
|
|
49
52
|
siteName: navData.siteName,
|
|
50
|
-
siteDescription: navData.siteDescription,
|
|
51
53
|
links: navData.links,
|
|
52
54
|
currentPath: navData.currentPath,
|
|
53
55
|
isAuthenticated: navData.isAuthenticated,
|
|
54
56
|
collections: navData.collections,
|
|
55
57
|
homeDefaultView: navData.homeDefaultView,
|
|
58
|
+
headerNavMaxVisible: navData.headerNavMaxVisible,
|
|
56
59
|
siteAvatarUrl: navData.siteAvatarUrl,
|
|
57
60
|
showHeaderAvatar: navData.showHeaderAvatar,
|
|
58
61
|
siteFooterHtml: navData.siteFooterHtml,
|
|
@@ -68,7 +71,7 @@ export function renderPublicPage(c: Context, options: RenderPublicPageOptions) {
|
|
|
68
71
|
return c.html(
|
|
69
72
|
<BaseLayout
|
|
70
73
|
title={title}
|
|
71
|
-
description={
|
|
74
|
+
description={metaDescription}
|
|
72
75
|
c={c}
|
|
73
76
|
faviconUrl={faviconUrl}
|
|
74
77
|
faviconVersion={faviconVersion}
|
|
@@ -112,6 +112,13 @@ export function resolveConfig(
|
|
|
112
112
|
siteDescriptionExplicit,
|
|
113
113
|
siteLanguage: resolve("SITE_LANGUAGE", allSettings, env),
|
|
114
114
|
homeDefaultView: resolve("HOME_DEFAULT_VIEW", allSettings, env),
|
|
115
|
+
headerNavMaxVisible: (() => {
|
|
116
|
+
const parsed = parseInt(
|
|
117
|
+
resolve("HEADER_NAV_MAX_VISIBLE", allSettings, env),
|
|
118
|
+
10,
|
|
119
|
+
);
|
|
120
|
+
return Math.max(0, Math.min(5, isNaN(parsed) ? 3 : parsed));
|
|
121
|
+
})(),
|
|
115
122
|
timeZone: resolve("TIME_ZONE", allSettings, env),
|
|
116
123
|
siteFooter: resolve("SITE_FOOTER", allSettings, env),
|
|
117
124
|
noindex: resolve("NOINDEX", allSettings, env) === "true",
|
package/src/lib/view.ts
CHANGED
|
@@ -122,10 +122,20 @@ export function toPostView(post: PostWithMedia, _ctx: MediaContext): PostView {
|
|
|
122
122
|
// Pre-compute HTML summary for article-style posts (with title)
|
|
123
123
|
let summaryHtml: string | undefined;
|
|
124
124
|
let summaryHasMore: boolean | undefined;
|
|
125
|
+
let bodyHtmlWithAnchor = post.bodyHtml;
|
|
125
126
|
if (post.title && post.bodyHtml) {
|
|
126
127
|
const result = getHtmlExcerpt(post.bodyHtml);
|
|
127
128
|
summaryHtml = result.excerpt;
|
|
128
129
|
summaryHasMore = result.hasMore;
|
|
130
|
+
|
|
131
|
+
// Inject #continue anchor at the excerpt boundary for scroll targeting
|
|
132
|
+
if (result.hasMore) {
|
|
133
|
+
const pos = result.excerptEnd;
|
|
134
|
+
bodyHtmlWithAnchor =
|
|
135
|
+
post.bodyHtml.slice(0, pos) +
|
|
136
|
+
'<span id="continue"></span>' +
|
|
137
|
+
post.bodyHtml.slice(pos);
|
|
138
|
+
}
|
|
129
139
|
}
|
|
130
140
|
|
|
131
141
|
// Convert media attachments
|
|
@@ -144,7 +154,7 @@ export function toPostView(post: PostWithMedia, _ctx: MediaContext): PostView {
|
|
|
144
154
|
permalink,
|
|
145
155
|
path: post.path ?? undefined,
|
|
146
156
|
title: post.title ?? undefined,
|
|
147
|
-
bodyHtml:
|
|
157
|
+
bodyHtml: bodyHtmlWithAnchor ?? undefined,
|
|
148
158
|
excerpt,
|
|
149
159
|
summaryHtml,
|
|
150
160
|
summaryHasMore,
|
|
@@ -219,26 +229,43 @@ export function toPageView(page: Page): PageView {
|
|
|
219
229
|
|
|
220
230
|
/**
|
|
221
231
|
* Converts a NavItem to a NavItemView with pre-computed state.
|
|
232
|
+
*
|
|
233
|
+
* @param item - Raw nav item from database
|
|
234
|
+
* @param currentPath - Current URL path for active state
|
|
235
|
+
* @param isAuthenticated - Whether the user is logged in (affects system dashboard item)
|
|
222
236
|
*/
|
|
223
|
-
export function toNavItemView(
|
|
224
|
-
|
|
225
|
-
|
|
237
|
+
export function toNavItemView(
|
|
238
|
+
item: NavItem,
|
|
239
|
+
currentPath: string,
|
|
240
|
+
isAuthenticated = false,
|
|
241
|
+
): NavItemView {
|
|
242
|
+
let url = item.url;
|
|
243
|
+
let label = item.label;
|
|
244
|
+
|
|
245
|
+
// System dashboard item: resolve URL and label based on auth
|
|
246
|
+
if (item.type === "system" && item.url === "/dash") {
|
|
247
|
+
url = isAuthenticated ? "/dash" : "/signin";
|
|
248
|
+
if (!isAuthenticated) {
|
|
249
|
+
label = "Sign in";
|
|
250
|
+
}
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
const isExternal = url.startsWith("http://") || url.startsWith("https://");
|
|
226
254
|
|
|
227
255
|
let isActive = false;
|
|
228
256
|
if (!isExternal) {
|
|
229
|
-
if (
|
|
257
|
+
if (url === "/") {
|
|
230
258
|
isActive = currentPath === "/";
|
|
231
259
|
} else {
|
|
232
|
-
isActive =
|
|
233
|
-
currentPath === item.url || currentPath.startsWith(item.url + "/");
|
|
260
|
+
isActive = currentPath === url || currentPath.startsWith(url + "/");
|
|
234
261
|
}
|
|
235
262
|
}
|
|
236
263
|
|
|
237
264
|
return {
|
|
238
265
|
id: item.id,
|
|
239
266
|
type: item.type as NavItemType,
|
|
240
|
-
label
|
|
241
|
-
url
|
|
267
|
+
label,
|
|
268
|
+
url,
|
|
242
269
|
pageId: item.pageId ?? undefined,
|
|
243
270
|
isActive,
|
|
244
271
|
isExternal,
|
|
@@ -247,12 +274,17 @@ export function toNavItemView(item: NavItem, currentPath: string): NavItemView {
|
|
|
247
274
|
|
|
248
275
|
/**
|
|
249
276
|
* Batch converts NavItem[] to NavItemView[].
|
|
277
|
+
*
|
|
278
|
+
* @param items - Raw nav items from database
|
|
279
|
+
* @param currentPath - Current URL path for active state
|
|
280
|
+
* @param isAuthenticated - Whether the user is logged in
|
|
250
281
|
*/
|
|
251
282
|
export function toNavItemViews(
|
|
252
283
|
items: NavItem[],
|
|
253
284
|
currentPath: string,
|
|
285
|
+
isAuthenticated = false,
|
|
254
286
|
): NavItemView[] {
|
|
255
|
-
return items.map((item) => toNavItemView(item, currentPath));
|
|
287
|
+
return items.map((item) => toNavItemView(item, currentPath, isAuthenticated));
|
|
256
288
|
}
|
|
257
289
|
|
|
258
290
|
// =============================================================================
|
|
@@ -46,6 +46,22 @@ export const errorHandler: ErrorHandler<Env> = (err, c) => {
|
|
|
46
46
|
return dsToast("An unexpected error occurred", "error");
|
|
47
47
|
}
|
|
48
48
|
|
|
49
|
+
// JSON-accepting requests (Lit bridges)
|
|
50
|
+
if (c.req.header("accept")?.includes("application/json")) {
|
|
51
|
+
if (err instanceof DomainError) {
|
|
52
|
+
const body: Record<string, unknown> = {
|
|
53
|
+
error: err.message,
|
|
54
|
+
code: err.code,
|
|
55
|
+
};
|
|
56
|
+
if (err instanceof ValidationError && err.details)
|
|
57
|
+
body.details = err.details;
|
|
58
|
+
return c.json(body, err.statusCode as ContentfulStatusCode);
|
|
59
|
+
}
|
|
60
|
+
// eslint-disable-next-line no-console -- Server error logging is intentional
|
|
61
|
+
console.error("[Jant] Unhandled error:", err);
|
|
62
|
+
return c.json({ error: "Internal server error" }, 500);
|
|
63
|
+
}
|
|
64
|
+
|
|
49
65
|
// Non-API routes: map NotFoundError to Hono's built-in 404
|
|
50
66
|
if (err instanceof NotFoundError) {
|
|
51
67
|
return c.notFound();
|
|
@@ -484,5 +484,85 @@ describe("Posts API Routes", () => {
|
|
|
484
484
|
|
|
485
485
|
expect(res.status).toBe(404);
|
|
486
486
|
});
|
|
487
|
+
|
|
488
|
+
it("deletes media records when post is deleted", async () => {
|
|
489
|
+
const { app, services } = createTestApp({ authenticated: true });
|
|
490
|
+
app.route("/api/posts", postsApiRoutes);
|
|
491
|
+
|
|
492
|
+
const post = await services.posts.create({
|
|
493
|
+
format: "note",
|
|
494
|
+
body: "with media",
|
|
495
|
+
});
|
|
496
|
+
|
|
497
|
+
const m1 = await services.media.create({
|
|
498
|
+
filename: "a.jpg",
|
|
499
|
+
originalName: "a.jpg",
|
|
500
|
+
mimeType: "image/jpeg",
|
|
501
|
+
size: 1024,
|
|
502
|
+
storageKey: "media/2025/01/a.jpg",
|
|
503
|
+
});
|
|
504
|
+
const m2 = await services.media.create({
|
|
505
|
+
filename: "b.jpg",
|
|
506
|
+
originalName: "b.jpg",
|
|
507
|
+
mimeType: "image/jpeg",
|
|
508
|
+
size: 2048,
|
|
509
|
+
storageKey: "media/2025/01/b.jpg",
|
|
510
|
+
});
|
|
511
|
+
|
|
512
|
+
await services.media.attachToPost(post.id, [m1.id, m2.id]);
|
|
513
|
+
|
|
514
|
+
const res = await app.request(`/api/posts/${sqid.encode(post.id)}`, {
|
|
515
|
+
method: "DELETE",
|
|
516
|
+
});
|
|
517
|
+
|
|
518
|
+
expect(res.status).toBe(200);
|
|
519
|
+
|
|
520
|
+
// Media records should be deleted, not just detached
|
|
521
|
+
expect(await services.media.getById(m1.id)).toBeNull();
|
|
522
|
+
expect(await services.media.getById(m2.id)).toBeNull();
|
|
523
|
+
});
|
|
524
|
+
|
|
525
|
+
it("deletes media for all posts in a thread when root is deleted", async () => {
|
|
526
|
+
const { app, services } = createTestApp({ authenticated: true });
|
|
527
|
+
app.route("/api/posts", postsApiRoutes);
|
|
528
|
+
|
|
529
|
+
const root = await services.posts.create({
|
|
530
|
+
format: "note",
|
|
531
|
+
body: "thread root",
|
|
532
|
+
});
|
|
533
|
+
const reply = await services.posts.create({
|
|
534
|
+
format: "note",
|
|
535
|
+
body: "reply",
|
|
536
|
+
replyToId: root.id,
|
|
537
|
+
});
|
|
538
|
+
|
|
539
|
+
const rootMedia = await services.media.create({
|
|
540
|
+
filename: "root.jpg",
|
|
541
|
+
originalName: "root.jpg",
|
|
542
|
+
mimeType: "image/jpeg",
|
|
543
|
+
size: 1024,
|
|
544
|
+
storageKey: "media/2025/01/root.jpg",
|
|
545
|
+
});
|
|
546
|
+
const replyMedia = await services.media.create({
|
|
547
|
+
filename: "reply.jpg",
|
|
548
|
+
originalName: "reply.jpg",
|
|
549
|
+
mimeType: "image/jpeg",
|
|
550
|
+
size: 2048,
|
|
551
|
+
storageKey: "media/2025/01/reply.jpg",
|
|
552
|
+
});
|
|
553
|
+
|
|
554
|
+
await services.media.attachToPost(root.id, [rootMedia.id]);
|
|
555
|
+
await services.media.attachToPost(reply.id, [replyMedia.id]);
|
|
556
|
+
|
|
557
|
+
const res = await app.request(`/api/posts/${sqid.encode(root.id)}`, {
|
|
558
|
+
method: "DELETE",
|
|
559
|
+
});
|
|
560
|
+
|
|
561
|
+
expect(res.status).toBe(200);
|
|
562
|
+
|
|
563
|
+
// Both root and reply media should be deleted
|
|
564
|
+
expect(await services.media.getById(rootMedia.id)).toBeNull();
|
|
565
|
+
expect(await services.media.getById(replyMedia.id)).toBeNull();
|
|
566
|
+
});
|
|
487
567
|
});
|
|
488
568
|
});
|