Back to Waveterm

WaveEnv Narrowing Skill

.kilocode/skills/waveenv/SKILL.md

0.14.56.5 KB
Original Source

WaveEnv Narrowing Skill

Purpose

A WaveEnv narrowing creates a named subset type of WaveEnv that:

  1. Documents exactly which parts of the environment a component tree actually uses.
  2. Forms a type contract so callers and tests know what to provide.
  3. Enables mocking in the preview/test server — you only need to implement what's listed.

When To Create One

Create a narrowing whenever you are writing a component (or group of components) that you want to test in the preview server, or when you want to make the environmental dependencies of a component tree explicit.

Core Principle: Only Include What You Use

Only list the fields, methods, atoms, and keys that the component tree actually accesses. If you don't call wos, don't include wos. If you only call one RPC command, only list that one command. The narrowing is a precise dependency declaration — not a copy of WaveEnv.

File Location

  • Separate file (preferred for shared/complex envs): name it <feature>env.ts next to the component, e.g. frontend/app/block/blockenv.ts.
  • Inline (acceptable for small, single-file components): export the type directly from the component file, e.g. WidgetsEnv in frontend/app/workspace/widgets.tsx.

Imports Required

ts
import {
  MetaKeyAtomFnType, // only if you use getBlockMetaKeyAtom or getTabMetaKeyAtom
  ConnConfigKeyAtomFnType, // only if you use getConnConfigKeyAtom
  SettingsKeyAtomFnType, // only if you use getSettingsKeyAtom
  WaveEnv,
  WaveEnvSubset,
} from "@/app/waveenv/waveenv";

The Shape

ts
export type MyEnv = WaveEnvSubset<{
  // --- Simple WaveEnv properties ---
  // Copy the type verbatim from WaveEnv with WaveEnv["key"] syntax.
  isDev: WaveEnv["isDev"];
  createBlock: WaveEnv["createBlock"];
  showContextMenu: WaveEnv["showContextMenu"];
  platform: WaveEnv["platform"];

  // --- electron: list only the methods you call ---
  electron: {
    openExternal: WaveEnv["electron"]["openExternal"];
  };

  // --- rpc: list only the commands you call ---
  rpc: {
    ActivityCommand: WaveEnv["rpc"]["ActivityCommand"];
    ConnEnsureCommand: WaveEnv["rpc"]["ConnEnsureCommand"];
  };

  // --- atoms: list only the atoms you read ---
  atoms: {
    modalOpen: WaveEnv["atoms"]["modalOpen"];
    fullConfigAtom: WaveEnv["atoms"]["fullConfigAtom"];
  };

  // --- wos: always take the whole thing, no sub-typing needed ---
  wos: WaveEnv["wos"];

  // --- services: list only the services you call; no method-level narrowing ---
  services: {
    block: WaveEnv["services"]["block"];
    workspace: WaveEnv["services"]["workspace"];
  };

  // --- key-parameterized atom factories: enumerate the keys you use ---
  getSettingsKeyAtom: SettingsKeyAtomFnType<"app:focusfollowscursor" | "window:magnifiedblockopacity">;
  getBlockMetaKeyAtom: MetaKeyAtomFnType<"view" | "frame:title" | "connection">;
  getTabMetaKeyAtom: MetaKeyAtomFnType<"tabid" | "name">;
  getConnConfigKeyAtom: ConnConfigKeyAtomFnType<"conn:wshenabled">;

  // --- other atom helpers: copy verbatim ---
  getConnStatusAtom: WaveEnv["getConnStatusAtom"];
  getLocalHostDisplayNameAtom: WaveEnv["getLocalHostDisplayNameAtom"];
  getConfigBackgroundAtom: WaveEnv["getConfigBackgroundAtom"];
}>;

Automatically Included Fields

Every WaveEnvSubset<T> automatically includes the mock fields — you never need to declare them:

  • isMock: boolean
  • mockSetWaveObj: <T extends WaveObj>(oref: string, obj: T) => void
  • mockModels?: Map<any, any>

Rules for Each Section

SectionPatternNotes
electronelectron: { method: WaveEnv["electron"]["method"]; }List every method called; omit the rest.
rpcrpc: { Cmd: WaveEnv["rpc"]["Cmd"]; }List every RPC command called; omit the rest.
atomsatoms: { atom: WaveEnv["atoms"]["atom"]; }List every atom read; omit the rest.
woswos: WaveEnv["wos"]Take the whole wos object (no sub-typing needed), but only add it if wos is actually used.
servicesservices: { svc: WaveEnv["services"]["svc"]; }List each service used; take the whole service object (no method-level narrowing).
getSettingsKeyAtomSettingsKeyAtomFnType<"key1" | "key2">Union all settings keys accessed.
getBlockMetaKeyAtomMetaKeyAtomFnType<"key1" | "key2">Union all block meta keys accessed.
getTabMetaKeyAtomMetaKeyAtomFnType<"key1" | "key2">Union all tab meta keys accessed.
getConnConfigKeyAtomConnConfigKeyAtomFnType<"key1">Union all conn config keys accessed.
All other WaveEnv fieldsWaveEnv["fieldName"]Copy type verbatim.

Using the Narrowed Type in Components

ts
import { useWaveEnv } from "@/app/waveenv/waveenv";
import { MyEnv } from "./myenv";

const MyComponent = memo(() => {
    const env = useWaveEnv<MyEnv>();
    // TypeScript now enforces you only access what's in MyEnv.
    const val = useAtomValue(env.getSettingsKeyAtom("app:focusfollowscursor"));
    ...
});

The generic parameter on useWaveEnv<MyEnv>() casts the context to your narrowed type. The real production WaveEnv satisfies every narrowing; mock envs only need to implement the listed subset.

Real Examples

  • BlockEnv in frontend/app/block/blockenv.ts — complex narrowing with all section types, in a separate file.
  • WidgetsEnv in frontend/app/workspace/widgets.tsx — smaller narrowing defined inline in the component file.