A multi-tenant ActivityPub server for Ghost, built with Fedify. This service makes it possible for independent websites to publish their content directly to the Fediverse, enabling networked publishing to the open social web.
All requests to /.ghost/activitypub/*, /.well-known/webfinger and /.well-known/nodeinfo are proxied to this ActivityPub service using nginx. All other requests are forwarded to Ghost. This setup has only been tested on macOS using Docker for Mac and OrbStack.
- Install Ghost using
Install from sourceinstructions - Expose your local port 80 (nginx) with Tailscale (or ngrok)
- Use
tailscale funnel 80orngrok http 80to expose your local port 80 (not 2368!), on which nginx will be running
- Use
- Configure Ghost
- In the Ghost monorepo, create a
config.local.jsonfile underghost/core/config.local.json, with the following configuration:
{ "url": <Tailscale or ngrok URL from step 2> } - In the Ghost monorepo, create a
- Start the ActivityPub Service
- Run
yarn dev && yarn logsin the root directory of this project
- Run
- Start Ghost
- Run
yarn devin the Ghost monorepo - If you were already running Ghost locally, make sure to restart it!
- Run
For AI assistants: See AGENTS.md for comprehensive guidance with code examples. For developers: See Architecture Decision Records in
/adrfor detailed rationale.
This service follows Domain-Driven Design with specific patterns:
- Immutable Entities with Events (ADR-0003) - Entities return new instances with domain events
- Result Type Pattern (ADR-0004) - Use Result<T, E> for explicit error handling
- Error Objects in Results (ADR-0005) - Enhanced Result types with contextual error objects
- Class-Based Architecture (ADR-0006) - All components use classes with dependency injection
- Repository Pattern (ADR-0007) - Services orchestrate logic, repositories handle data access
- View Pattern for Reads (ADR-0008) - Optimized read queries separate from write path
- Hash-Based Lookups (ADR-0009)
⚠️ - ActivityPub IDs use SHA256 hashes - Decorator Routing (ADR-0010) - Routes defined via decorators
src/
├── account/ # Immutable entities
├── post/ # Being migrated to immutable
├── activity-handlers/ # Class-based handlers
├── http/api/ # REST controllers
├── core/ # Shared utilities
└── dispatchers.ts # Legacy - don't add here
1. Database lookups MUST use SHA256 hashes (ADR-0009)
- Never use
where('ap_id', apId)- it returns empty results silently! - Always use
whereRaw('ap_id_hash = UNHEX(SHA2(?, 256))', [apId]) - Applies to:
ap_id,domain(with LOWER),ap_inbox_url(with LOWER)
2. Result types require helper functions
- Use
isError(result),getValue(result),getError(result) - Never destructure directly like
[error, value]
3. Services must use repositories
- Views can query DB directly (read optimization)
- Services MUST go through repositories (write path)
4. Dependency injection names must match
- Parameter
accountService→ registered as'accountService' - Parameter
db→ registered as'db'
5. Avoid these anti-patterns:
- Adding to
dispatchers.ts→ create new handler classes - Using
AccountType→ useAccountentity - Direct DB queries in services → use repositories
- String comparisons for AP IDs → use hash lookups
For complete code examples demonstrating correct patterns, see AGENTS.md.
We use Biome for code formatting and linting.
If you use VS Code, you can install the Biome extension to get inline feedback.
To enable auto-formatting on save, you'll need to set the default formatter to Biome and enable Format on Save in your VS Code settings.
- Run
yarn testto execute tests within a Docker Compose stack.
The below command will populate the DB with ~5000 followers for the activitypub host
- Run
docker compose run scripts populate-activitypub-db
docker compose run migrate or docker compose run migrate-testing will run the up migrations against your dev or testing db respectively.
If you would like to run other commands you can run docker compose exec -it migrate /bin/bash or docker compose exec -it migrate-testing /bin/bash - This will drop you into a shell with the migrate binary available as well as a MYSQL_DB environment variable that is correctly formated for use as the -database argument to the migrate binary
Copyright (c) 2013-2025 Ghost Foundation - Released under the MIT license. Ghost and the Ghost Logo are trademarks of Ghost Foundation Ltd. Please see our trademark policy for info on acceptable usage.