CSS-Electronics / can_decoder

API module for decoding raw CAN bus data to physical values
MIT License
67 stars 14 forks source link

CAN Decoder - DBC decode CAN Bus Data

This package lets you DBC decode raw CAN data from the CANedge to human-readable form (physical values) in Python.


Key features

1. Easily decode raw CAN bus data via DBC files
2. Support for regular CAN, OBD2 and J1939
3. Very fast data conversion and minimal external dependencies
4. Output to pandas dataframe for easy use of the data
5. Conversion can be done iteratively (from iterator) or in bulk (from DataFrame) 
6. Can be used together with our mdf_iter and canedge_browser

Installation

Use pip to install the can_decoder module:

pip install can_decoder

Optionally install canmatrix and pandas to load DBC files and enable conversion of pandas dataframes:

pip install canmatrix pandas

Dependencies


Module usage example

Below we load a log file via mdf_iter and use can_decoder to DBC decode it:

import can_decoder
import mdf_iter

mdf_path = "00000001.MF4"
dbc_path = "j1939.dbc"

db = can_decoder.load_dbc(dbc_path)
df_decoder = can_decoder.DataFrameDecoder(db)

with open(mdf_path, "rb") as handle:
    mdf_file = mdf_iter.MdfFile(handle)
    df_raw = mdf_file.get_data_frame()

df_phys = df_decoder.decode_frame(df_raw)
print(df_phys)

Further examples are included in the repo.


Documentation

Supplying decoding rules

Data decoding is based on a set of signals which can be grouped together in frames. The frames in turn are grouped together in a single database. The list of rules can be crafted by hand, using the primitives Signal, Frame and SignalDB - or generated from a DBC file.

From a DBC file

If canmatrix is installed, the library can load the conversion rules from a DBC file:

db = can_decoder.load_dbc(dbc_path)

By default, the output will distinguish signals by the signal name (e.g. EngineSpeed). It is possible to switch from the primary signal name to another signal attribute in the DBC file by supplying the optional use_custom_attribute keyword. This takes the form of a string, and can e.g. be used to select SPNs instead of signal names in a J1939 DBC file. If no valid attribute is found, the signal name is used instead.

db = can_decoder.load_dbc(dbc_path, use_custom_attribute="SPN")

Data conversion

The library supports two methods of decoding data:

Data conversion (iterator)

For iterative decoding (frame-by-frame), the library uses the IteratorDecoder class. This class takes a set of conversion rules (e.g. from a DBC file) and an iterable object (e.g. a MDF file):

decoder = can_decoder.IteratorDecoder(mdf_file, db)

for record in decoder:
    ...

This method expects an iterator structure like that of mdf_iter - incl. the following fields:

In the case multiple signals are defined from a single ID, the library iterator will queue them internally, deferring the request for more data until all signals have been consumed from the iterator.

The output is of the form decoded_signal, which is a namedtuple with the following fields:

Data conversion (DataFrame)

For batch conversion of messages, the library uses the DataFrameDecoder class. This is constructed with the conversion rules as a parameter and can be re-used several times from the same set of parameters:

df_decoder = can_decoder.DataFrameDecoder(db)

df_phys_1 = df_decoder.decode_frame(df_raw_1)
df_phys_2 = df_decoder.decode_frame(df_raw_2)

The data supplied should be similar to that of the iterator method, but as a DataFrame. See also the initial example. Unlike the iterator component, this method does not require the presence of a time stamp entry. Instead, the index of the DataFrame passed to the decoder will be used as the index in the resulting DataFrame.

The output is a dataframe with the same index as the input dataframe, containing decoded results for the frames matched by the loaded DBC file.

DataFrame output columns

The available signals in the output depends on the type of conversion. For generic CAN data (incl. OBD2), the following output columns are included:

When decoding data using a J1939 DBC, the output includes the following extra columns:

To remove columns from the output you can use the keyword columns_to_drop:

df_phys = df_decoder.decode_frame(df_raw, columns_to_drop=["CAN ID", "Raw Value"])