원클릭으로
feature-switch
Feature switch system guide for gating new user-facing features behind feature flags
Codex 또는 Claude로 설치 이 Prompt를 복사해 Codex, Claude 또는 다른 어시스턴트에 붙여 넣으면 Skill 페이지를 검토하고 설치를 진행할 수 있습니다.
메뉴
Feature switch system guide for gating new user-facing features behind feature flags
Codex 또는 Claude로 설치 이 Prompt를 복사해 Codex, Claude 또는 다른 어시스턴트에 붙여 넣으면 Skill 페이지를 검토하고 설치를 진행할 수 있습니다.
SOC 직업 분류 기준
Patterns and best practices for using ccstate state management in the vm0 platform
Deep code review and quality analysis for vm0 project
Database migrations and Drizzle ORM guidelines for the vm0 project
Technical debt management - scan codebase for bad smells and create tracking issues
Comprehensive testing patterns and anti-patterns for writing and reviewing tests
Complete pre-commit workflow - run quality checks (format, lint, type, test) and validate/create conventional commit messages
| name | feature-switch |
| description | Feature switch system guide for gating new user-facing features behind feature flags |
This skill documents the feature switch system and provides step-by-step instructions for adding new feature switches. All new user-facing features must be gated behind a feature switch for gradual rollout.
A feature switch is required when adding:
A feature switch is not required for:
File: turbo/packages/core/src/feature-switch-key.ts
Add a new entry to FeatureSwitchKey:
export enum FeatureSwitchKey {
// ... existing keys
MyFeature = "myFeature",
}
File: turbo/packages/core/src/feature-switch.ts
Add an entry to the FEATURE_SWITCHES record:
[FeatureSwitchKey.MyFeature]: {
maintainer: "you@vm0.ai",
enabled: false,
enabledOrgIdHashes: STAFF_ORG_ID_HASHES, // optional: staff-only access
},
Configuration options:
| Field | Type | Description |
|---|---|---|
maintainer | string | Email of the responsible person |
enabled | boolean | true = on for everyone, false = off by default |
enabledUserHashes | string[] | FNV-1a hashes of allowed user IDs |
enabledEmailHashes | string[] | FNV-1a hashes of allowed emails |
enabledOrgIdHashes | string[] | FNV-1a hashes of allowed org IDs |
Common default states:
enabled: false — fully hidden until manually enabled via Lab pageenabled: false + enabledOrgIdHashes: STAFF_ORG_ID_HASHES — staff-only (most common for new features)enabled: true — on for everyone (use when feature is ready for GA)Choose the pattern that matches where your feature is consumed.
import { isFeatureEnabled, FeatureSwitchKey } from "@vm0/core";
// In route handler:
if (!isFeatureEnabled(FeatureSwitchKey.MyFeature, { userId, orgId })) {
return createErrorResponse("FORBIDDEN", "Feature not available");
}
import { FeatureSwitchKey } from "@vm0/core";
import { featureSwitch$ } from "../../signals/external/feature-switch.ts";
// In component:
const features = useLastResolved(featureSwitch$);
const showMyFeature = features?.[FeatureSwitchKey.MyFeature] ?? false;
// Conditional rendering:
{showMyFeature && <MyFeatureComponent />}
In turbo/apps/platform/src/views/zero-page/zero-sidebar.tsx, add a featureGate to the sidebar item:
{
id: "my-feature",
label: "My Feature",
icon: MyIcon,
featureGate: FeatureSwitchKey.MyFeature,
}
In turbo/packages/core/src/contracts/connectors.ts, add featureFlag to the connector config:
myConnector: {
label: "My Connector",
featureFlag: FeatureSwitchKey.MyConnector,
// ...
}
In turbo/apps/api/src/signals/auth/tokens.ts, add to CONDITIONAL_CAPABILITIES:
const CONDITIONAL_CAPABILITIES: ReadonlyMap<ZeroCapability, FeatureSwitchKey> =
new Map([
// ... existing entries
["my-feature:write", FeatureSwitchKey.MyFeature],
]);
| File | Role |
|---|---|
turbo/packages/core/src/feature-switch-key.ts | Enum of all feature switch keys |
turbo/packages/core/src/feature-switch.ts | Registry and evaluation logic |
turbo/apps/platform/src/signals/external/feature-switch.ts | Client-side reactive state with override layers |
turbo/apps/platform/src/views/zero-page/zero-sidebar.tsx | Sidebar nav items with featureGate |
turbo/packages/core/src/contracts/connectors.ts | Connector type definitions with featureFlag field |
turbo/apps/api/src/signals/auth/tokens.ts | Token capability gating |
Evaluation has two layers (lowest to highest priority):
userId / email / orgId hashes.user_feature_switches keyed by (orgId, userId). Written via the Lab page toggles or window._vm0.featureSwitches.myFeature = true (both call POST /api/zero/feature-switches). Cleared via the Lab page "Reset all" button (DELETE /api/zero/feature-switches).The same two-layer resolution applies on the server: route handlers that call isFeatureEnabled(..., { userId, orgId, overrides }) pass overrides loaded via loadFeatureSwitchOverrides(orgId, userId).
There is no client-only layer. window._vm0.featureSwitches requires auth and persists across refreshes; there is no device-local override.