skills/blog-seo-check/SKILL.md
Post-writing SEO validation with pass/fail checklist covering title tag length and keyword placement, meta description quality, heading hierarchy and keyword density, internal/external link audit with anchor text analysis, canonical URL verification, Open Graph meta tags (og:title, og:description, og:image), Twitter Card validation, URL structure optimization, and image alt text presence. Produces prioritized fix list with specific recommendations. Use when user says "seo check", "check seo", "validate seo", "blog seo", "seo validation", "on-page seo", "title tag check", "meta description check", "heading check", "link audit".
npx skillsauth add agricidaniel/claude-blog blog-seo-checkInstall 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.
Runs a comprehensive on-page SEO validation against a completed blog post and generates a pass/fail checklist with specific fixes for each failure. Designed to run after writing - catches issues before publishing.
Read the target file and extract:
If the user provides a URL instead of a file path, use WebFetch to retrieve the page and extract the relevant elements.
| Check | Pass Criteria | |-------|---------------| | Character count | 40-60 characters (no truncation in SERPs) | | Keyword placement | Primary keyword in first half of title | | Power word | Contains at least one power word (e.g., Guide, Best, How, Why, Essential, Proven, Complete) | | Truncation risk | No critical meaning lost if truncated at 60 chars | | Uniqueness | Not generic - specific to the content |
| Check | Pass Criteria | |-------|---------------| | Character count | 150-160 characters | | Statistic included | Contains at least one specific number or data point | | Value proposition | Ends with clear reader benefit or value proposition | | Keyword presence | Primary keyword appears naturally (not stuffed) | | No keyword stuffing | Keyword appears at most once | | Call to action | Implies action (learn, discover, find out, see) |
| Check | Pass Criteria | |-------|---------------| | Single H1 | Exactly one H1 tag (the title) | | No skipped levels | H1 -> H2 -> H3, never H1 -> H3 or H2 -> H4 | | Keyword in headings | Primary keyword in 2-3 headings (natural, not forced) | | Question format | 60-70% of H2 headings are questions | | H2 count | 6-8 H2 sections for a standard blog post | | Heading length | Each heading under 70 characters |
| Check | Pass Criteria | |-------|---------------| | Link count | 3-10 internal links per post | | Anchor text | Descriptive (not "click here" or "read more") | | Bidirectional | Check if linked pages also link back (flag if not) | | No orphan status | Post links to at least 3 other pages on the site | | Link distribution | Links spread across the post, not clustered | | No self-links | Post does not link to itself |
Use Grep and Glob to scan the project for existing blog content and verify bidirectional linking where possible.
| Check | Pass Criteria | |-------|---------------| | No duplicate URLs | Each URL appears at most once in body content | | Best instance kept | If duplicates exist, keep the one with most descriptive anchor text | | Navigation exempt | Header/footer nav links don't count toward body dedup | | Fragment normalization | URLs with different #fragments treated as same URL |
For each duplicate found:
Google records 1-2 anchor texts per URL per page (Zyppy 2023). Optimal: link to same URL once in body content; 5-10 internal links per 2,000 words; max ~50 total links per page.
| Check | Pass Criteria | |-------|---------------| | Source tier | Links to tier 1-3 sources only (authoritative, not SEO blogs) | | Broken links | Use WebFetch to verify top external links are reachable | | Rel attributes | External links have appropriate rel attributes (nofollow for sponsored/UGC) | | Link count | At least 3 external links to authoritative sources | | No competitor links | Not linking to direct competitors unnecessarily |
For every public statistic in the post, verify all three components:
retrieved YYYY-MM-DD for each cited source.Posts that fail any of the three either drop the unverifiable claim or replace it with a verified alternative. See skills/blog/references/flow-alignment.md. For a one-shot prompt-driven check, see /blog flow optimize.
| Check | Pass Criteria | |-------|---------------| | Present | Canonical URL is defined in frontmatter or meta tags | | Correct format | Full absolute URL (https://domain.com/path) | | Trailing slash | Consistent with site convention (no mixed trailing slashes) | | Self-referencing | Canonical points to the page itself (unless intentional cross-domain) |
| Check | Pass Criteria | |-------|---------------| | og:title | Present, matches or complements the title tag | | og:description | Present, 150-160 characters, compelling for social sharing | | og:image | Present, 1200x630 minimum dimensions, absolute URL | | og:type | Set to "article" for blog posts | | og:url | Present, matches canonical URL | | og:site_name | Present, matches site/brand name |
| Check | Pass Criteria | |-------|---------------| | twitter:card | Set to "summary_large_image" for blog posts | | twitter:title | Present, under 70 characters | | twitter:description | Present, under 200 characters | | twitter:image | Present, same as or similar to og:image | | twitter:site | Present if the site has a Twitter/X account |
| Check | Pass Criteria | |-------|---------------| | Length | Short - under 75 characters for the path portion | | Keyword presence | Primary keyword or close variant in the URL slug | | No dates | URL does not contain /2025/ or /2026/ date segments | | No special characters | Only lowercase letters, numbers, and hyphens | | Lowercase | Entire URL path is lowercase | | No stop words | Minimal use of "the", "a", "and", "of" in slug | | No file extension | No .html or .php in the URL (clean URLs) |
Output a comprehensive SEO validation report in this format:
## SEO Validation Report: [Title]
**File**: [path or URL]
**Date**: [check date]
**Overall**: [X/Y checks passed] - [PASS/NEEDS WORK/FAIL]
### Results
| # | Check | Status | Details | Fix |
|---|-------|--------|---------|-----|
| 1 | Title length | PASS | 52 chars | - |
| 2 | Title keyword | PASS | "keyword" in first half | - |
| 3 | Title power word | FAIL | No power word found | Add "Guide", "Essential", or "Complete" |
| 4 | Meta description length | PASS | 155 chars | - |
| 5 | Meta description stat | FAIL | No number found | Add a key statistic from the post |
| ... | ... | ... | ... | ... |
### Summary
**Passed**: [N] checks
**Failed**: [N] checks
### Priority Fixes
1. [Most impactful fix - what to change and where]
2. [Second most impactful fix]
3. [Third most impactful fix]
### Notes
- [Any observations about overall SEO health]
- [Suggestions for improvement beyond the checklist]
Status values:
If the post has a published URL and blog-google credentials are available:
python3 skills/blog-google/scripts/run.py google_auth --check --jsonpython3 skills/blog-google/scripts/run.py pagespeed_check <url> --jsondevelopment
Research what people are actually saying about a topic in the last 30 days across Reddit, X / Twitter, YouTube, Hacker News, dev.to, Medium, and other public discourse platforms. API-free; uses WebSearch with platform-targeted site operators plus recency filters. Produces DISCOURSE.md (a structured brief) and JSON output the writer can consume. Complements blog-researcher (which focuses on authority sources) with a recency-and-engagement lens. Use when user says "blog discourse", "discourse research", "what are people saying about", "research what people are saying", "voice of customer", "social listening", "30-day research", "trend research", "what's the discussion on", "real-time research", "practitioner discourse", "/blog discourse".
documentation
Establish durable brand and voice context for cross-skill consumption. Generates BRAND.md (audience, positioning, do/don't editorial rules, taboo phrases, competitor differentiation) and VOICE.md (existing persona JSON re-expressed as readable prose), both written to the project root. When present, all blog sub-skills auto-load these files before writing or reviewing. Pairs with blog-persona, which manages the structured persona JSON. Use when user says "blog brand", "create brand context", "brand voice doc", "BRAND.md", "VOICE.md", "establish editorial brand", "brand guidelines for blog".
testing
Translate existing blog posts into one or more target languages with SEO-optimized localization. Produces native-quality translations that preserve markdown structure, frontmatter, schema JSON-LD, image and chart embeds, and citation capsules. Localizes keywords, meta tags, numbers, dates, currencies, and quote styles per locale. Flags machine-translation artifacts for review. Run BEFORE blog-localize: this handles language conversion; localize handles cultural adaptation after translation completes. Use when user says "translate blog", "blog translate", "uebersetzen", "traduire", "traducir", "translate post", "blog auf Deutsch", "blog en espanol".
testing
One-command multilingual blog creation. Writes a blog post, translates it into user-specified languages, applies cultural adaptation, and emits hreflang tags, sitemap entries, and a CMS-ready language map. The complete write-to-publish pipeline for international content. Orchestrates blog-write, blog-translate, blog-localize, and (optionally) seo-hreflang. Use when user says "multilingual blog", "blog multilingual", "write in multiple languages", "international blog", "mehrsprachiger Blog", "blog multilingue", "blog multilingue", "create blog in German and French".