This document describes all the message interfaces defined for WebSocket communication between the Second Life viewer and an external editor such as a VSCode extension.
- Usage Flow
- JSON-RPC Method Summary
- Session Management Interfaces
- Language and Syntax Interfaces
- Script Subscription Interfaces
- Compilation Interfaces
- Runtime Event Interfaces
- Handler and Configuration Interfaces
-
Connection Establishment:
- Viewer sends
session.handshakecall withSessionHandshakedata - Extension responds with
SessionHandshakeResponse - Viewer confirms with
session.oknotification
- Viewer sends
-
Language Information Exchange:
- Extension makes
language.syntax.idcall to get current syntax version - Extension makes
language.syntaxcalls with differentkindparameters to get specific language data - Viewer responds with a
LanguageInfoobject containing the requested definitions
- Extension makes
-
Script Subscription Management:
- Extension makes
script.subscribecall withScriptSubscribedata to request live synchronization for a script - Viewer responds with
ScriptSubscribeResponseindicating success or failure - When subscription needs to be terminated, viewer sends
script.unsubscribenotification withScriptUnsubscribedata - Extension handles unsubscription by cleaning up local script tracking
- Extension makes
-
Runtime Events:
- Viewer sends
language.syntax.changenotification withSyntaxChangewhen language changes - Viewer sends
script.compilednotification withCompilationResultafter script compilation - Viewer sends
runtime.debugnotification withRuntimeDebugfor debug messages during script execution - Viewer sends
runtime.errornotification withRuntimeErrorwhen runtime errors occur
- Viewer sends
-
Connection Termination:
- Either side can send
session.disconnectnotification withSessionDisconnectdata - Connection is closed gracefully
- Either side can send
| Method | Direction | Type | Interface/Parameters |
|---|---|---|---|
session.handshake |
Viewer → Extension | Call | SessionHandshake |
session.handshake (response) |
Extension → Viewer | Response | SessionHandshakeResponse |
session.ok |
Viewer → Extension | Notification | (no interface) |
session.disconnect |
Bidirectional | Notification | SessionDisconnect |
script.subscribe |
Extension → Viewer | Call | ScriptSubscribe |
script.subscribe (response) |
Viewer → Extension | Response | ScriptSubscribeResponse |
script.unsubscribe |
Viewer → Extension | Notification | ScriptUnsubscribe |
script.list |
Extension → Viewer | Call | (no parameters) |
script.list (response) |
Viewer → Extension | Response | ScriptList |
language.syntax.id |
Extension → Viewer | Call | (no parameters) |
language.syntax.id (response) |
Viewer → Extension | Response | { id: string } |
language.syntax |
Extension → Viewer | Call | { kind: string } |
language.syntax (response) |
Viewer → Extension | Response | LanguageInfo |
language.syntax.cache |
Extension → Viewer | Call | (no parameters) |
language.syntax.cache (response) |
Viewer → Extension | Response | SyntaxCacheList |
language.syntax.get |
Extension → Viewer | Call | { filename: string, as_json?: boolean } |
language.syntax.get (response) |
Viewer → Extension | Response | SyntaxCacheFile |
language.syntax.change |
Viewer → Extension | Notification | SyntaxChange |
script.compiled |
Viewer → Extension | Notification | CompilationResult |
runtime.debug |
Viewer → Extension | Notification | RuntimeDebug |
runtime.error |
Viewer → Extension | Notification | RuntimeError |
JSON-RPC Method: session.handshake (call from viewer)
The initial handshake call sent by the viewer to establish a session.
interface SessionHandshake {
server_version: "1.0.0";
protocol_version: "1.0";
viewer_name: string;
viewer_version: string;
agent_id: string;
agent_name: string;
challenge?: string;
languages: string[];
syntax_id: string;
features: { [feature: string]: boolean };
}Fields:
server_version: Fixed version "1.0.0" indicating the server API versionprotocol_version: Fixed version "1.0" for the communication protocolviewer_name: Name of the Second Life viewer applicationviewer_version: Version string of the vieweragent_id: Unique identifier for the user/agentagent_name: Human-readable name of the agentchallenge(optional): Path to a temporary file on the local filesystem containing a UUID. The client must read this file and return the UUID aschallenge_responseto authenticate the connection.languages: Array of supported scripting languages (e.g.,["lsl", "luau"])syntax_id: Current active syntax identifier as a UUID stringfeatures: Dictionary of feature flags indicating viewer capabilities. Known flags:live_sync: Viewer supports live script synchronisation with the external editorcompilation: Viewer will forward compilation results viascript.compiledsyntax_cache: Viewer supportslanguage.syntax.cacheandlanguage.syntax.getfor retrieving syntax definition files
JSON-RPC Method: Response to session.handshake
The response sent by the VS Code extension to complete the handshake.
interface SessionHandshakeResponse {
client_name: string;
client_version: "1.0";
protocol_version: string;
challenge_response?: string;
languages: string[];
features: { [feature: string]: boolean };
script_name?: string;
script_language?: string;
}Fields:
client_name: Name of the client (VS Code extension)client_version: Fixed version "1.0" of the clientprotocol_version: Protocol version the client supportschallenge_response(optional): The UUID read from the temporary file identified by thechallengefield in the handshake. Must be provided ifchallengewas present, otherwise the connection will be closed.languages: Array of languages supported by the clientfeatures: Dictionary of features supported by the clientscript_name(optional): Name of the script currently open in the editorscript_language(optional): Language of the script currently open in the editor (e.g."lsl","luau")
JSON-RPC Method: session.ok (notification from viewer)
Confirmation notification sent by the viewer after successful handshake completion. No parameters are sent with this notification.
JSON-RPC Method: session.disconnect (notification, bidirectional)
Message sent when terminating the connection.
interface SessionDisconnect {
reason: number;
message: string;
}Fields:
reason: Numeric code indicating the reason for disconnection:0: Normal closure1: Editor closed2: Protocol error3: Connection timeout4: Internal server error
message: Human-readable description of the disconnect reason
JSON-RPC Method: language.syntax.change (notification from viewer)
Notification sent when the active language syntax changes in the viewer.
interface SyntaxChange {
id: string;
}Fields:
id: UUID string identifying the new syntax version
JSON-RPC Method: language.syntax.id (call from extension to viewer)
Requests the current active language syntax identifier from the viewer. This method takes no parameters.
Response: Returns { id: string } where id is the current syntax version as a UUID string.
JSON-RPC Method: language.syntax (call from extension to viewer)
Requests the in-memory keyword definitions for a specific language. These definitions are the deserialized, viewer-processed form of the syntax data for the current region.
Parameters:
{
kind: string; // The language whose definitions to retrieve
}Valid kind values:
| Value | Description |
|---|---|
"defs.lsl" |
Returns the LSL keyword definitions |
"defs.lua" |
Returns the Luau keyword definitions |
Response:
interface LanguageInfo {
id: string;
defs?: object; // Present only on success
success: boolean;
error?: string; // Present only on failure
}Response Fields:
id: The current syntax version identifierdefs(optional): The keyword definitions object. Only present whensuccessistrue. Structure varies by language.success: Whether the definitions were found and returned successfullyerror(optional): Human-readable error description. Only present whensuccessisfalse
Error cases:
- No
kindparameter supplied:success: false,error: "No syntax category specified" - Unknown
kindvalue:success: false,error: "Unknown syntax category requested"
JSON-RPC Method: language.syntax.cache (call from extension to viewer)
Requests the list of file names currently held in the LLSyntaxDefCache. This provides the extension with the available syntax definition file names that can subsequently be retrieved with language.syntax.get. This method takes no parameters.
Response:
interface SyntaxCacheList {
files: string[]; // Array of file names (e.g. ["lsl_keywords.xml", "slua_definitions.yaml"])
success: boolean;
}Response Fields:
files: Array of file name strings, each of which can be passed as thefilenameparameter tolanguage.syntax.getsuccess: Whether the request was handled successfully
Known cache files:
| File name | Description |
|---|---|
builtins.txt |
LSL built-in keyword list in plain text format |
lsl_definitions.yaml |
LSL language definitions in YAML format |
lsl_keywords.xml |
LSL keyword definitions in LLSD XML format |
lsl_keywords_pretty.xml |
LSL keyword definitions in formatted LLSD XML format |
slua_default.d.luau |
Luau type definition file for editor tooling |
slua_default.docs.json |
Luau documentation data in JSON format |
slua_definitions.yaml |
Luau language definitions in YAML format |
slua_keywords.xml |
Luau keyword definitions in LLSD XML format |
slua_keywords_pretty.xml |
Luau keyword definitions in formatted LLSD XML format |
slua_selene.yml |
Luau Selene linter configuration in YAML format |
Not all files may be present in every cache — the actual list returned by language.syntax.cache reflects only what is available on the viewer's local filesystem at the time of the request.
JSON-RPC Method: language.syntax.get (call from extension to viewer)
Requests the content of a specific file from the syntax definition cache. The file name must be one of the names returned by a prior language.syntax.cache call. Content is returned either as a raw text string or as a parsed JSON/LLSD object depending on the as_json parameter.
Parameters:
{
filename: string; // The file name to retrieve, as returned by language.syntax.cache
as_json?: boolean; // Optional. If true, content is returned as a parsed object rather than raw text
}Fields:
filename: The file name to retrieve (e.g."lsl_keywords.xml","slua_definitions.yaml")as_json(optional): Whentrue, the file is deserialized and returned as a structured object incontent. When omitted orfalse,contentis the raw text of the file.
Response:
interface SyntaxCacheFile {
content?: string | object; // Present only on success. String if as_json is false/omitted, object if as_json is true
success: boolean;
error?: string; // Present only on failure
}Response Fields:
content: The file content. Only present whensuccessistrue. Is a raw text string whenas_jsonis omitted orfalse; is a parsed object whenas_jsonistrue.success: Whether the file was found and read successfullyerror(optional): Human-readable error description. Only present whensuccessisfalse
Error cases:
- No
filenameparameter supplied:success: false,error: "No filename specified" - Name not found in cache:
success: false,error: "Requested syntax cache file not found" - File could not be loaded:
success: false,error: "Failed to load syntax cache file"(or"Failed to load and format syntax cache file."whenas_jsonistrue)
JSON-RPC Method: script.subscribe (call from extension to viewer)
Requests subscription to a script for live synchronization between the editor and viewer.
interface ScriptSubscribe {
script_id: string;
script_name: string;
script_language: string;
}Fields:
script_id: Unique identifier for the script to subscribe toscript_name: Display name of the script filescript_language: Programming language of the script (e.g., "lsl", "luau")
JSON-RPC Method: Response to script.subscribe
Response from the viewer indicating whether script subscription was successful.
interface ScriptSubscribeResponse {
script_id: string;
success: boolean;
status: number;
object_id?: string;
item_id?: string;
message?: string;
}Fields:
script_id: The script identifier that was subscribed tosuccess: Whether the subscription was successfulstatus: Numeric status code indicating the result:0: Success1: Invalid editor — the script editor panel is no longer open2: Invalid subscription — no subscription found for the givenscript_id3: Already subscribed — another connection is already subscribed to this script4: Internal server error
object_id(optional): The in-world UUID of the object containing the scriptitem_id(optional): The inventory item UUID of the script within the objectmessage(optional): Additional information about the subscription result
JSON-RPC Method: script.unsubscribe (notification from viewer)
Notification sent by the viewer when a script subscription should be terminated.
interface ScriptUnsubscribe {
script_id: string;
}Fields:
script_id: Unique identifier for the script to unsubscribe from
JSON-RPC Method: script.list (call from extension to viewer)
Requests the list of all scripts currently open and tracked by the viewer, along with the viewer's temp directory. This is intended for use by a file watcher tool that needs to discover which script temp files are active without going through the full script.subscribe flow. This method takes no parameters.
Response:
interface ScriptList {
temp_dir: string;
script_ids: string[];
success: boolean;
}Response Fields:
temp_dir: The absolute path to the viewer's temp directory where live-sync script files are written. Combined with ascript_id, the caller can locate the corresponding temp file on disk.script_ids: Array of script ID strings for all currently subscribed scripts, across all active connections.success: Alwaystrue.
Individual compilation error record.
interface CompilationError {
row: number;
column: number;
level: string;
message: string;
format?: "lsl"; // Present only for LSL compilation errors
}Fields:
row: Line number where the error occurred (1-based for both LSL and Luau)column: Column position of the error (1-based for LSL; always0for Luau as the compiler does not provide column information)level: Compiler severity string (e.g."ERROR","WARNING")message: Error descriptionformat(optional): Present and set to"lsl"for LSL compilation errors; absent for Luau errors
JSON-RPC Method: script.compiled (notification from viewer)
Result of a compilation operation in the viewer.
interface CompilationResult {
script_id: string;
success: boolean;
running: boolean;
errors?: CompilationError[];
}Fields:
script_id: Unique identifier for the script that was compiledsuccess: Whether the compilation was successfulrunning: Whether the compiled script is currently runningerrors(optional): Array of compilation errors if any occurred
JSON-RPC Method: runtime.debug (notification from viewer)
Debug message notification sent by the viewer during script execution.
interface RuntimeDebug {
script_id: string;
object_id: string;
object_name: string;
message: string;
}Fields:
script_id: Unique identifier for the script generating the debug messageobject_id: Unique identifier for the object containing the scriptobject_name: Human-readable name of the objectmessage: The debug message content
JSON-RPC Method: runtime.error (notification from viewer)
Runtime error notification sent by the viewer when a script encounters an error during execution.
interface RuntimeError {
script_id: string;
object_id: string;
object_name: string;
message: string;
error: string;
line: number;
stack?: string[];
}Fields:
script_id: Unique identifier for the script that encountered the errorobject_id: Unique identifier for the object containing the scriptobject_name: Human-readable name of the objectmessage: The full raw chat text of the runtime error message as received from the simulatorerror: Extracted error description. Currently always an empty string — runtime error extraction from the simulator's multi-message format is not yet fully implemented.line: Line number where the error occurred. Currently always0for the same reason.stack(optional): Stack trace lines if they could be extracted from the error message
Event handler interface for WebSocket events.
interface WebSocketHandlers {
onHandshake?: (message: SessionHandshake) => SessionHandshakeResponse;
onHandshakeOk?: () => void;
onDisconnect?: (message: SessionDisconnect) => void;
onSubscribe?: (message: ScriptSubscribe) => ScriptSubscribeResponse;
onUnsubscribe?: (message: ScriptUnsubscribe) => void;
onSyntaxChange?: (message: SyntaxChange) => void;
onConnectionClosed?: () => void;
onCompilationResult?: (message: CompilationResult) => void;
onRuntimeDebug?: (message: RuntimeDebug) => void;
onRuntimeError?: (message: RuntimeError) => void;
}Methods:
onHandshake: Handler for initial handshake message, returns handshake responseonHandshakeOk: Handler called when handshake is successfully completedonDisconnect: Handler for disconnect notificationsonSubscribe: Handler called when the extension sends ascript.subscriberequest, returns subscription responseonUnsubscribe: Handler for script unsubscription notifications from vieweronSyntaxChange: Handler for syntax change notificationsonConnectionClosed: Handler called when connection is closedonCompilationResult: Handler for compilation result notificationsonRuntimeDebug: Handler for runtime debug message notificationsonRuntimeError: Handler for runtime error notifications
Client information used in handshake responses.
interface ClientInfo {
scriptName: string;
scriptId: string;
extension: string;
}Fields:
scriptName: Name of the script being editedscriptId: Unique identifier for the scriptextension: File extension or script type