NMEA 2000 encoder and decoder
Project description
NMEA 2000 Python Library
A Python library for encoding and decoding NMEA 2000 frames. The encoding and decoding is based on the extensive canboat database. It also supports inexpensive CANBUS USB and TCP devices as gateways between your NMEA 2000 boat network and any Python code that wants to receive or send these messages. This package is the backend for the Home Assistant NMEA 2000 Integration.
Features
- Decode NMEA 2000 frames: Parse and interpret raw NMEA 2000 data.
- Encode NMEA 2000 frames: Convert structured data back into the NMEA 2000 frame format.
- Gateway clients: Send and receive NMEA 2000 data through various hardware gateways:
- EByte — binary TCP gateways like ECAN-E01 and ECAN-W01S
- Text — any line-based ASCII TCP gateway with auto-sensing or explicit format selection (e.g. Actisense W2K-1, Yacht Devices YDEN-02, Actisense PRO-NDC-1E2K in CAN ASCII mode)
- Actisense BST — Actisense devices using the BST binary protocol over TCP, supporting both BST-95 (raw CAN frames) and BST-D0 (pre-assembled N2K). Compatible with the PRO-NDC-1E2K and W2K-1 in CAN Actisense mode
- WaveShare — USB serial devices like Waveshare USB-CAN-A
- python-can — any generic USB or SocketCAN device supported by the python-can library
- PGN-specific parsing: Handle various PGNs with specific parsing rules based on canboat.
- Stateful decoder: The decoder supports NMEA 2000 fast messages, which are split across multiple CANBUS messages.
- CLI support: Built-in command-line interface for encoding and decoding frames.
Installation
You can install the library using pip:
pip install nmea2000
Alternatively, you can clone the repository and install it locally:
git clone https://github.com/tomer-w/nmea2000.git
cd nmea2000
pip install .
Usage
Decode NMEA 2000 Frame (CLI)
To decode a frame, use the decode command followed by the frame in Actisense hex format:
nmea2000-cli decode --frame "09FF7 0FF00 3F9FDCFFFFFFFFFF"
65280 Furuno: Heave: Manufacturer Code = Furuno (bytes = "3F 07"), Reserved = 3 (bytes = "03"), Industry Code = Marine (bytes = "04"), Heave = -0.036000000000000004 (bytes = "DC"), Reserved = 65535 (bytes = "FF FF 00")
Or in JSON format:
{"PGN":65280,"id":"furunoHeave","description":"Furuno: Heave","fields":[{"id":"manufacturer_code","name":"Manufacturer Code","description":"Furuno","unit_of_measurement":"","value":"Furuno","raw_value":1855},{"id":"reserved_11","name":"Reserved","description":"","unit_of_measurement":"","value":3,"raw_value":3},{"id":"industry_code","name":"Industry Code","description":"Marine Industry","unit_of_measurement":"","value":"Marine","raw_value":4},{"id":"heave","name":"Heave","description":"","unit_of_measurement":"m","value":-0.036000000000000004,"raw_value":-36},{"id":"reserved_48","name":"Reserved","description":"","unit_of_measurement":"","value":65535,"raw_value":65535}],"source":9,"destination":255,"priority":7}
Example Code
from nmea2000.decoder import NMEA2000Decoder
# Initialize decoder
decoder = NMEA2000Decoder()
# Decode a frame
frame_str = "09FF7 0FF00 3F9FDCFFFFFFFFFF"
decoded_frame = decoder.decode(frame_str)
# Print decoded frame
print(decoded_frame)
Repeating Fields
Some PGNs (e.g. AC Input/Output Status) contain repeating field sets — for example, one set of measurements per AC line. These are exposed as a list field whose value is a list of dicts, where each dict maps field IDs to NMEA2000Field objects:
from nmea2000.decoder import NMEA2000Decoder
from nmea2000.consts import FieldTypes
decoder = NMEA2000Decoder(already_combined=True)
msg = decoder.decode(
"2021-07-29-09:00:42.386,6,127503,1,255,20,"
"00,01,f0,00,05,2c,01,88,13,e8,03,80,15,00,00,80,15,00,00,64"
)
for field in msg.fields:
if field.type == FieldTypes.VARIABLE and isinstance(field.value, list):
for idx, entry in enumerate(field.value):
print(f"AC Line {idx}:")
for field_id, sub_field in entry.items():
print(f" {sub_field.name}: {sub_field.value} {sub_field.unit_of_measurement or ''}")
Output:
AC Line 0:
Line: Line 1
Acceptability: Bad level
Reserved: 15
Voltage: 12.8 V
Current: 30.0 A
Frequency: 50.0 Hz
Breaker Size: 100.0 A
Real Power: 5504 W
Reactive Power: 5504 VAR
Power factor: 1.0 Cos Phi
Example reading packets using python-can
import can
from nmea2000.decoder import NMEA2000Decoder
# Initialize decoder
decoder = NMEA2000Decoder()
# Connect to CAN bus (e.g. slcan device on /dev/ttyUSB0)
bus = can.interface.Bus(interface='slcan', channel="/dev/ttyUSB0", bitrate=250000)
# Decode frames
for msg in bus:
decoded_frame = decoder.decode(msg)
# Print decoded frame when ready (fast data intermediate frames return None)
if decoded_frame is not None:
print(decoded_frame)
Simple N2KDevice example
If you want to behave like a small NMEA 2000 device instead of just reading frames, N2KDevice wraps the transport client, handles address claiming, and lets you send/receive NMEA2000Message objects directly:
import asyncio
from nmea2000.device import N2KDevice
from nmea2000.message import NMEA2000Field, NMEA2000Message
async def handle_received(message: NMEA2000Message) -> None:
print(f"received PGN {message.PGN} from source {message.source}")
async def main() -> None:
device = N2KDevice.for_python_can(
interface="socketcan",
channel="can0",
preferred_address=25,
model_id="Python demo device",
manufacturer_information="nmea2000 README example",
)
device.set_receive_callback(handle_received)
try:
await device.start()
await device.wait_ready(timeout=5)
await device.send(
NMEA2000Message(
PGN=127250,
id="vesselHeading",
priority=2,
source=0, # 0 means "use the address claimed by this device"
destination=255,
fields=[
NMEA2000Field(id="sid", raw_value=0),
NMEA2000Field(id="heading", value=1.0),
NMEA2000Field(id="deviation", raw_value=0),
NMEA2000Field(id="variation", raw_value=0),
NMEA2000Field(id="reference", raw_value=0),
NMEA2000Field(id="reserved_58", raw_value=0),
],
)
)
await asyncio.sleep(10)
finally:
await device.close()
asyncio.run(main())
Use N2KDevice.for_ebyte(...), N2KDevice.for_yacht_devices(...), N2KDevice.for_waveshare(...), or N2KDevice.for_actisense(...) if you are connecting through one of those gateways instead of python-can.
Gateway Client CLI
Each gateway type has its own subcommand:
# EByte binary TCP gateway
nmea2000-cli ebyte --server 192.168.0.46 --port 8881
# Text/line-based TCP gateway with auto-sensing (W2K-1, YDEN-02, PRO-NDC-1E2K in CAN ASCII mode)
nmea2000-cli text --server 192.168.0.46 --port 8881
# Text gateway with explicit format
nmea2000-cli text --server 192.168.0.46 --port 8881 --format N2K_ASCII_RAW
# Actisense BST over TCP (PRO-NDC-1E2K / W2K-1 in CAN Actisense mode)
nmea2000-cli actisense_bst --server 192.168.0.46 --port 8881
# WaveShare USB-CAN-A serial
nmea2000-cli waveshare --port /dev/ttyUSB0
# Generic python-can adapter
nmea2000-cli can --interface slcan --channel /dev/ttyUSB0 --bitrate 250000
Use --json to output received messages as JSON (one object per line), useful for piping into other tools:
nmea2000-cli text --server 192.168.0.46 --port 8881 --json
The --json flag is available on all gateway subcommands. Use --dump_file to record raw frames to a file.
Gateway Client code
async def handle_received_data(message: NMEA2000Message):
"""User-defined callback function for received data."""
print(f"Callback: Received {message}")
client = EByteNmea2000Gateway(ip, port)
client.set_receive_callback(handle_received_data) # Register callback
Encode NMEA 2000 Frame (CLI)
You can also encode data into NMEA 2000 frames using the encode command:
nmea2000-cli encode --data "your_data_to_encode"
Example:
nmea2000-cli encode --data '{"PGN":65280,"id":"furunoHeave","description":"Furuno: Heave","fields":[{"id":"manufacturer_code","name":"Manufacturer Code","description":"Furuno","unit_of_measurement":"","value":"Furuno","raw_value":1855},{"id":"reserved_11","name":"Reserved","description":"","unit_of_measurement":"","value":3,"raw_value":3},{"id":"industry_code","name":"Industry Code","description":"Marine Industry","unit_of_measurement":"","value":"Marine","raw_value":4},{"id":"heave","name":"Heave","description":"","unit_of_measurement":"m","value":-0.036000000000000004,"raw_value":-36},{"id":"reserved_48","name":"Reserved","description":"","unit_of_measurement":"","value":65535,"raw_value":65535}],"source":9,"destination":255,"priority":7}'
Encoding frame: {"PGN":65280,"id":"furunoHeave","description":"Furuno: Heave","fields":[{"id":"manufacturer_code","name":"Manufacturer Code","description":"Furuno","unit_of_measurement":"","value":"Furuno","raw_value":1855},{"id":"reserved_11","name":"Reserved","description":"","unit_of_measurement":"","value":3,"raw_value":3},{"id":"industry_code","name":"Industry Code","description":"Marine Industry","unit_of_measurement":"","value":"Marine","raw_value":4},{"id":"heave","name":"Heave","description":"","unit_of_measurement":"m","value":-0.036000000000000004,"raw_value":-36},{"id":"reserved_48","name":"Reserved","description":"","unit_of_measurement":"","value":65535,"raw_value":65535}],"source":9,"destination":255,"priority":7}'
output:
09FF7 0FF00 3F9FDCFFFFFFFFFF
Example Code
from nmea2000.encoder import NMEA2000Encoder
from nmea2000.input_formats import N2KFormat
# Initialize encoder
encoder = NMEA2000Encoder(output_format=N2KFormat.TCP)
# Data to encode: vessel heading message (PGN 127250)
message = NMEA2000Message(
PGN=127250,
priority=2,
source=1,
destination=255,
fields=[
NMEA2000Field(
id="sid",
raw_value=0,
),
NMEA2000Field(
id="heading",
value=1, # 1 radian is 57 degrees
),
NMEA2000Field(
id="deviation",
raw_value=0,
),
NMEA2000Field(
id="variation",
raw_value=0,
),
NMEA2000Field(
id="reference",
raw_value=0,
),
NMEA2000Field(
id="reserved_58",
raw_value=0,
)
]
)
msg_bytes = encoder.encode(_generate_test_message())
print(msg_bytes)
Node-RED Integration
You can stream decoded NMEA 2000 data into Node-RED using the CLI with the --json flag and a Node-RED exec node.
Setup
-
Exec node — add an
execnode and configure:- Command:
nmea2000-cli ebyte --server 192.168.1.100 --port 8881 --json - Output: select "when stdout has data" so it emits a message for each line (not on process exit)
- Use spawn mode: enable
Use spawn() instead of exec() - Timeout: leave blank (this is a long-running process)
- Append msg.payload: uncheck
- Command:
-
JSON node — connect the exec node's first output (stdout) to a
jsonnode to parse each line into a JavaScript object. -
Process the data — use a
switchorfunctionnode to route by PGN:// Example function node: add topic by PGN msg.topic = "nmea2000/pgn/" + msg.payload.PGN; return msg;
Importable Flow
Copy and import this JSON into Node-RED (Menu → Import → Clipboard):
[
{
"id": "nmea2000_exec",
"type": "exec",
"name": "NMEA2000 Stream",
"command": "nmea2000-cli ebyte --server 192.168.1.100 --port 8881 --json",
"addpay": "",
"append": "",
"useSpawn": "true",
"oldrc": false,
"timer": "",
"wires": [["nmea2000_json"], [], []]
},
{
"id": "nmea2000_json",
"type": "json",
"name": "Parse JSON",
"property": "payload",
"wires": [["nmea2000_debug"]]
},
{
"id": "nmea2000_debug",
"type": "debug",
"name": "NMEA2000 Data",
"active": true,
"tosidebar": true,
"wires": []
}
]
Each received NMEA 2000 message arrives as a parsed JSON object:
{
"PGN": 127250,
"id": "vesselHeading",
"description": "Vessel Heading",
"source": 3,
"destination": 255,
"priority": 2,
"fields": [
{
"id": "heading",
"name": "Heading",
"value": 182.5,
"unit_of_measurement": "deg"
}
]
}
Tip: Replace
ebytewithtext,actisense_bst,waveshare, orcandepending on your gateway hardware. All gateway subcommands support the--jsonflag.
Development
Contributions, feedback, and suggestions to improve this project are welcome. If you have ideas for new features, bug fixes, or improvements, feel free to open an issue or create a pull request. I’m always happy to collaborate and learn from the community!
Please don't hesitate to reach out with any questions, comments, or suggestions.
Setup for Development
To contribute to this library, clone the repository and install the required dependencies:
git clone https://github.com/tomer-w/nmea2000.git
cd nmea2000
pip install -e .[dev]
Running Tests
To run the tests, use:
pytest
Running the CLI Locally
To test the CLI locally, you can use the following command:
python -m nmea2000.cli decode --frame "your_hex_encoded_frame"
License
This project is licensed under the Apache 2.0 license - see the LICENSE file for details.
Acknowledgements
- This library leverages the canboat as the source for all PGN data.
- Special thanks to Rob from Smart Boat Innovations. His code was the initial inspiration for this project. Some of the code here may still be based on his latest open-source version.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file nmea2000-2026.5.1.tar.gz.
File metadata
- Download URL: nmea2000-2026.5.1.tar.gz
- Upload date:
- Size: 467.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
0cbc7fa7bc77c2795de2bd47d4f40e5151d6d120ef897ba3ddecd6b027a98d75
|
|
| MD5 |
fc86822e4485f15e13e6675cda6219c6
|
|
| BLAKE2b-256 |
51a004bad5064acb92f3b927721aaa821502558259f99cb6f5365565e07261a9
|
Provenance
The following attestation bundles were made for nmea2000-2026.5.1.tar.gz:
Publisher:
python-publish.yml on tomer-w/nmea2000
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
nmea2000-2026.5.1.tar.gz -
Subject digest:
0cbc7fa7bc77c2795de2bd47d4f40e5151d6d120ef897ba3ddecd6b027a98d75 - Sigstore transparency entry: 1532708705
- Sigstore integration time:
-
Permalink:
tomer-w/nmea2000@f6a1576537ff2bd1dd35445474460b172f2e7f09 -
Branch / Tag:
refs/tags/v2026.5.1 - Owner: https://github.com/tomer-w
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python-publish.yml@f6a1576537ff2bd1dd35445474460b172f2e7f09 -
Trigger Event:
release
-
Statement type:
File details
Details for the file nmea2000-2026.5.1-py3-none-any.whl.
File metadata
- Download URL: nmea2000-2026.5.1-py3-none-any.whl
- Upload date:
- Size: 453.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
beade7ecd7111eb9fddfdf540096a0405b3546808c74df300c0aca6dbc323cfc
|
|
| MD5 |
224243336f03e57b0ee8b738e391804d
|
|
| BLAKE2b-256 |
47f6e3b6c2ee68dcfe7bf93ee3934ab4db9af06bec3365c08d9392cb1e197f66
|
Provenance
The following attestation bundles were made for nmea2000-2026.5.1-py3-none-any.whl:
Publisher:
python-publish.yml on tomer-w/nmea2000
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
nmea2000-2026.5.1-py3-none-any.whl -
Subject digest:
beade7ecd7111eb9fddfdf540096a0405b3546808c74df300c0aca6dbc323cfc - Sigstore transparency entry: 1532708884
- Sigstore integration time:
-
Permalink:
tomer-w/nmea2000@f6a1576537ff2bd1dd35445474460b172f2e7f09 -
Branch / Tag:
refs/tags/v2026.5.1 - Owner: https://github.com/tomer-w
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python-publish.yml@f6a1576537ff2bd1dd35445474460b172f2e7f09 -
Trigger Event:
release
-
Statement type: