Integrate your Node.js application with Microsoft Flight Simulator (X, 2020, 2024) and Prepar3D.
node-simconnect implements the client side directly in TypeScript.node-simconnect is a community-developed, open-source project and is not an official Microsoft product. The SimConnect API evolves frequently with new Microsoft Flight Simulator releases, and this library aims to stay current with new features.
npm install node-simconnectThere are also auto generated API-docs.
You always start by calling open(...) which will attempt to open a connection with the SimConnect server (your flight simulator). If this succeeds you will get access to:
Example:
import { open, Protocol } from 'node-simconnect';
const EVENT_ID_PAUSE = 1;
open('My SimConnect client', Protocol.FSX_SP2)
.then(function ({ recvOpen, handle }) {
console.log('Connected to', recvOpen.applicationName);
// 1. Subscribe for pause event
handle.subscribeToSystemEvent(EVENT_ID_PAUSE, 'Pause');
// 2. Handle pause event
handle.on('event', function (recvEvent) {
switch (recvEvent.clientEventId) {
case EVENT_ID_PAUSE:
console.log(recvEvent.data === 1 ? 'Sim paused' : 'Sim unpaused');
break;
}
});
handle.on('exception', function (recvException) {
console.log(recvException);
});
handle.on('quit', function () {
console.log('Simulator quit');
});
handle.on('close', function () {
console.log('Connection closed unexpectedly (simulator CTD?)');
});
})
.catch(function (error) {
console.log('Connection failed:', error);
});
The protocol is forward compatible, so you can use the same protocol version for all Microsoft Flight Simulator releases. For example, Protocol.FSX_SP2 will work for MSFS 2020 and MSFS 2024, although it limits you to the features available in FSX SP2. See the table below for more details on protocol versions.
| Protocol | Description |
|---|---|
FSX_SP1 |
FSX Service Pack 1. Original protocol version for FSX. |
FSX_SP2 |
FSX Service Pack 2. Recommended for broadest compatibility. |
KittyHawk |
MSFS 2020 |
SunRise |
MSFS 2024 |
Most of the APIs described in the official SimConnect documentation are implemented in node-simconnect. For information on how each feature works, please refer to the official documentation. If you are missing any features in node-simconnect feel free to open a new issue or create a pull request.
For a complete list of available API methods, please refer to the SimConnectConnection class.
A major feature used by C/C++/C# implementation of SimConnect client libraries is the ability to directly cast a memory block to a user-defined structure. This is technically impossible to do in JavaScript or TypeScript since memory layout of classes and types are not accessible. Consequently, the wrapping/unwrapping steps must be done by the user.
Example using the official SimConnect SDK (C++):
// C++ code ////////////////////
struct Struct1 {
double kohlsmann;
double altitude;
double latitude;
double longitude;
int verticalSpeed;
};
// ....
hr = SimConnect_AddToDataDefinition(hSimConnect, DEFINITION_1, "Kohlsman setting hg", "inHg");
hr = SimConnect_AddToDataDefinition(hSimConnect, DEFINITION_1, "Indicated Altitude", "feet");
hr = SimConnect_AddToDataDefinition(hSimConnect, DEFINITION_1, "Plane Latitude", "degrees");
hr = SimConnect_AddToDataDefinition(hSimConnect, DEFINITION_1, "Plane Longitude", "degrees");
hr = SimConnect_AddToDataDefinition(hSimConnect, DEFINITION_1, "VERTICAL SPEED", "Feet per second", SimConnectDataType.INT32);
SimConnect_RequestDataOnSimObject(hSimConnect, REQUEST_1, DEFINITION_1, SIMCONNECT_OBJECT_ID_USER, SIMCONNECT_PERIOD_SECOND);
// ....
void CALLBACK MyDispatchProc(SIMCONNECT_RECV* pData, DWORD cbData) {
switch(pData->dwID) {
case SIMCONNECT_RECV_ID_SIMOBJECT_DATA: {
SIMCONNECT_RECV_SIMOBJECT_DATA *pObjData = (SIMCONNECT_RECV_SIMOBJECT_DATA*) pData;
switch(pObjData->dwRequestID) {
case REQUEST_1:
Struct1 *pS = (Struct1*)&pObjData->dwData;
break;
}
break;
}
}
}
The code below demonstrates how the same is achieved with node-simconnect:
// TypeScript code ////////////////////
const REQUEST_1 = 0;
const DEFINITION_1 = 0;
// ....
handle.addToDataDefinition(DEFINITION_1, 'Kohlsman setting hg', 'inHg', SimConnectDataType.FLOAT64);
handle.addToDataDefinition(DEFINITION_1, 'Indicated Altitude', 'feet', SimConnectDataType.FLOAT64);
handle.addToDataDefinition(DEFINITION_1, 'Plane Latitude', 'degrees', SimConnectDataType.FLOAT64);
handle.addToDataDefinition(DEFINITION_1, 'Plane Longitude', 'degrees', SimConnectDataType.FLOAT64);
handle.addToDataDefinition(DEFINITION_1, 'VERTICAL SPEED', 'Feet per second', SimConnectDataType.INT32);
handle.requestDataOnSimObject(REQUEST_1, DEFINITION_1, SimConnectConstants.OBJECT_ID_USER, SimConnectPeriod.SIM_FRAME);
// ....
handle.on('simObjectData', recvSimObjectData => {
switch (recvSimObjectData.requestID) {
case REQUEST_1: {
const receivedData = {
// Read order is important!
kohlsmann: recvSimObjectData.data.readFloat64(),
altitude: recvSimObjectData.data.readFloat64(),
latitude: recvSimObjectData.data.readFloat64(),
longitude: recvSimObjectData.data.readFloat64(),
verticalSpeed: recvSimObjectData.data.readInt32(),
}
break;
}
}
});
When the simObjectData callback is triggered, the recvSimObjectData.data is used to extract the requested simulation variables. These values are stored as a single continuous binary data chunk/buffer, maintaining the order in which the simulation variables were added to the data definition. In this case, the buffer is 288 bits long (64 + 64 + 64 + 64 + 32), or 36 bytes.
The read...() functions are used to extract each value individually. When the correct function is used, the reading "cursor" (offset) automatically moves after each read, positioning it at the beginning of the next value in the buffer. Consequently, it is crucial to ensure that the values are read in the same order and using the same data type as initially requested.
If the Node.js application runs on the same computer as the flight simulator you don't need to worry about this part.
To connect from an external computer you must configure SimConnect to accept connections from other hosts. This procedure is also described in the official docs, but here is the short version:
Open SimConnect.xml.
X:\Users\<USER>\AppData\Roaming\Microsoft\FSXX:\Users\<USER>\AppData\Local\Packages\Microsoft.FlightSimulator_**********\LocalCache.Set property <Address>0.0.0.0</Address>. Example of a working SimConnect.xml file:
<?xml version="1.0" encoding="Windows-1252"?>
<SimBase.Document Type="SimConnect" version="1,0">
<Filename>SimConnect.xml</Filename>
<SimConnect.Comm>
<Protocol>IPv4</Protocol>
<Scope>local</Scope>
<Port>5111</Port>
<MaxClients>64</MaxClients>
<MaxRecvSize>41088</MaxRecvSize>
<Address>0.0.0.0</Address>
</SimConnect.Comm>
</SimBase.Document>
Connecting from a remote script can be done by providing the IP address of the flight simulator PC and the port number when calling open:
const options = { remote: { host: 'localhost', port: 5111 } };
open('My SimConnect client', Protocol.FSX_SP2, options).then(/* ... */).catch(/* try again? */);
Note that if no connection options are specified, node-simconnect will auto-discover connection details in the following order:
SimConnect.cfg in the folder where Node.js is located. If the script is running in Electron, this will be the folder where the Electron executable is installed.SimConnect.cfg in the user's home directory (%USERPROFILE%, eg. C:\Users\<username>)localhost:<port>.This project is a port of the Java client library jsimconnect, originally written by lc0277. Details about the SimConnect protocol can be found on lc0277's old website. A huge thanks to everyone involved in that project! 🙏
node-simconnect was originally created to power a cross-platform virtual-airline flight tracker, and later grew into a general-purpose SimConnect library for Node.js.
The node-simconnect API mirrors the official SimConnect C/C++ SDK closely. This means developers familiar with the official SDK docs will recognize the same method names and patterns, and new SimConnect features are straightforward to implement. Note that an optional higher-level API is under development. In the meantime, if you want a more user-friendly API (including less manual data unwrapping), check out msfs-simconnect-api-wrapper, which supports some of the features.