skills/coding-standards/SKILL.md
Universal coding standards, best practices, and patterns for TypeScript, JavaScript, React, and Node.js development.
npx skillsauth add fmflurry/settings-opencode coding-standardsInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Universal coding standards applicable across all projects.
// ✅ GOOD: Descriptive names
const marketSearchQuery = "election";
const isUserAuthenticated = true;
const totalRevenue = 1000;
// ❌ BAD: Unclear names
const q = "election";
const flag = true;
const x = 1000;
// ✅ GOOD: Fluent naming
private readonly searchCustomers = inject(SearchCustomersUseCase);
public searchCustomersBy(filters: SearchFilters) {
return this.searchCustomers.by(filters); // ✅ GOOD - fluent
}
// ❌ BAD: Unclear names
public searchCustomers(searchFilters: SearchFilters) {
return this.searchCustomers.search(searchFilters); // ❌ BAD - can't read it as fluent
}
// ✅ GOOD: Verb-noun pattern
async function fetchMarketData(marketId: string) {}
function calculateSimilarity(a: number[], b: number[]) {}
function isValidEmail(email: string): boolean {}
// ❌ BAD: Unclear or noun-only
async function market(id: string) {}
function similarity(a, b) {}
function email(e) {}
// ✅ ALWAYS use spread operator
const updatedUser = {
...user,
name: "New Name",
};
const updatedArray = [...items, newItem];
// ❌ NEVER mutate directly
user.name = "New Name"; // BAD
items.push(newItem); // BAD
// ✅ GOOD: Comprehensive error handling
async function fetchData(url: string) {
try {
const response = await fetch(url);
if (!response.ok) {
throw new Error(`HTTP ${response.status}: ${response.statusText}`);
}
return await response.json();
} catch (error) {
console.error("Fetch failed:", error);
throw new Error("Failed to fetch data");
}
}
// ❌ BAD: No error handling
async function fetchData(url) {
const response = await fetch(url);
return response.json();
}
// ✅ GOOD: Parallel execution when possible
const [users, markets, stats] = await Promise.all([
fetchUsers(),
fetchMarkets(),
fetchStats(),
]);
// ❌ BAD: Sequential when unnecessary
const users = await fetchUsers();
const markets = await fetchMarkets();
const stats = await fetchStats();
// ✅ GOOD: Proper types
interface Market {
id: string;
name: string;
status: "active" | "resolved" | "closed";
created_at: Date;
}
function getMarket(id: string): Promise<Market> {
// Implementation
}
// ❌ BAD: Using 'any'
function getMarket(id: any): Promise<any> {
// Implementation
}
GET /api/markets # List all markets
GET /api/markets/:id # Get specific market
POST /api/markets # Create new market
PUT /api/markets/:id # Update market (full)
PATCH /api/markets/:id # Update market (partial)
DELETE /api/markets/:id # Delete market
# Query parameters for filtering
GET /api/markets?status=active&limit=10&offset=0
See Codemaps in DOCS/CODEMAPS
components/Button.tsx # PascalCase for components
hooks/useAuth.ts # camelCase with 'use' prefix
lib/formatDate.ts # camelCase for utilities
types/market.types.ts # camelCase with .types suffix
// ✅ GOOD: Explain WHY, not WHAT
// Use exponential backoff to avoid overwhelming the API during outages
const delay = Math.min(1000 * Math.pow(2, retryCount), 30000);
// Deliberately using mutation here for performance with large arrays
items.push(newItem);
// ❌ BAD: Stating the obvious
// Increment counter by 1
count++;
// Set name to user's name
name = user.name;
/**
* Searches markets using semantic similarity.
*
* @param query - Natural language search query
* @param limit - Maximum number of results (default: 10)
* @returns Array of markets sorted by similarity score
* @throws {Error} If OpenAI API fails or Redis unavailable
*
* @example
* ```typescript
* const results = await searchMarkets('election', 5)
* console.log(results[0].name) // "Trump vs Biden"
* ```
*/
export async function searchMarkets(
query: string,
limit: number = 10,
): Promise<Market[]> {
// Implementation
}
import { useMemo, useCallback } from "react";
// ✅ GOOD: Memoize expensive computations
const sortedMarkets = useMemo(() => {
return markets.sort((a, b) => b.volume - a.volume);
}, [markets]);
// ✅ GOOD: Memoize callbacks
const handleSearch = useCallback((query: string) => {
setSearchQuery(query);
}, []);
import { lazy, Suspense } from 'react'
// ✅ GOOD: Lazy load heavy components
const HeavyChart = lazy(() => import('./HeavyChart'))
export function Dashboard() {
return (
<Suspense fallback={<Spinner />}>
<HeavyChart />
</Suspense>
)
}
// ✅ GOOD: Select only needed columns
const { data } = await supabase
.from("markets")
.select("id, name, status")
.limit(10);
// ❌ BAD: Select everything
const { data } = await supabase.from("markets").select("*");
test("calculates similarity correctly", () => {
// Arrange
const vector1 = [1, 0, 0];
const vector2 = [0, 1, 0];
// Act
const similarity = calculateCosineSimilarity(vector1, vector2);
// Assert
expect(similarity).toBe(0);
});
// ✅ GOOD: Descriptive test names
test("returns empty array when no markets match query", () => {});
test("throws error when OpenAI API key is missing", () => {});
test("falls back to substring search when Redis unavailable", () => {});
// ❌ BAD: Vague test names
test("works", () => {});
test("test search", () => {});
Watch for these anti-patterns:
// ❌ BAD: Function > 50 lines
function processMarketData() {
// 100 lines of code
}
// ✅ GOOD: Split into smaller functions
function processMarketData() {
const validated = validateData();
const transformed = transformData(validated);
return saveData(transformed);
}
// ❌ BAD: 5+ levels of nesting
if (user) {
if (user.isAdmin) {
if (market) {
if (market.isActive) {
if (hasPermission) {
// Do something
}
}
}
}
}
// ✅ GOOD: Early returns
if (!user) return;
if (!user.isAdmin) return;
if (!market) return;
if (!market.isActive) return;
if (!hasPermission) return;
// Do something
// ❌ BAD: Unexplained numbers
if (retryCount > 3) {
}
setTimeout(callback, 500);
// ✅ GOOD: Named constants
const MAX_RETRIES = 3;
const DEBOUNCE_DELAY_MS = 500;
if (retryCount > MAX_RETRIES) {
}
setTimeout(callback, DEBOUNCE_DELAY_MS);
Remember: Code quality is not negotiable. Clear, maintainable code enables rapid development and confident refactoring.
development
Scaffolds and extends Angular 18+ standalone features using Clean Architecture with DDD layering (presentation/application/domain/infrastructure), custom signal-based stores, facade pattern, and ports/adapters dependency inversion. Use when creating new Angular features/domains, adding use cases/facades/stores/ports/adapters, refactoring legacy NgModule/NgRx code toward clean architecture, or working with cross-domain communication via context registry.
development
Pre-merge code review for Angular + TypeScript pull requests. Diffs current branch against a target branch, applies Angular-specific checklists (signals, RxJS, clean architecture, flurryx, TS strict), runs lint + tsc, and emits a tiered report (verbose for juniors, terse for seniors). Auto-loads project AGENTS.md rules. Use when user runs /cop-review, says "pre-merge review", "review before merging", "check my PR against <branch>", or invokes the merge-cop agent.
testing
Use this skill for any git work such as creating branches, staging changes, writing commit messages, pushing branches, or preparing pull requests. Delegates git execution to the git-specialist agent.
development
Signal-first reactive state management for Angular. Bridge RxJS streams into cache-aware stores, keyed resources, mirrored state, and replayable history. Use when generating or modifying Angular code that uses flurryx for state management, or when scaffolding new feature modules that follow the flurryx facade pattern.