CLI tool for switching between Z.AI (GLM), MiniMax, and OpenCode API providers. Supports both Claude Code and OpenCode (experimental, future default). Built with oclif framework and Ink (React for CLI UI), providing multi-account management, MCP server control, and auto-rotation capabilities.
cohe - This project's CLI tool name (short for "coding helper"). When referring to the CLI command in this codebase, it's always written as cohe (e.g., cohe init, cohe account add). Do not confuse with Cohere (the AI company).
- Runtime: Bun
- CLI Framework: oclif with Ink (React for CLI)
- Language: TypeScript 5.9
- Validation: Zod 4
- Testing: Bun, happy-dom, React Testing Library
- SDK: @anthropic-ai/sdk, @anthropic-ai/claude-agent-sdk
- Linting: Biome via Ultracite preset
bun run dev: Start development server with HMRbun run build: Bundle CLI withbun buildbun run typecheck: Full TypeScript validationbun test: Run unit testsbun test:watch: Run tests in watch modebun run test:coverage: Run tests with coverage report
- Use functional components with hooks (no class components)
- Prefer composition over inheritance
- Use custom hooks for shared logic
- TypeScript strict mode enabled
- Biome for formatting and linting (
bun x ultracite fix) - Conventional commits (feat:, fix:, docs:, etc.)
- Feature-based folder structure under
src/commands/ - oclif flexible taxonomy with
.tsxcommand files - Provider pattern for API abstraction (
src/providers/) - Configuration with Zod schemas in
src/config/ - Ink React components for CLI UI
- Barrel exports from
index.tsfiles (disable linter for barrel files if needed)
- Write tests before implementing features (TDD)
- Focus on user behavior, not implementation details
- Mock external dependencies, test integration at boundaries
- Use happy-dom for React component testing
- Aim for meaningful coverage of core logic
- Avoid defaultProps in TypeScript components (use default parameters)
- Don't commit
.envfiles (use.env.exampleinstead) - Always handle loading and error states in CLI components
- Don't use
anyorunknowntypes without good reason - Don't use
--no-verifywhen committing (bypasses pre-commit hooks)
- Use
React.memofor expensive Ink components - Debounce user input prompts
- Minimize bundle size with
bun buildtree-shaking - Lazy load commands when possible
- Use React DevTools for Ink components
- Enable source maps in development
- Use VS Code debugger with
launch.jsonconfiguration - Log structured data for easier parsing
- Avoid adding new dependencies without consideration
- Prefer built-in Bun APIs over external packages
- Use Zod for validation instead of custom schemas
- UI components from
@inkjs/uiwhen possible
- All public functions should have JSDoc comments
- README files for each major feature/command
- Keep CLAUDE.md updated with project-specific conventions
Default to using Bun instead of Node.js.
- Use
bun <file>instead ofnode <file>orts-node <file> - Use
bun testinstead ofjestorvitest - Use
bun build <file.html|file.ts|file.css>instead ofwebpackoresbuild - Use
bun installinstead ofnpm installoryarn installorpnpm install - Use
bun run <script>instead ofnpm run <script>oryarn run <script>orpnpm run <script> - Use
bunx <package> <command>instead ofnpx <package> <command> - Bun automatically loads .env, so don't use dotenv.
Bun.serve()supports WebSockets, HTTPS, and routes. Don't useexpress.bun:sqlitefor SQLite. Don't usebetter-sqlite3.Bun.redisfor Redis. Don't useioredis.Bun.sqlfor Postgres. Don't usepgorpostgres.js.WebSocketis built-in. Don't usews.- Prefer
Bun.fileovernode:fs's readFile/writeFile - Bun.$
lsinstead of execa.
Use bun test to run tests.
import { test, expect } from "bun:test";
test("hello world", () => {
expect(1).toBe(1);
});Use HTML imports with Bun.serve(). Don't use vite. HTML imports fully support React, CSS, Tailwind.
Server:
import index from "./index.html"
Bun.serve({
routes: {
"/": index,
"/api/users/:id": {
GET: (req) => {
return new Response(JSON.stringify({ id: req.params.id }));
},
},
},
// optional websocket support
websocket: {
open: (ws) => {
ws.send("Hello, world!");
},
message: (ws, message) => {
ws.send(message);
},
close: (ws) => {
// handle close
}
},
development: {
hmr: true,
console: true,
}
})HTML files can import .tsx, .jsx or .js files directly and Bun's bundler will transpile & bundle automatically. <link> tags can point to stylesheets and Bun's CSS bundler will bundle.
<html>
<body>
<h1>Hello, world!</h1>
<script type="module" src="./frontend.tsx"></script>
</body>
</html>With the following frontend.tsx:
import React from "react";
import { createRoot } from "react-dom/client";
// import .css files directly and it works
import './index.css';
const root = createRoot(document.body);
export default function Frontend() {
return <h1>Hello, world!</h1>;
}
root.render(<Frontend />);Then, run index.ts
bun --hot ./index.tsFor more information, read the Bun API docs in node_modules/bun-types/docs/**.mdx.
This project uses Ultracite, a zero-config preset that enforces strict code quality standards through automated formatting and linting.
- Format code:
bun x ultracite fix - Check for issues:
bun x ultracite check - Diagnose setup:
bun x ultracite doctor
Biome (the underlying engine) provides robust linting and formatting. Most issues are automatically fixable.
Write code that is accessible, performant, type-safe, and maintainable. Focus on clarity and explicit intent over brevity.
- Use explicit types for function parameters and return values when they enhance clarity
- Prefer
unknownoveranywhen the type is genuinely unknown - Use const assertions (
as const) for immutable values and literal types - Leverage TypeScript's type narrowing instead of type assertions
- Use meaningful variable names instead of magic numbers - extract constants with descriptive names
- Use arrow functions for callbacks and short functions
- Prefer
for...ofloops over.forEach()and indexedforloops - Use optional chaining (
?.) and nullish coalescing (??) for safer property access - Prefer template literals over string concatenation
- Use destructuring for object and array assignments
- Use
constby default,letonly when reassignment is needed, nevervar
- Always
awaitpromises in async functions - don't forget to use the return value - Use
async/awaitsyntax instead of promise chains for better readability - Handle errors appropriately in async code with try-catch blocks
- Don't use async functions as Promise executors
- Use function components over class components
- Call hooks at the top level only, never conditionally
- Specify all dependencies in hook dependency arrays correctly
- Use the
keyprop for elements in iterables (prefer unique IDs over array indices) - Nest children between opening and closing tags instead of passing as props
- Don't define components inside other components
- Use semantic HTML and ARIA attributes for accessibility:
- Provide meaningful alt text for images
- Use proper heading hierarchy
- Add labels for form inputs
- Include keyboard event handlers alongside mouse events
- Use semantic elements (
<button>,<nav>, etc.) instead of divs with roles
- Remove
console.log,debugger, andalertstatements from production code - Throw
Errorobjects with descriptive messages, not strings or other values - Use
try-catchblocks meaningfully - don't catch errors just to rethrow them - Prefer early returns over nested conditionals for error cases
- Keep functions focused and under reasonable cognitive complexity limits
- Extract complex conditions into well-named boolean variables
- Use early returns to reduce nesting
- Prefer simple conditionals over nested ternary operators
- Group related code together and separate concerns
- Add
rel="noopener"when usingtarget="_blank"on links - Avoid
dangerouslySetInnerHTMLunless absolutely necessary - Don't use
eval()or assign directly todocument.cookie - Validate and sanitize user input
- Avoid spread syntax in accumulators within loops
- Use top-level regex literals instead of creating them in loops
- Prefer specific imports over namespace imports
- Avoid barrel files (index files that re-export everything)
- Use proper image components (e.g., Next.js
<Image>) over<img>tags
Next.js:
- Use Next.js
<Image>component for images - Use
next/heador App Router metadata API for head elements - Use Server Components for async data fetching instead of async Client Components
React 19+:
- Use ref as a prop instead of
React.forwardRef
Solid/Svelte/Vue/Qwik:
- Use
classandforattributes (notclassNameorhtmlFor)
- Write assertions inside
it()ortest()blocks - Avoid done callbacks in async tests - use async/await instead
- Don't use
.onlyor.skipin committed code - Keep test suites reasonably flat - avoid excessive
describenesting
Biome's linter will catch most issues automatically. Focus your attention on:
- Business logic correctness - Biome can't validate your algorithms
- Meaningful naming - Use descriptive names for functions, variables, and types
- Architecture decisions - Component structure, data flow, and API design
- Edge cases - Handle boundary conditions and error states
- User experience - Accessibility, performance, and usability considerations
- Documentation - Add comments for complex logic, but prefer self-documenting code
Most formatting and common issues are automatically fixed by Biome. Run bun x ultracite fix before committing to ensure compliance.