| name | vscode-insiders-extension |
| description | Contains verified WSL-specific extension development workflows, webview CSP pitfall resolutions, and profile-based install fixes that produce more correct results than reasoning from general training alone. Targets code-insiders on WSL/Linux — not stable VS Code on Windows/macOS. Consult when building or debugging a VS Code extension under WSL, scaffolding with esbuild, fixing blank webview panels or silent CSP violations, troubleshooting extensions that won't load after install with profiles, or publishing to the marketplace with code-insiders. Do NOT use for stable VS Code, VS Code user configuration (themes, keybindings, tasks.json), Chrome/browser extensions, Electron apps, JupyterLab plugins, or Monaco Editor embeds.
|
VS Code Extension Development (WSL + Code Insiders)
Guide for building VS Code extensions under WSL using VS Code Insiders, from scaffold through
publication.
WSL + Code Insiders Environment
This skill targets VS Code Insiders (code-insiders) running with a WSL remote backend.
Every CLI invocation, launch config, README instruction, and documentation reference must use
code-insiders. If the user says "VS Code" or "code", they mean code-insiders.
WSL Remote (VS Code Insiders on Windows, extension host in WSL)
VS Code Insiders runs on Windows, the extension host runs inside WSL. Extensions split across
two locations — UI extensions (themes, keymaps) stay on Windows, workspace extensions run in WSL.
- CLI: Always
code-insiders, never code
- Server-side extensions (WSL):
~/.vscode-server-insiders/extensions/
- UI-side extensions (Windows):
%USERPROFILE%\.vscode-insiders\extensions\
(from WSL: /mnt/c/Users/<user>/.vscode-insiders/extensions/)
- Server-side profiles (WSL):
~/.vscode-server-insiders/data/User/profiles/<profile-id>/extensions.json
— CLI installs register in the global registry but NOT the active profile, requiring manual
registration
- Settings/keybindings (Windows):
%APPDATA%\Code - Insiders\User\profiles\<profile-id>\settings.json
and keybindings.json (from WSL: /mnt/c/Users/<user>/AppData/Roaming/Code - Insiders/User/profiles/<profile-id>/)
— the Windows side is authoritative for UI/editor settings (theme, font, keybindings,
layout). The WSL server-side profile at
~/.vscode-server-insiders/data/User/profiles/<profile-id>/ has only a minimal settings.json
— most settings are not duplicated there. Workspace-level overrides go in
.vscode/settings.json in the project dir on WSL.
- User data (Windows):
%APPDATA%\Code - Insiders\User\
(from WSL: /mnt/c/Users/<user>/AppData/Roaming/Code - Insiders/User/)
- File watchers: inotify-backed via VS Code's API — always prefer
createFileSystemWatcher()
over Node.js fs.watch()
- Path translation: Not needed for extension code — the extension host runs natively in WSL
- Install:
code-insiders --install-extension my-ext.vsix (runs inside WSL)
- Launch config: Use
${execPath} (resolves correctly); docs/comments reference code-insiders
process.env: The extension host does NOT have terminal-specific variables like
VSCODE_IPC_HOOK_CLI — read those from /proc/<terminal_pid>/environ instead
Quick Start — Scaffold a New Extension
Generate a TypeScript extension project:
npx --package yo --package generator-code -- yo code
Select "New Extension (TypeScript)" when prompted. This creates a ready-to-run project with
src/extension.ts, package.json, and TypeScript configuration.
Open the project and launch the Extension Development Host with F5 to test immediately.
After scaffolding, convert the build system from tsc to esbuild — see the esbuild section below.
The yo generator creates a tsc-based build by default, but esbuild bundles are smaller, faster,
and what VS Code itself uses internally (as of v1.110).
Project Structure
extension-name/
package.json # manifest: contribution points, activation, settings
esbuild.js # build script (replaces default tsc pipeline)
tsconfig.json # TypeScript config
.vscodeignore # files excluded from .vsix package
.vscode-test.mjs # test runner configuration
src/
extension.ts # activate() / deactivate() entry point
*.ts # feature modules
dist/
extension.js # bundled output (single file)
Keep src/ organized by feature. Each major capability (a TreeView provider, a webview, a
command group) gets its own module file. The extension.ts entry point wires them together
in activate() and tears them down in deactivate().
package.json — The Extension Manifest
The manifest declares everything the extension contributes to VS Code. Key fields:
{
"name": "my-extension",
"displayName": "My Extension",
"version": "0.1.0",
"engines": { "vscode": "^1.110.0" },
"main": "./dist/extension",
"activationEvents": [],
"contributes": { }
}
Activation Events
VS Code auto-generates activation events for most contribution points — you rarely need to
declare them manually. Notable exceptions:
onWebviewPanel:viewType — for webview panel restoration
onUri — for URI handlers
* — activate on startup (avoid unless truly needed)
For views (contributes.views), VS Code automatically activates on onView:viewId.
Contribution Points
The contributes object declares UI elements, commands, settings, and more:
{
"contributes": {
"commands": [{
"command": "myExt.refresh",
"title": "Refresh",
"icon": "$(refresh)",
"category": "My Extension"
}],
"views": {
"explorer": [{
"id": "myView",
"name": "My View"
}]
},
"viewsContainers": {
"activitybar": [{
"id": "myContainer",
"title": "My Container",
"icon": "media/icon.svg"
}],
"panel": [{
"id": "myPanel",
"title": "My Panel",
"icon": "media/icon.svg"
}]
},
"menus": {
"view/title": [{
"command": "myExt.refresh",
"when": "view == myView",
"group": "navigation"
}],
"view/item/context": [{
"command": "myExt.openItem",
"when": "view == myView && viewItem == editable",
"group": "inline"
}]
},
"viewsWelcome": [{
"view": "myView",
"contents": "No items found.\n[Get Started](command:myExt.init)"
}],
"configuration": {
"title": "My Extension",
"properties": {
"myExt.refreshInterval": {
"type": "number",
"default": 3000,
"description": "Refresh interval in milliseconds"
}
}
}
}
}
Menu when Clauses
Control visibility with boolean expressions:
view == myView — only in a specific view
viewItem == someContext — only for tree items with that contextValue
resourceScheme == file — only for file URIs
- Combine with
&&, ||, !
Menu Group Ordering
Use @ suffix for ordering within a group: "navigation@1", "navigation@2".
Extension Entry Point
import * as vscode from 'vscode';
export function activate(context: vscode.ExtensionContext) {
const disposable = vscode.commands.registerCommand('myExt.hello', () => {
vscode.window.showInformationMessage('Hello!');
});
context.subscriptions.push(disposable);
}
export function deactivate() {
}
Every disposable (commands, watchers, providers, views) should be pushed to
context.subscriptions so VS Code cleans them up on deactivation.
Core API Patterns
This section covers the most common patterns. For detailed API reference with full code examples,
read references/api-patterns.md.
TreeView
The most common way to add structured data to the sidebar or panel:
- Implement
TreeDataProvider<T> with getChildren() and getTreeItem()
- Fire
onDidChangeTreeData to refresh
- Register via
vscode.window.createTreeView() for full API access
TreeItems support labels, descriptions, tooltips, icons (ThemeIcon or custom SVG), collapsible
state, contextValue (for menu filtering), and a command to execute on click.
Webview
For custom HTML/CSS/JS interfaces when TreeView isn't sufficient:
- Create with
vscode.window.createWebviewPanel() or implement WebviewViewProvider
- Set HTML content, load local resources via
webview.asWebviewUri()
- Communicate via
postMessage() / onDidReceiveMessage()
- Always set a Content Security Policy
Webviews are sandboxed — they cannot access the filesystem directly. All communication goes
through the message passing API.
Dual View: TreeView + WebviewView Toggle
When you need richer presentation than TreeView allows (multi-line rows, colored status text,
custom layouts), add a WebviewView alongside the existing TreeView with a toolbar toggle. The
pattern involves:
- Register both views in the same
viewsContainers entry with when clauses on a context key
- Toggle commands flip the context key and persist to
globalState
- Both providers share a single data pipeline and refresh together
- User interactions in both views route through shared VS Code commands (not direct calls)
- CSP requires careful layering: nonce for scripts,
'unsafe-inline' for styles, and
${webview.cspSource} for external CSS
This pattern has several non-obvious failure modes (silent CSP blocking, cache invalidation
blanking the panel, stale focus handlers). For the complete implementation guide with all
pitfalls and fixes, read references/api-patterns.md §Dual View.
Terminal
For interacting with VS Code's integrated terminal:
window.createTerminal() — create terminals
terminal.show() — focus a terminal (handles tabs and split panes)
terminal.processId — get the underlying PID
window.terminals — enumerate all open terminals
onDidOpenTerminal / onDidCloseTerminal — lifecycle events
FileSystemWatcher
For reacting to file changes on disk. Prefer the VS Code API over Node.js fs.watch() because
it runs outside the editor process and uses efficient OS-level notifications (inotify on Linux):
const watcher = vscode.workspace.createFileSystemWatcher(
new vscode.RelativePattern(folder, '**/*.json')
);
watcher.onDidChange(uri => { });
watcher.onDidCreate(uri => { });
watcher.onDidDelete(uri => { });
context.subscriptions.push(watcher);
StatusBar
window.createStatusBarItem(alignment, priority) — left or right alignment
- Convention: left side = workspace-scoped info, right side = file-scoped info
- Set
text, tooltip, command, color, backgroundColor
Building with esbuild
esbuild is the recommended bundler (VS Code's own built-in extensions switched to it in v1.110).
It produces a single dist/extension.js file from the TypeScript source.
esbuild.js
const esbuild = require('esbuild');
const production = process.argv.includes('--production');
const watch = process.argv.includes('--watch');
const esbuildProblemMatcherPlugin = {
name: 'esbuild-problem-matcher',
setup(build) {
build.onStart(() => { console.log('[watch] build started'); });
build.onEnd(result => {
for (const { text, location } of result.errors) {
console.error(`✘ [ERROR] ${text}`);
if (location) {
console.error(` ${location.file}:${location.line}:${location.column}:`);
}
}
console.log('[watch] build finished');
});
},
};
async function main() {
const ctx = await esbuild.context({
entryPoints: ['src/extension.ts'],
bundle: true,
format: 'cjs',
minify: production,
sourcemap: !production,
sourcesContent: false,
platform: 'node',
outfile: 'dist/extension.js',
external: ['vscode'],
logLevel: 'warning',
plugins: [esbuildProblemMatcherPlugin],
});
if (watch) {
await ctx.watch();
} else {
await ctx.rebuild();
await ctx.dispose();
}
}
main().catch(e => { console.error(e); process.exit(1); });
npm Scripts
{
"scripts": {
"compile": "npm run check-types && node esbuild.js",
"check-types": "tsc --noEmit",
"watch": "npm-run-all -p watch:*",
"watch:esbuild": "node esbuild.js --watch",
"watch:tsc": "tsc --noEmit --watch --project tsconfig.json",
"package": "npm run check-types && node esbuild.js --production",
"vscode:prepublish": "npm run package",
"test": "vscode-test"
}
}
The watch script runs esbuild and tsc in parallel — esbuild handles bundling while tsc
provides type checking. This gives sub-second rebuilds with full type safety.
vscode:prepublish runs automatically during vsce package and vsce publish.
.vscodeignore
.vscode/
node_modules/
out/
src/
tsconfig.json
esbuild.js
.vscode-test.mjs
**/*.ts
!dist/**
Everything bundled into dist/extension.js can be excluded. This keeps the .vsix small.
Testing
For testing setup, configuration, and debugging details, read references/build-test-publish.md.
Quick setup:
npm install --save-dev @vscode/test-cli @vscode/test-electron
Tests run inside an Extension Development Host with full VS Code API access. Configure in
.vscode-test.mjs:
import { defineConfig } from '@vscode/test-cli';
export default defineConfig({
files: 'out/test/**/*.test.js',
mocha: { ui: 'tdd', timeout: 20000 }
});
Run: npm test or npx vscode-test --label unitTests
Publishing
For full publishing workflow, read references/build-test-publish.md.
Quick reference:
npm install -g @vscode/vsce
vsce package
vsce publish
vsce publish minor
vsce publish --pre-release
code-insiders --install-extension my-ext-0.1.0.vsix
Platform-specific builds: vsce package --target linux-x64 linux-arm64 darwin-x64
Stable vs Insiders — Same Extension
A .vsix package works on both VS Code stable and Insiders — there is no separate build. The
only factor is the engines.vscode version in package.json:
"^1.110.0" means the extension requires VS Code 1.110+. Insiders runs ~1 month ahead of
stable, so APIs available in Insiders may not be in stable yet. Use the lowest version that
supports the APIs you need if you want stable compatibility.
- The marketplace serves the same package to both builds.
- Local install just differs by CLI:
code-insiders --install-extension vs code --install-extension.
- When developing on Insiders, test against stable before publishing if you want broad reach.
WSL Platform Notes
Profile-Based Extension Registration (WSL Gotcha)
When VS Code profiles are active, code-insiders --install-extension registers the extension
in the global registry (~/.vscode-server-insiders/extensions/extensions.json) but NOT in the
active profile's registry (~/.vscode-server-insiders/data/User/profiles/<profile-id>/extensions.json).
The --profile flag does not fix this. The extension won't load until manually added to the
profile-specific extensions.json. Without a publisher field in package.json, the extension
installs as undefined_publisher.<name>-<version>.
Webview Content Security Policy
- Inline
onclick attributes are blocked by default CSP — use nonce-based <script> blocks
with addEventListener instead
style-src 'nonce-...' blocks inline style="" attributes — use
style-src ${webview.cspSource} 'unsafe-inline' for style-src, or move dynamic styling to
CSS classes with data-* attribute selectors
- Always include
${webview.cspSource} in style-src for CSS to load correctly
Architecture Constraints
- Extensions run in the Extension Host process, separate from the VS Code UI
- No DOM access — you cannot manipulate VS Code's UI directly
- No custom CSS injection — use contribution points and the API
- Webviews are sandboxed with their own security context
Reference Files
references/api-patterns.md — Detailed API patterns with full code examples for TreeView,
Webview, WebviewView, Terminal, FileSystemWatcher, and StatusBar. Read when implementing
any of these APIs.
references/build-test-publish.md — esbuild configuration rationale and additional options,
TypeScript config, test runner setup, debugging tests, CI configuration, publishing workflow,
and marketplace metadata. Read when setting up the build pipeline or preparing to publish.