Open Graph protocol
Open Graph tags control how pages look when shared on social platforms and chat apps. Set og:title, og:description, og:image, og:url, and og:type on every page.
What it is
Open Graph (OG) is a set of <meta> tags, originally introduced by Facebook in 2010 and now supported by virtually every social platform, chat app, and link-preview tool. They tell the platform how to display your page when someone pastes the URL.
<meta property="og:title" content="The lang attribute on <html>" />
<meta property="og:description" content="Set a valid BCP 47 language tag on <html> so screen readers, translators, and search engines know what language the page is in." />
<meta property="og:image" content="https://example.com/og/html-lang.png" />
<meta property="og:image:width" content="1200" />
<meta property="og:image:height" content="675" />
<meta property="og:url" content="https://example.com/foundations/html-lang" />
<meta property="og:type" content="article" />
Note property= (not name=) — Open Graph predates the standard meta name registry and uses the RDFa attribute.
Why it matters
When a link is pasted into Slack, Discord, iMessage, WhatsApp, LinkedIn, Mastodon, Bluesky, or X, the platform fetches the page and reads the OG tags to build a preview card. Without them, the platform falls back to whatever it can scrape — sometimes the <title> and meta description, sometimes the first image it can find, sometimes nothing. The result is unpredictable and often unflattering.
A good preview card gives the link a thumbnail, a headline, and a one-line description. Posts with card previews get measurably more clicks than bare URLs. For any content you want shared, OG tags are the difference between a polished preview and a naked URL.
How to implement
Six tags do most of the work:
og:title— the headline. Usually shorter than the HTML<title>(no site suffix needed; the platform shows the domain separately).og:description— the snippet under the title. 60–200 characters reads well on most platforms.og:image— an absolute URL to the preview image.og:url— the canonical absolute URL of the page. Match your<link rel="canonical">.og:type— the kind of object:websitefor the homepage,articlefor posts,productfor shop items.og:site_name— the human-readable site name. Slack renders it as the small label above the title; Facebook and LinkedIn also use it when present.
For the image, there are two sensible sizes:
- 1200 × 630 (1.91:1) — the historical Open Graph default. Works on Facebook, LinkedIn, Slack, Discord, iMessage, Bluesky, and Mastodon.
- 1200 × 675 (16:9) — also works on every OG platform, and satisfies Google Discover, which only surfaces large-format cards when the image is at least 1200 px wide and roughly 16:9. If you care about appearing in Discover (and you should, if your content is Discover-eligible), 1200 × 675 is the more portable choice — and you also need
<meta name="robots" content="max-image-preview:large">for Google to use the full-width version.
Other constraints, regardless of which size you pick:
- File size under 5 MB across most platforms, but WhatsApp drops the preview entirely if the image is over 300 KB — and WhatsApp is one of the most-used unfurlers on the planet. Treat 300 KB as the real ceiling.
- Format: PNG or JPEG is the safe universal choice. WebP is now accepted by virtually every modern unfurler (Facebook, LinkedIn, Discord, iMessage, Slack, Mastodon, Bluesky, X, WhatsApp) and is a good pick when you need to stay under WhatsApp’s 300 KB cap on a denser image. AVIF is not yet safe — only Facebook, Threads, Pinterest, and WhatsApp render AVIF previews today; the rest will silently drop them. If your CMS auto-converts uploads to AVIF, force the
og:imageURL to stay PNG/JPEG/WebP. - No critical content within 60 pixels of any edge — platforms crop differently.
- Declare
og:image:widthandog:image:height. Some platforms refuse to use images they cannot pre-size. - Serve over HTTPS. HTTP images are rejected.
- Use absolute URLs. Relative
og:imagepaths are not portable.
For X (Twitter), add Twitter Card tags as a fallback. X prefers them when present:
<meta name="twitter:card" content="summary_large_image" />
<meta name="twitter:title" content="The lang attribute on <html>" />
<meta name="twitter:description" content="Set a valid BCP 47 language tag on <html>..." />
<meta name="twitter:image" content="https://example.com/og/html-lang.png" />
If you skip the Twitter Card tags, X will fall back to your OG tags, which is fine. The one extra value worth adding is twitter:card set to summary_large_image so the preview shows the full-width image instead of a small thumbnail.
Two more Twitter Card tags are worth setting because Slack renders them as extra key–value rows below the description in the unfurl card:
<meta name="twitter:label1" content="Reading time" />
<meta name="twitter:data1" content="6 min" />
<meta name="twitter:label2" content="Written by" />
<meta name="twitter:data2" content="Jane Doe" />
Up to two pairs are shown. Keep each value short — roughly 25 characters fits without truncation. Good uses are reading time, author, price, publication date, category, or rating. Slack is currently the main consumer of these tags; X used to render them on a now-deprecated card type, and most other platforms ignore them, so treat them as a Slack-specific bonus, not a replacement for og:description.
Slack also checks for oEmbed discovery before falling back to OG/Twitter Card tags — either a <link rel="alternate" type="application/json+oembed"> in the head or an application/json+oembed link in the HTTP Link header. If you ship oEmbed (most CMSes do for video and rich embeds), Slack uses it first; OG tags are the fallback. The two should describe the same page.
Generate the image per page when you can. A unique illustration or screenshot per article is more engaging than a single site-wide card. Many sites generate them on-demand at build time or with an edge function.
Locale tags for multilingual sites. If your page has translated versions, declare them with og:locale (this page) and og:locale:alternate (every other locale). Use the underscored language_TERRITORY form, not the BCP 47 hyphenated form used by lang and hreflang:
<meta property="og:locale" content="en_GB" />
<meta property="og:locale:alternate" content="fr_FR" />
<meta property="og:locale:alternate" content="de_DE" />
Without og:locale, social platforms guess from the crawler’s Accept-Language and may render the wrong-language card. The full set of head changes for a translated page — title, description, OG, JSON-LD inLanguage, image alt — is covered in localised-metadata.
Common mistakes
- Using
name="og:title"instead ofproperty="og:title". The protocol requiresproperty. - Relative URLs in
og:imageorog:url. Always absolute, always HTTPS. - Image dimensions that drift from 1.91:1 or 16:9 — cropped badly on every platform.
- Shipping a 1200 × 630 image and wanting Google Discover traffic. Discover wants 16:9 at ≥ 1200 px wide; 1200 × 675 satisfies both worlds.
- A beautiful 600 KB PNG that WhatsApp silently refuses to preview. Compress, downscale, or switch to WebP until you are under 300 KB.
- Serving
og:imageas AVIF. Most platforms cannot render it and will skip the card entirely. - Missing
og:image:widthandog:image:height. Some platforms skip the image entirely without them. - One generic OG image reused on every page. Works, but missing an opportunity.
og:urlthat does not match the canonical URL. The two should agree.- Missing
og:site_name. Slack and Facebook fall back to the bare domain, which looks unbranded. - Stuffing a sentence into
twitter:data1. The field is for a short value, not a description — long strings get truncated in the Slack card. - Burying OG tags below 32 KB of inline CSS or scripts in the
<head>. Slackbot only fetches the first 32 KB with aRangerequest; meta tags past that cut-off are never seen and the link silently fails to unfurl. Put OG tags near the top of<head>, above any large inline blocks.
Verification
- View source and check the five core tags are present and absolute.
- Paste the URL into a debugger (Facebook Sharing Debugger, LinkedIn Post Inspector, Slack’s link unfurler, or a direct paste into a chat).
- Confirm the image renders at full width, the title is correct, and the description matches.
- After updating tags, re-scrape on platforms that cache previews aggressively (Facebook, LinkedIn) so they pick up the new values.
Related topics
Sources & further reading
- The Open Graph protocol — ogp.me
- MDN — The Open Graph protocol — MDN
- X — About Cards — X
- Google Search Central — Google Discover and your website — Google
- WhatsApp — Link previews — Meta
- Slack — Unfurling links in messages — Slack
- Open Graph Plus — Slack unfurl tags — Open Graph Plus
- Open Graph Plus — How Slack crawls your page — Open Graph Plus