- Introduction
- License
- Download
- Setup
- Source Code
- Development
- Creating a Release
- Internal Design
- The Transaction Resolver
- Plugin Development
- Frequently Asked Developer Questions
This document describes DaLI setup instructions, development workflow, design principles, source tree structure, internal design and plugin architecture.
DaLI is released under the terms of Apache License Version 2.0. For more information see LICENSE or http://www.apache.org/licenses/LICENSE-2.0.
The latest DaLI source can be downloaded at: https://github.com/eprbell/dali-rp2.
DaLI has been tested on Ubuntu Linux, macOS and Windows 10 but it should work on all systems that have Python version 3.8.0 or greater. Virtualenv is recommended for DaLI development.
First make sure Python, pip and virtualenv are installed. If not, open a terminal window and enter the following commands:
sudo apt-get update
sudo apt-get install python3 python3-pip virtualenv
Then install DaLI Python package requirements:
cd <dali_directory>
virtualenv -p python3 .venv
. .venv/bin/activate
.venv/bin/pip3 install -e '.[dev]'
First make sure Homebrew is installed, then open a terminal window and enter the following commands:
brew update
brew install python3 virtualenv
Then install DaLI Python package requirements:
cd <dali_directory>
virtualenv -p python3 .venv
. .venv/bin/activate
.venv/bin/pip3 install -e '.[dev]'
First make sure Python 3.8 or greater is installed (in the Python installer window be sure to click on "Add Python to PATH").
Then make sure Microsoft Visual C++ 14.0 or greater is installed (in the Visual Studio Installer window be sure to click on "Desktop development with C++"), which is needed for multidict.
Then open a PowerShell window and enter the following to install DaLI:
python -m pip install virtualenv
Then install DaLI Python package requirements:
cd <dali_directory>
virtualenv -p python .venv
.venv\Scripts\activate.ps1
python -m pip install -e ".[dev]"
If activate.ps1
cannot be loaded because running scripts is disabled on the system, run activate.bat
instead or change the PowerShell execution policy Set-ExecutionPolicy RemoteSigned -Scope CurrentUser
.
- install python 3.8 or greater
- install pip3
- install virtualenv
Then install DaLI Python package requirements:
cd <dali_directory>
virtualenv -p python3 .venv
.venv/bin/pip3 install -e '.[dev]'
The RP2 source tree is organized as follows:
.bumpversion.cfg
: bumpversion configuration;CHANGELOG.md
: change log document;config/
: config files for examples and tests;CONTRIBUTING.md
: contribution guidelines;docs/
: additional documentation, referenced from the README files;.editorconfig
;.gitattributes
;.github/workflows/
: configuration of Github continuous integration;.gitignore
;input/
: examples and tests;input/golden/
: expected outputs that Dali tests compare against;.isort.cfg
: isort configuration;LICENSE
: license information;Makefile
: alternative old-school build flow;MANIFEST.in
: source distribution configuration;mypy.ini
: mypy configuration;.pre-commit-config.yaml
: pre-commit configuration;.pylintrc
: pylint configuration;pyproject.toml
: packaging configuration;README.dev.md
: developer documentation;README.md
: user documentation;setup.cfg
: static packaging configuration file;setup.py
: dynamic packaging configuration file;src/dali
: DaLI code, including classes for transactions, ODS and config genator, transaction resolver, etc.;src/dali/data/
: spreadsheet templates that are used by the ODS generator;src/dali/plugin/country/
: country plugins/entry points;src/dali/plugin/input/csv/
: CSV-based data loader plugins;src/dali/plugin/input/rest/
: REST-based data loader plugins;src/dali/plugin/pair_converter/
: pair converter plugins;src/stubs/
: DaLI relies on third-party libraries, some of which don't have typing information, so it is added here;tests/
: unit tests.
Read the Contributing document on pull requests guidelines.
DaLI code adheres to these principles:
- user privacy is of paramount importance: user data never leaves the user's machine. The only networking logic allowed is read-only REST API calls in data loader plugins to collect transaction data;
- all identifiers have descriptive names;
- immutability:
- global variables have upper case names, are initialized where declared and are never modified afterwards;
- generally data structures are read-only (the only exceptions are for data structures that would incur a major complexity increase without write permission: e.g. AVL tree node):
- class fields are private (prepended with double-underscore). Fields that need public access have a read-only property. Write-properties are not used;
- @dataclass classes have
frozen=True
;
- data encapsulation: all data fields are private (prepended with double-underscore):
- for private access nothing else is needed;
- for protected access add a read-only property starting with single underscore or an accessor function starting with
_get_
; - for public access add a read-only property starting with no underscore or an accessor function starting with
get_
;
- runtime checks: parameters of public functions are type-checked at runtime;
- type hints: all variables and functions have Python type hints (with the exception of local variables, for which type hints are optional);
- no id-based hashing: hashable classes must redefine eq(), neq(), hash(). For hash(), utilizing immutable (i.e., read-only) attributes is appropriate;
- encapsulated math: all high-precision math is done via
RP2Decimal
(a subclass of Decimal), to ensure the correct precision is used throughout the code.RP2Decimal
instances are never mixed with other types in expressions; - f-strings only: every time string interpolation is needed, f-strings are used;
- no raw strings (unless they occur only once): use global constants instead;
- logging: logging is done via the
logger
module; - no unnamed tuples: dataclasses or named tuples are used instead;
- one class per file (with exceptions for trivial classes);
- files containing a class must have the same name as the class (but lowercase with underscores): e.g. class AbstractEntry lives in file abstract_entry.py;
- abstract classes' name starts with
Abstract
; - no imports with
*
.
DaLI uses pre-commit hooks for quick validation at commit time and continuous integration via Github actions for deeper testing. Pre-commit hooks invoke: flake8, black, isort, pyupgrade and more. Github actions invoke: mypy, pylint, bandit, unit tests (on Linux, Mac and Windows), markdown link check and more.
While every commit and push are automatically tested as described, sometimes it's useful to run some of the above commands locally without waiting for continuous integration. Here's how to run the most common ones:
- run unit tests:
pytest --tb=native --verbose
- type check:
mypy src tests
- lint:
pylint -r y src tests/*.py
- security check:
bandit -r src
- reformat code:
black src tests
- sort imports:
isort .
- run pre-commit tests without committing:
pre-commit run --all-files
Logs are stored in the log
directory. To generate debug logs on Linux or Mac, prepend the command line with LOG_LEVEL=DEBUG
, e.g.:
LOG_LEVEL=DEBUG dali_us -s -o output/ config/test_config.ini
In Windows Powershell, debug logs can be generated with the following command:
$env:LOG_LEVEL='DEBUG'; dali_us -s -o output/ config/test_config.ini
Unit tests are in the tests directory. Please add unit tests for any new code.
This section is for project maintainers.
To create a new release:
- add a section named as the new version in CHANGELOG.md
- use the output of
git log
to collect significant changes since last version and add them to CHANGELOG.md as a list of brief bullet points git add CHANGELOG.md
git commit -m "Updated with latest changes" CHANGELOG.md
bumpversion patch
(orbumpversion minor
orbumpversion major
)git push
- wait for all tests to pass successfully on Github
- add a tag in Github (named the same as the version but with a
v
in front, e.g.v1.0.4
): click on "Releases" and then "Draft a new release"
To create a Pypi distribution:
make distribution
make upload_distribution
DaLI's control flow is as follows (see dali_main.py):
- parse the INI configuration file which includes plugin initialization parameters and global configuration sections;
- discover and instantiate pair converter plugins using the initialization parameters from the config file and store them in a list to be passed to the transaction resolver (see below);
- discover and instantiate data loader plugins using the initialization parameters from the config file and call their
load()
method, which reads data from native sources (CSV files or REST endpoints) and returns it in a standardized format: a list of AbstractTransaction instances. This list can contain instances of anyAbstractTransaction
subclass: InTransaction (acquired crypto), OutTransaction (disposed-of crypto) or IntraTransaction (crypto transferred across accounts controlled by the same person or by people filing together); - join the lists returned by plugin
load()
calls and pass them to the transaction resolver, which merges incomplete transactions, filling in any missing information (e.g. the spot price) and returning a normalized list of transactions (see below for more details); - pass the resolved data to the RP2 ODS input file generator and the RP2 config file generator, which create the input files for RP2.
The transaction resolver is a critical component of DaLI and has the purpose of merging and normalizing transaction data from data loader plugins. Data loader plugins operate on incomplete information: e.g. if a transaction transfers crypto from Coinbase to Trezor, the Coinbase data loader plugin has no way of knowing that the destination address represents a Trezor account (because Coinbase itself doesn't have this information): so the plugin cannot fill the to_exchange
, to_holder
and crypto_received
fields of the IntraTransaction (so it fills them with Keyword.UNKNOWN
). Similarly the Trezor data loader plugin cannot know that the source address belongs to a Coinbase account and therefore it cannot fill the from_exchange
, from_holder
and crypto_sent
fields of the IntraTransaction (so it fills them with Keyword.UNKNOWN
). So how does DaLI merge these two incomplete transaction parts into one complete IntraTransaction? It uses the transaction resolver, which relies on the unique_id
field of each incomplete transaction to pair them: typically this is the transaction hash, but in certain special cases it could also be an exchange-specific value that identifies uniquely the transaction. The transaction resolver analyzes all generated transactions, looks for pairs of incomplete ones with the same unique_id
and merges them into a single one (be sure to read the FAQ on how to populate the unique_id field, which discusses this topic in detail, and the Manual Plugin examples on partial transaction resolution).
For this reason it's essential that all data loader plugins populate the unique_id
field as best they can and with a global identifier that is understood by all plugins (again, the transaction hash): without it the transaction resolver cannot merge incomplete data. Sometimes hash information is missing (especially in CSV files) and so it's impossible to populate the unique_id
field: in such cases use Keyword.UNKNOWN
, but the user will have to manually modify the generated result and perform transaction resolution manually, which is not ideal.
Another feature of the transaction resolver is filling in or converting certain fiat values, using the pair converter plugin list passed to it:
spot_price
: sometimes the native sources read by data loader plugins don't provide this information. If instructed by the user with the-s
option, the transaction resolver tries to retrieve this information from pair converter plugins;- foreign fiat values: transactions that occurred on foreign exchanges can have their fiat values denominated in non-native fiat. When the transaction resolver detects this condition, it converts these foreign fiat values to native fiat (e.g. USD for US, JPY for Japan, etc.), using pair converter plugins.
DaLI has a plugin architecture for data loaders, pair converters and countries, which makes it extensible for new use cases:
- data loader plugins read crypto data from a native source (REST endpoint or CSV file) and convert it into DaLI's internal format;
- pair converter plugins convert from a currency to another (both fiat and crypto);
- country plugins enable support for new countries.
Data loader plugins live in one of the following directories, depending on their type (CSV or REST):
src/dali/plugin/input/csv/
;src/dali/plugin/input/rest/
.
If at all possible prefer implementing a REST plugin over a CSV one, because CSV sources are often incomplete. Furthermore, if the exchange is compatible with CCXT, a CCXT-based REST plugin is the most preferred.
CCXT-based data loader plugins are subclasses of AbstractCcxtInputPlugin.
- define their own constructor with any custom parameters;
- invoke the superclass constructor in their own constructor;
- implement the
exchange_name()
andplugin_name()
methods that return the name of the exchange and the name of the plugin without spaces as astr
. - implement the
_initialize_client()
method that returns an initialized instance of the subclass ofExchange
that handles the plugin's exchange. The initialization of a CCXT exchange instance is described here. - implement the
_get_process_deposits_pagination_detail_set()
,_get_process_trades_pagination_detail_set()
and_get_process_withdrawals_pagination_detail_set()
methods that return a AbstractPaginationDetailSet instance. If an exchange does not support one of these unified functions, returnNone
. - implement the
_process_gains()
method, which reads data for gains received using the implicit API of CCXT, accepts a list of InTransaction instances, and OutTransaction instances. If the exchange doesn't support gains, the plugin can usepass
in the method. - implement the
_process_implicit_api()
method, which reads any data not covered by_process_gains()
using the implicit api of CCXT and accepts a list of InTransaction instances, OutTransaction instances, and IntraTransaction. If the the implicit API is not needed, the plugin can usepass
in the method. - the protected property
_client
can be used in the subclasses to access the exchange instance to make calls to the implicit api in the subclass. - the protected property
_logger
can be used for logging. - implementing the
load()
method is not required for CCXT-based data loader plugins.
All other data loader plugins are subclasses of AbstractInputPlugin.
- define their own constructor with any custom parameters;
- invoke the superclass constructor in their own constructor;
- implement the
load()
method, which reads data from the native source and returns a list of AbstractTransaction instances, which can be of any of the following classes: InTransaction (acquired crypto), OutTransaction (disposed-of crypto) or IntraTransaction (crypto transferred across accounts controlled by the same person or by people filing together). The fields of transaction classes are described here.
If a field is unknown the plugin can fill it with Keyword.UNKNOWN
, unless it's an optional field (check its type hints in the Python code), in which case it can be None
. The unique_id
requires special attention, because the transaction resolver uses it to match and join incomplete transactions: the plugin must ensure to populate it with the correct value. See the transaction resolver section for more details on unique_id
.
For an example of a CCXT-based data loader look at the Binance plugin, for an example of a REST-based data loader look at the Coinbase plugin, for an example of a CSV-based data loader look at the Trezor plugin.
Pair converter plugins live in the following directory:
src/dali/plugin/pair_converter
.
All pair converter plugins are subclasses of AbstractPairConverterPlugin and they must:
- implement the
name()
method; - implement the
cache_key()
method; - implement the
get_historic_bar_from_native_source()
method.
For an example of pair converter look at the Coinbase-advanced plugin.
Country plugins are reused from RP2 and their DaLI counterpart has trivial implementation.
To add support for a new country in DaLI:
- add a country plugin to RP2;
- add a new Python file to the
src/dali/plugin/country
directory and name it after the ISO 3166-1 alpha-2 2-letter code for the country; - in the newly added file add a DaLI-specific entry point instantiating the new country instance and passing it to
dali_main()
. As an example see the us.py file; - add a console script to setup.cfg pointing the new country dali_entry (see the US example in the console_scripts section of setup.cfg).
When submitting a new plugin open a PR and make sure all the following bullet points apply to your code:
- the plugin is privacy-focused: it doesn't send user data anywhere;
- the plugin follows the contribution guidelines;
- the plugin has one or more unit test;
- the plugin and its initialization parameters are documented in a section of docs/configuration_file.md.
- the plugin lives in the appropriate subdirectory of
src/dali/plugin/
;
Data-loader-specific list:
- the plugin lives in
src/dali/plugin/input/rest/
orsrc/dali/plugin/input/csv/
, depending on its type; - the plugin creates transactions that have
unique_id
populated (typically with the transaction hash), unless the information is missing from the native source: this is essential to the proper operation of the transaction resolver; - REST plugins have three comments at the beginning of the file, containing links to:
- REST API documentation
- authentication procedure documentation
- URL of the REST endpoint
This makes it easier to review the code. E.g.:
# REST API: https://developers.coinbase.com/api/v2
# Authentication: https://developers.coinbase.com/docs/wallet/api-key-authentication
# Endpoint: https://api.coinbase.com
- CSV plugins have a comment at the beginning of the file, documenting the format. E.g.:
# CSV Format: timestamp; type; transaction_id; address; fee; total
- REST plugins document what a sample JSON response looks like after the JSON call.
This also makes it easier to review the code. For example:
# [
# {
# 'time': '1624233772000', // epoch timestamp in ms
# 'asset': 'BTC', // locked asset
# 'amount': '0.017666', // Amount locked
# },
# ]
- the plugin's
load()
method is implemented and returns a list of AbstractTransaction subclasses; - the plugin's
__init__()
method calls the superclass constructor:super().__init__(account_holder, native_fiat=native_fiat)
- the plugin's
__init__()
method creates a plugin-specific logger with a name that uniquely identifies the specific instance of the plugin (typically you can add a subset of constructor parameters to ensure uniqueness): this way log lines can be easily distinguished by plugin instance. Example of a plugin-specific log in the constructor of the Trezor plugin:self.__logger: logging.Logger = create_logger(f"{self.__TREZOR}/{currency}/{self.__account_nickname}/{self.account_holder}")
- the plugin uses
self.__logger.debug()
throughout its code to capture all native-format data (which is is useful for debugging). Note thatlogger.debug()
calls only occur if the user setsLOG_LEVEL=DEBUG
; - CSV plugins have one or more unit test;
- REST plugins have one or more unit tests: use pytest-mock to simulate network calls (see test_plugin_coinbase.py for an example of this);
Read the frequently asked developer questions.