skills/nuxt-seo-audit/SKILL.md
Use when auditing or setting up SEO for a Nuxt site. Triggers include missing page titles, no structured data, sitemap not including all pages, hreflang/canonical issues, pages not prerendered, trailing slash redirect loops, AI crawlers blocked in robots.txt, or preparing a site for search engine indexing.
npx skillsauth add razbakov/skills nuxt-seo-auditInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Full SEO audit for a Nuxt site. Check live responses, not just source code — what Google sees is what matters.
Run each check against the live deployed site using curl.
# Check every page type
for url in "/" "/about" "/blog" "/projects" "/cv"; do
title=$(curl -sL "https://example.com${url}" | grep -oP '(?<=<title>).*(?=</title>)')
echo "$url — $title"
done
What to check:
<title>titleTemplate set in nuxt.config.ts (e.g., %s · Site Name)curl -sL "https://example.com/about" | grep -oP '(?<=name="description" content=")[^"]*'
What to check:
useSeoMeta({ description })t('seo.pageDesc') not hardcoded English/de/about, /ru/blog, etc.curl -sL "https://example.com/about" | grep -oE '<meta property="og:[^"]*" content="[^"]*"'
What to check:
og:title — can differ from <title> (voice-first for social sharing)og:description — short, punchy versionog:image — set on every page (avatar for static, hero for blog posts)twitter:card — summary for static pages, summary_large_image for posts with imagescurl -sL "https://example.com/" | grep -oP '(?<=<script type="application/ld\+json">).*?(?=</script>)' | python3 -m json.tool
Required schema per page type:
| Page | Schema types |
|---|---|
| Home | WebSite, WebPage |
| About | AboutPage, Person (job, employer, languages, social links) |
| Blog index | CollectionPage, BreadcrumbList |
| Blog post | BlogPosting (headline, date, author, image), BreadcrumbList |
| Project index | WebPage, BreadcrumbList |
| Project detail | SoftwareApplication (name, url, author), BreadcrumbList |
Nuxt implementation: Use useSchemaOrg() with defineArticle, definePerson, defineBreadcrumb, defineSoftwareApp, defineWebSite, defineWebPage from nuxt-schema-org.
Validate at: Google Rich Results Test (search.google.com/test/rich-results)
curl -sL "https://example.com/sitemap_index.xml" | head -30
curl -sL "https://example.com/__sitemap__/en.xml" | grep -c "<url>"
What to check:
robots.txthreflang alternates// server/api/__sitemap__/urls.ts
import { serverQueryContent } from "#content/server";
import { defineSitemapEventHandler, asSitemapUrl } from "#imports";
export default defineSitemapEventHandler(async (e) => {
const content = await serverQueryContent(e).find();
return content
.filter((c) => c._path && !c._path.startsWith("/data/"))
.map((c) => asSitemapUrl({ loc: c._path, lastmod: c.date }));
});
curl -sL "https://example.com/robots.txt"
What to check:
Disallow on ClaudeBot, GPTBot, Google-Extended, Applebot-Extended. If blocked, content is invisible to AI search (Perplexity, ChatGPT, Google AI Overviews)for url in "/" "/about" "/de/about" "/blog"; do
echo "=== $url ==="
curl -sL "https://example.com${url}" | grep -oE '<link[^>]*(canonical|hreflang)[^>]*>'
echo
done
What to check:
<link rel="canonical">hreflang alternates for all languages + x-defaultNuxt i18n setup:
baseUrl in i18n config (required for hreflang generation)language property to each locale definitionuseLocaleHead doesn't work in app.vue during SSR, generate links manually based on route.pathfor url in "/" "/about" "/blog"; do
cache=$(curl -sI "https://example.com${url}" | grep -i "cache-control" | head -1)
code=$(curl -sI "https://example.com${url}" | head -1 | awk '{print $2}')
echo "$code $url — $cache"
done
What to check:
nuxt generate (not nuxt build) for full staticnitro.preset: "static" to prevent hosting platform SSR overridecache-control: public indicates static file servingcache-control or SSR-style headers means server-renderedNuxt config for static:
nitro: {
preset: "static",
prerender: {
crawlLinks: true,
routes: ["/", "/about", "/blog", "/projects"],
failOnError: false,
},
}
curl -sI "https://example.com/about" | grep -i "HTTP\|location"
What to check:
pretty_urls = false in netlify.toml to prevent trailing slash redirects_redirects file for legacy rules that conflict with current pagescat public/_redirects
What to check:
/about / when /about is a real page)## SEO Audit: [site URL]
| Check | Status | Issues |
|---|---|---|
| Page titles | Pass/Fail | ... |
| Meta descriptions | Pass/Fail | ... |
| OG tags | Pass/Fail | ... |
| Structured data | Pass/Fail | ... |
| Sitemap | Pass/Fail | ... |
| Robots.txt | Pass/Fail | ... |
| Hreflang/canonical | Pass/Fail | ... |
| Prerendering | Pass/Fail | ... |
| Trailing slashes | Pass/Fail | ... |
| Legacy redirects | Pass/Fail | ... |
### Issues Found
...
### Fixes Applied
...
t() keysbaseUrl in i18n config — hreflang tags won't generate without ituseLocaleHead in app.vue not reactive to route during SSR — may need manual link generationnuxt generate_redirects containing legacy rules that 301 away from real pagesfailOnError: true (default) failing build on broken internal linksdevelopment
Seed a new or empty Instagram account with a 9-post grid (3×3) so the profile looks established the moment a new visitor lands. Designed for festivals, new businesses, product launches, conferences, communities — any time an empty IG profile would hurt conversion from external traffic (QR scans, flyer drops, cross-promo). Generates assets via /image-from-gemini (per content-publishing rules — never HTML), writes captions with hashtag sets, and outputs a posting order + cadence plan. Trigger generously: phrases like '9 posts for instagram', 'fill my IG', 'starter grid', 'launch grid', 'instagram seed', '9-post grid', 'IG account not to look empty', 'first instagram posts', 'feed bootstrap', '3x3 grid', 'instagram launch content'. Even if the user mentions only one piece (just the images, just the captions, just the order), use this skill — the grid only works as an integrated bundle.
testing
Translate one English blog post into multiple target languages via parallel sub-agents, preserving frontmatter conventions, hero image, and brand voice. Use when the user shares a published English post URL or markdown path and says 'translate it', 'add other languages', 'publish in DE/ES/RU/UK', 'translate to 5 languages', or asks for localized versions of a specific post.
development
Build a complete press kit for an event, product launch, or campaign — in multiple languages — and publish it as a shareable Google Drive folder ready to send to journalists, partners, or a delegate. Produces press releases (typically DE/EN/ES, or configurable), uploads press photos and flyers, creates an Overview document for at-a-glance briefing, and creates a Handover document with pending tasks, contacts, risks, and decisions so press distribution can be delegated. Use when the user says 'I need a press release', 'create a press kit', 'press release in X languages', 'set up a Drive folder for press', 'handover doc for someone else to run press', or has an upcoming announcement that needs to be sent to media. Trigger generously: even partial requests (just a press release, just a flyer folder) typically evolve into the full kit.
development
Track ticket sales for a live event (concert, festival, conference, workshop) with daily snapshots, generate a burndown chart comparing actual sales to ideal-linear targets and tier-cumulative milestones, and report whether the event is on pace. Use when the user asks how sales are going, wants to know if their event will sell out, asks for a daily sales report, wants to set up sales tracking for an upcoming event, or asks about ticket pace / velocity / projection. Trigger generously: phrases like 'how is concert sales going', 'burndown for my event', 'are we going to sell out', 'sales velocity', 'daily ticket chart', 'how many tickets do we need to sell', or any case where the user has a ticketed event with a fixed sales window and wants visibility on pacing.