Write and publish blog posts for the block/goose open source project
52
58%
Does it follow best practices?
Impact
—
No eval scenarios have been run
Passed
No known issues
Optimize this skill with Tessl
npx tessl skill review --optimize ./goose-blog-post/SKILL.mdWrite blog posts for the block/goose open source project blog powered by Docusaurus.
<goose-repo>/documentation/blog/authors.yml and blog directory exist before proceedingIf the repo is not cloned, clone it:
gh repo clone block/gooseAsk the user how they want to work. There are three modes:
"I have a draft" — The author has written (or will write) their own content. The agent scaffolds the directory, frontmatter, and image setup, then places the author's content into the correct format and reviews it against blog conventions.
"I have notes/an outline" — The author has rough ideas, bullet points, or an outline. The agent expands these into a full draft while preserving the author's voice and key points. Present the draft for the author's review before finalizing.
"Write it for me" — The author provides a topic and key points. The agent writes the full post. Present the draft for the author's review before finalizing.
Important: For modes 1 and 2, the author's voice and intent take priority. Do not rewrite their content unnecessarily. Focus on structure, conventions, and polish — not on replacing their words.
Ask the user for the following (do not assume any of these):
authors.yml)For modes 2 and 3, also ask: 4. Key points: What are the main things the post should cover? 5. Tone: Technical deep-dive, casual walkthrough, announcement, etc.?
For mode 1, ask: 4. Where is the draft? A file path, or ask them to paste it in.
Check <goose-repo>/documentation/blog/authors.yml for the author key.
If the author does not exist, create a new entry. The format is:
authorkey:
name: Full Name
title: Job Title
image_url: https://avatars.githubusercontent.com/u/<github-id>?v=4
page: true
socials:
github: github-username
x: x-username
linkedin: linkedin-usernameAsk the user for any missing details (name, title, GitHub username, social handles). The image_url can be derived from their GitHub profile: https://avatars.githubusercontent.com/u/<id>?v=4. Look up their GitHub user ID if needed:
gh api users/<username> --jq '.id'Blog posts use the naming convention:
YYYY-MM-DD-slug-title/
index.md
banner.pngRules:
date +%Y-%m-%d to get it)<goose-repo>/documentation/blog/mkdir -p documentation/blog/YYYY-MM-DD-slug-titleCreate index.md inside the new directory. The file structure is:
---
title: "Your Blog Post Title"
description: "A concise summary of the post (1-2 sentences). Used in social previews and SEO."
authors:
- authorkey
---Frontmatter rules:
title — wrap in quotes, use title casedescription — wrap in quotes, keep to 1-2 sentences, make it compelling for social sharingauthors — a YAML list of author keys from authors.yml (supports multiple authors)date field — Docusaurus extracts the date from the directory nametags in frontmatter — the blog does not use Docusaurus tagsEvery post must include a banner image immediately after the frontmatter:
Banner image requirements:
index.mdbanner.png (or .jpg, .webp)If the user does not have an image ready, add the image reference as a placeholder and remind them to add the file before publishing. You can also offer to generate one if image generation tools are available.
The intro paragraph(s) appear before the truncate marker. This controls the preview on the blog index page:

