cloudflare/polystella
TypeScript
Captured source
source ↗cloudflare/polystella
Description: Astro integration for localization and automated AI translation of content
Language: TypeScript
License: MIT
Stars: 2
Forks: 0
Open issues: 0
Created: 2026-05-18T15:03:06Z
Pushed: 2026-06-08T14:49:47Z
Default branch: main
Fork: no
Archived: no
README:
PolyStella
> ⚠️ Work in progress. PolyStella is in active development. APIs, configuration shapes, and internal behaviour may change without notice. Do not adopt for new projects yet.
PolyStella is an Astro integration that translates content into additional locales at build time using AI, caches translations in Cloudflare R2, and injects locale-prefixed routes for the translated pages.
What it does
- Build-time translation. Translates
.md,.mdx, and.tomlcontent into additional locales duringastro build. Visitors get static bytes; no runtime AI calls. - R2-cached. Translations are content-addressed by source bytes + glossary + model. Unchanged pages cost zero on rebuild. Translations are never committed to the repo.
- Glossary control. Per-locale YAML files pin do-not-translate terms, preferred translations, and free-form translator notes.
- Hand-translation overrides. Drop a file at
i18n/overrides/{locale}/and it wins over AI output verbatim. - Locale-prefixed routing. Ships its own route shims that locale-prefix pages via injected dynamic routes.
- UI-string maintenance. Per-locale JSON files for chrome text, with build-time drift detection and a CLI for sync + AI-fill.
Install
Install from npm:
pnpm add @cloudflare/polystella
Peer dependencies: astro ^6.0.0, optionally react ^17 || ^18 || ^19.
Quick start
Four files participate in a typical setup.
1. `astro.config.mjs` — register the integration. Locale set lives here.
import { defineConfig } from "astro/config";
import polystella from "@cloudflare/polystella";
import polystellaConfig from "./polystella.config.mjs";
export default defineConfig({
i18n: {
defaultLocale: "en-US",
locales: ["en-US", "pt-BR", "ja-JP"],
},
integrations: [polystella(polystellaConfig)],
});2. `polystella.config.mjs` — provider, glossary, R2, format-specific keys. Every option is documented in the configuration reference.
3. `src/content.config.ts` — register sibling collections so Astro's content layer picks up translations. Locale set is auto-derived from astro.config.mjs.
import { defineCollection } from "astro:content";
import { polystellaCollections } from "@cloudflare/polystella/content";
import { i18nLoader, i18nSchema } from "@cloudflare/polystella/i18n";
import { blog, authors } from "./content-schemas";
export const collections = {
...polystellaCollections({
source: { blog, authors },
}),
i18n: defineCollection({ loader: i18nLoader(), schema: i18nSchema() }),
};4. `src/env.d.ts` — pick up types for PolyStella's virtual modules:
///
Catalog-Only Usage
Projects that already handle localized content and routing can adopt only PolyStella's JSON catalog flow:
import catalogAstro from "@cloudflare/polystella/catalog/astro";
export default defineConfig({
i18n: { defaultLocale: "en-US", locales: ["en-US", "pt-BR"] },
integrations: [catalogAstro({ baseDir: "./src/i18n" })],
});This binds Astro.locals.t and Astro.locals.lhref only. It does not run content translation, route shims, R2 cache setup, or localized collection APIs.
Documentation
Full documentation lives at the Starlight docs site (under docs/ in this repo):
- Getting started — install, quick start, mental model
- Concepts — pipeline, cache, overrides, runtime bridge
- Configuration reference — every option
- CLI —
translate,check-ui,sync-ui,translate-ui - Runtime API —
Astro.locals, middleware, React hooks - Roadmap — shipped vs planned features
Contributing
See [CONTRIBUTING.md](./CONTRIBUTING.md). The agent-facing context is in [AGENTS.md](./AGENTS.md) and [ARCHITECTURE.md](./ARCHITECTURE.md).
License
[MIT](./LICENSE)
Notability
notability 2.0/10Low stars, routine new repo