RepoCloudflare (Workers AI)Cloudflare (Workers AI)published May 18, 2026seen 5d

cloudflare/polystella

TypeScript

Open original ↗

Captured source

source ↗
published May 18, 2026seen 5dcaptured 9hhttp 200method plain

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 .toml content into additional locales during astro 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):

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/10

Low stars, routine new repo