A Grammarly alternative, where YOU own what you type on your keyboard.
SafeTyper is a browser extension that provides real-time grammar and spelling suggestions powered by AI models through OpenRouter, Groq, or a local Ollama instance.
- Universal Input Detection — works on any editable field across the web
- Multiple AI Providers — OpenRouter (GPT, Claude, Gemini, Llama), Groq (Llama, Gemma, Mixtral), and Ollama (local models)
- Smart Caching — reduces API calls and improves response time
- Visual Diff Display — inline highlighting of changes
- Adaptive Icon Positioning — smart placement for any input field size
- Multi-Browser — Chrome and Firefox with automated publishing
- Privacy-First — secure API key storage, rate limiting, key format validation
- Install dependencies:
bun install - (Optional) Create
.envfrom.env.examplewith your API key - Start dev server:
bun run dev(Chrome) orbun run dev:firefox(Firefox) - Load the extension in your browser
- Configure your API key in the extension popup (or it auto-loads from
.envin dev mode)
You'll need Bun (or Node.js 18+) and an API key from OpenRouter or Groq, or a local Ollama instance.
Copy .env.example to .env for dev mode:
VITE_OPENROUTER_API_KEY=sk-or-v1-...
VITE_GROQ_API_KEY=gsk_...Keys auto-load in development only. The .env file is git-ignored.
bun run dev # Chrome dev server
bun run dev:firefox # Firefox dev server
bun run build # Build for Chrome
bun run build:firefox # Build for Firefox
bun run zip # Chrome distribution zip
bun run zip:firefox # Firefox distribution zip
bun run check # Type checking
bun run lint # ESLint
bun run lint:fix # ESLint with auto-fix
bun run format # Prettier
bun run format:check # Check formattingsrc/
├── entrypoints/
│ ├── background.ts # Service worker (API key injection)
│ ├── content.ts # Content script (input detection)
│ └── popup/ # Extension popup UI (Svelte 5)
├── lib/content/
│ ├── api-client.ts # Provider-aware API client with caching
│ ├── config.ts # Provider configurations
│ ├── diff-engine.ts # Text diff implementation
│ ├── dom-utils.ts # DOM utilities & icon positioning
│ ├── state-manager.ts # State management
│ ├── types.ts # TypeScript definitions
│ └── ui-manager.ts # UI manager
├── styles/ # Content script CSS
└── assets/ # Fonts & images
SafeTyper supports Ollama as a local, private provider — no API key needed.
- Install Ollama and pull a model (e.g.
ollama pull llama3.1) - Configure
OLLAMA_ORIGINSto allow the browser extension to connect (see below) - In the SafeTyper popup, select Ollama as the provider and pick your model
By default, Ollama only accepts requests from localhost. Browser extensions use a different origin (chrome-extension://... or moz-extension://...), so Ollama will block requests unless you allow it.
Set OLLAMA_ORIGINS to * to allow all origins, or to your specific extension origin:
macOS:
launchctl setenv OLLAMA_ORIGINS "*"Then restart the Ollama application.
Linux (systemd):
sudo systemctl edit ollamaAdd under [Service]:
[Service]
Environment="OLLAMA_ORIGINS=*"Then reload and restart:
sudo systemctl daemon-reload
sudo systemctl restart ollamaWindows:
Set OLLAMA_ORIGINS as a system environment variable to *, then restart Ollama.
Running from the terminal (any OS):
OLLAMA_ORIGINS="*" ollama serveThe default endpoint is http://localhost:11434. If your Ollama instance runs on a different host or port, you can change it in the extension popup under the Ollama provider settings.
Automated publishing via GitHub Actions:
- Chrome Web Store — triggered on version tags (
v*.*.*) or manual dispatch - Firefox Add-ons (AMO) — triggered on version tags (
v*.*.*) or manual dispatch
Both workflows run lint, format check, and type checking before building. See .env.example for required secrets.
WXT · Svelte 5 · TypeScript · ESLint · Prettier
Apache 2.0 — see LICENSE