Commit 0942a22
committed
feat(links): branded social-preview cards with auto OG metadata
When a Umami short link is shared on Twitter/X, Facebook, LinkedIn,
Slack, Discord, WhatsApp, Telegram, or iMessage, the platform's crawler
now receives an Open Graph card from Umami's own domain instead of being
307'd straight through to the destination. The card content is
auto-derived from the destination URL's OG tags at create/update time,
so existing-link UX stays set-and-forget. Humans keep the existing
307 redirect path with UTMs merged.
- 6 new Link columns: ogTitle / ogDescription / ogImage plus per-field
*Manual booleans tracking whether each value was user-set (preserved
across URL changes) or auto-detected (re-fetched on URL change).
- src/lib/og.ts: SSRF-hardened OG fetcher built on undici with a custom
connect.lookup (closes the DNS-rebinding TOCTOU), an IP-literal
pre-check that allows only `unicast` ranges (rejects loopback,
private, link-local, NAT64 / 6to4 / teredo transition prefixes, and
IPv4-mapped IPv6), manual redirect loop with per-hop revalidation,
body/timeout/content-type caps, and entity-decoded regex meta
extraction with surrogate guards.
- src/lib/og-html.ts: single-line HTML renderer with strict per-response
headers (CSP, X-Frame-Options DENY, Referrer-Policy, private/no-store
+ Vary: User-Agent). Conditional meta tag emission so empty fields
produce no broken-image cards.
- /q/[slug] route: isbot() branch returns OG HTML; human branch
unchanged.
- GET /api/links/og-preview: authed live-preview endpoint powering the
form's debounced auto-detection.
- API schemas (create + update) gain the three OG fields with the
'' → null Zod transform plus a public-http(s)-only refinement on
ogImage that reuses the same SSRF guard at the API edge. Update
route preserves the undefined-vs-null PATCH distinction.
- Form: collapsible "Customize preview" section with image preview,
AbortController-cancellable live-preview fetch as you type, dirty-
field-based submit normalization to avoid spurious clear-to-auto.
- LinksTable: minmax(0, 1fr) on flex columns so long destinations no
longer blow out row width.
- 5 new i18n keys added to en-US and translated into all 51 other
locales at correct alphabetical positions (each locale diff is
exactly +5 lines).
- New direct dep: undici@^8.2.0 for the Agent + connect.lookup APIs.
Existing links are non-destructively migrated; old rows fall back to a
basic preview card (name as title, "Redirects to <host>" as description,
og:image omitted → twitter:card downgrades to summary). Future edits
organically backfill via the URL-change re-parse path.1 parent b3b65fc commit 0942a22
67 files changed
Lines changed: 1180 additions & 66 deletions
File tree
- prisma
- migrations/22_add_link_og_metadata
- public/intl/messages
- src
- app
- (collect)/q/[slug]
- (main)/links
- api/links
- [linkId]
- og-preview
- components
- lib
- queries/prisma
Some content is hidden
Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
114 | 114 | | |
115 | 115 | | |
116 | 116 | | |
| 117 | + | |
117 | 118 | | |
118 | 119 | | |
119 | 120 | | |
| |||
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
| 1 | + | |
| 2 | + | |
| 3 | + | |
| 4 | + | |
| 5 | + | |
| 6 | + | |
| 7 | + | |
| 8 | + | |
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
287 | 287 | | |
288 | 288 | | |
289 | 289 | | |
290 | | - | |
291 | | - | |
292 | | - | |
293 | | - | |
294 | | - | |
295 | | - | |
296 | | - | |
297 | | - | |
298 | | - | |
299 | | - | |
300 | | - | |
301 | | - | |
302 | | - | |
303 | | - | |
| 290 | + | |
| 291 | + | |
| 292 | + | |
| 293 | + | |
| 294 | + | |
| 295 | + | |
| 296 | + | |
| 297 | + | |
| 298 | + | |
| 299 | + | |
| 300 | + | |
| 301 | + | |
| 302 | + | |
| 303 | + | |
| 304 | + | |
| 305 | + | |
| 306 | + | |
| 307 | + | |
| 308 | + | |
| 309 | + | |
304 | 310 | | |
305 | 311 | | |
306 | 312 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
27 | 27 | | |
28 | 28 | | |
29 | 29 | | |
| 30 | + | |
30 | 31 | | |
31 | 32 | | |
32 | 33 | | |
| |||
74 | 75 | | |
75 | 76 | | |
76 | 77 | | |
| 78 | + | |
| 79 | + | |
77 | 80 | | |
78 | 81 | | |
79 | 82 | | |
| |||
140 | 143 | | |
141 | 144 | | |
142 | 145 | | |
| 146 | + | |
143 | 147 | | |
144 | 148 | | |
145 | 149 | | |
| |||
230 | 234 | | |
231 | 235 | | |
232 | 236 | | |
| 237 | + | |
233 | 238 | | |
234 | 239 | | |
235 | 240 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
27 | 27 | | |
28 | 28 | | |
29 | 29 | | |
| 30 | + | |
30 | 31 | | |
31 | 32 | | |
32 | 33 | | |
| |||
74 | 75 | | |
75 | 76 | | |
76 | 77 | | |
| 78 | + | |
| 79 | + | |
77 | 80 | | |
78 | 81 | | |
79 | 82 | | |
| |||
140 | 143 | | |
141 | 144 | | |
142 | 145 | | |
| 146 | + | |
143 | 147 | | |
144 | 148 | | |
145 | 149 | | |
| |||
230 | 234 | | |
231 | 235 | | |
232 | 236 | | |
| 237 | + | |
233 | 238 | | |
234 | 239 | | |
235 | 240 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
27 | 27 | | |
28 | 28 | | |
29 | 29 | | |
| 30 | + | |
30 | 31 | | |
31 | 32 | | |
32 | 33 | | |
| |||
74 | 75 | | |
75 | 76 | | |
76 | 77 | | |
| 78 | + | |
| 79 | + | |
77 | 80 | | |
78 | 81 | | |
79 | 82 | | |
| |||
140 | 143 | | |
141 | 144 | | |
142 | 145 | | |
| 146 | + | |
143 | 147 | | |
144 | 148 | | |
145 | 149 | | |
| |||
230 | 234 | | |
231 | 235 | | |
232 | 236 | | |
| 237 | + | |
233 | 238 | | |
234 | 239 | | |
235 | 240 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
27 | 27 | | |
28 | 28 | | |
29 | 29 | | |
| 30 | + | |
30 | 31 | | |
31 | 32 | | |
32 | 33 | | |
| |||
74 | 75 | | |
75 | 76 | | |
76 | 77 | | |
| 78 | + | |
| 79 | + | |
77 | 80 | | |
78 | 81 | | |
79 | 82 | | |
| |||
140 | 143 | | |
141 | 144 | | |
142 | 145 | | |
| 146 | + | |
143 | 147 | | |
144 | 148 | | |
145 | 149 | | |
| |||
230 | 234 | | |
231 | 235 | | |
232 | 236 | | |
| 237 | + | |
233 | 238 | | |
234 | 239 | | |
235 | 240 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
25 | 25 | | |
26 | 26 | | |
27 | 27 | | |
| 28 | + | |
28 | 29 | | |
29 | 30 | | |
30 | 31 | | |
| |||
54 | 55 | | |
55 | 56 | | |
56 | 57 | | |
| 58 | + | |
| 59 | + | |
| 60 | + | |
| 61 | + | |
57 | 62 | | |
58 | 63 | | |
59 | 64 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
25 | 25 | | |
26 | 26 | | |
27 | 27 | | |
| 28 | + | |
28 | 29 | | |
29 | 30 | | |
30 | 31 | | |
| |||
54 | 55 | | |
55 | 56 | | |
56 | 57 | | |
| 58 | + | |
| 59 | + | |
| 60 | + | |
| 61 | + | |
57 | 62 | | |
58 | 63 | | |
59 | 64 | | |
| |||
0 commit comments