Your compelling introduction paragraph that hooks the reader.
<!--truncate-->The intro should:
For mode 1 (author's draft): Place the author's content here. Adjust only what's needed to match the formatting conventions below. Flag any issues for the author rather than silently rewriting.
For modes 2 and 3: Write the content following the guidelines below, then present the full draft to the author for review.
Formatting Conventions:
## for major sections and ### for subsections# (h1) in the body — the title from frontmatter is the h1```python, ```bash, etc.)Voice & Style:
goose-Specific Conventions:
https://block.github.io/goose/https://github.com/block/goosehttps://block.github.io/goose/extensions<head> Section) — RequiredEvery blog post must end with a <head> section containing Open Graph and Twitter card meta tags. This is placed at the very end of the index.md file, after all content:
<head>
<meta property="og:title" content="YOUR TITLE HERE" />
<meta property="og:type" content="article" />
<meta property="og:url" content="https://block.github.io/goose/blog/YYYY/MM/DD/slug-title" />
<meta property="og:description" content="YOUR DESCRIPTION HERE" />
<meta property="og:image" content="https://block.github.io/goose/assets/images/BANNER_FILENAME_WITH_HASH.png" />
<meta name="twitter:card" content="summary_large_image" />
<meta property="twitter:domain" content="block.github.io" />
<meta name="twitter:title" content="YOUR TITLE HERE" />
<meta name="twitter:description" content="YOUR DESCRIPTION HERE" />
<meta name="twitter:image" content="https://block.github.io/goose/assets/images/BANNER_FILENAME_WITH_HASH.png" />
</head>Important: The og:image and twitter:image URLs require a Docusaurus-generated filename that includes a content hash. You cannot guess this filename — it must be obtained from the local preview. See Step 6 for how to get it.
Field reference:
og:title / twitter:title — same as the frontmatter titleog:description / twitter:description — same as the frontmatter descriptionog:url — the production URL: https://block.github.io/goose/blog/YYYY/MM/DD/slug-titleog:image / twitter:image — the static image URL obtained from local preview (see Step 6)twitter:card — always summary_large_imagetwitter:domain — always block.github.ioWhen first creating the post, use https://block.github.io/goose/assets/images/BANNER_FILENAME_WITH_HASH.png as the placeholder for og:image and twitter:image. The BANNER_FILENAME_WITH_HASH portion will be replaced with the real filename after Step 6.
The author needs to preview the blog post locally to:
<head> social metadataTell the user to run this in a separate terminal (do not run this command directly — it blocks the terminal):
cd <goose-repo>/documentation
npm startThis starts a local dev server at http://localhost:3000 with hot reloading.
Once the preview is running, instruct the user to:
http://localhost:3000/goose/assets/images/banner-a1b2c3d4e5f6.pngbanner-a1b2c3d4e5f6.png)https://block.github.io/goose/assets/images/banner-a1b2c3d4e5f6.pngUpdate the <head> section's og:image and twitter:image with this production URL.
This step is critical. Before considering the post done:
index.md contentIterate based on feedback until the author is satisfied.
Run through this checklist and fix any issues:
YYYY-MM-DD-slug-title/ naming conventionindex.md exists in the directorytitle, description, and authorsauthors.yml<!--truncate--> marker is placed after the intro# (h1) headers in the body — only ## and below<head> section is present at the end of the fileog:image and twitter:image URLs are filled in from local previewWhen helping the author brainstorm or choose a direction, these formats tend to perform well on the goose blog:
A complete blog post:
---
title: "Building a Custom MCP Server for Your Team"
description: "A step-by-step guide to creating a Model Context Protocol server that connects goose to your team's internal tools."
authors:
- ebony
---

If your team has internal tools that aren't covered by existing extensions, building a custom MCP server is easier than you think. In this post, I'll walk through how I built one for our team's deployment pipeline.
<!--truncate-->
## Why Build a Custom MCP Server?
goose connects to tools through the Model Context Protocol (MCP). While there are hundreds of community extensions available, sometimes your team has unique internal tools that need a custom integration.
## Getting Started
First, scaffold a new MCP server project:
```bash
npx create-mcp-server my-server
cd my-server
```
## Defining Your Tools
The core of any MCP server is its tool definitions...
## Wrapping Up
Building an MCP server took about an hour and saved our team countless context switches. If you want to learn more, check out the [MCP documentation](https://modelcontextprotocol.io/introduction) and the [goose extensions directory](https://block.github.io/goose/extensions).
<head>
<meta property="og:title" content="Building a Custom MCP Server for Your Team" />
<meta property="og:type" content="article" />
<meta property="og:url" content="https://block.github.io/goose/blog/2025/07/01/custom-mcp-server" />
<meta property="og:description" content="A step-by-step guide to creating a Model Context Protocol server that connects goose to your team's internal tools." />
<meta property="og:image" content="https://block.github.io/goose/assets/images/banner-a1b2c3d4e5f6.png" />
<meta name="twitter:card" content="summary_large_image" />
<meta property="twitter:domain" content="block.github.io" />
<meta name="twitter:title" content="Building a Custom MCP Server for Your Team" />
<meta name="twitter:description" content="A step-by-step guide to creating a Model Context Protocol server that connects goose to your team's internal tools." />
<meta name="twitter:image" content="https://block.github.io/goose/assets/images/banner-a1b2c3d4e5f6.png" />
</head>60bfdad
If you maintain this skill, you can claim it as your own. Once claimed, you can manage eval scenarios, bundle related skills, attach documentation or rules, and ensure cross-agent compatibility.