Skip to content

murphy/credmark-models-py

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Credmark Models and Model Framework

Credmark has created a Model Framework for creators to allow them easy access to Credmark's inhouse integrated, curated, and historical blockchain data via standard data models and to enable creators to leverage these models to build their own models and publish them.

It contains dependencies to the Credmark Model Framework repo, that are installed by following the description below.

Moreover, the Credmark Model Framework includes the credmark-dev command-line tool that lets you list the models in the local repo and run local models as you develop. It will also run remote non-local models on the server by automatically doing API calls when models are not present locally. Once the Credmark model framework is installed you can use this command-line tool.

You can browse the models that are already deployed at the Credmark Model Documentation site.

Quickstart

Prerequisites

Fork Repository

Fork credmark-models-py repository

Virtual Env

Create a virtual env (if you want):

python3 -m venv venv
source venv/bin/activate

If you wish, you can run it on miniconda. Simply install the miniconda version mentioned in the prerequisite, open Anaconda prompt, navigate to the repo folder and continue with steps (commands) as mentioned below.

Install Dependencies

Then run:

pip install -r requirements.txt

For development, you can also run:

pip install -r requirements-dev.txt

Configure environment variables

Some configuration is done with environment variables. They can be set in your shell or a .env file, which can be created at the root folder of the cloned repository.

Environment variables

The CREDMARK_WEB3_PROVIDER_CHAIN_ID_{N} is a JSON object where the keys are chain ids (as strings) and the values are URLs to HTTP providers.

Set {N} with a chain id, for example CREDMARK_WEB3_PROVIDER_CHAIN_ID_1 and set the value as the URL of the HTTP provider.

For example, a .env file can contain the following:

CREDMARK_WEB3_PROVIDER_CHAIN_ID_1=https://eth-mainnet.alchemyapi.io/v2/ABC123
CREDMARK_WEB3_PROVIDER_CHAIN_ID_137=https://polygon-mainnet.g.alchemy.com/v2/ABC123

ALTERNATIVELY you may set all your providers in a single env var:

For example, a .env file can contain the following:

CREDMARK_WEB3_PROVIDERS='{1:"https://eth-mainnet.alchemyapi.io/v2/ABC123","137":"https://polygon-mainnet.g.alchemy.com/v2/ABC123"}'

This variable is used to run models which require web3. It can be ignored for those models which do not require web3.

Run a Model

To see a list of all models available, use the command:

credmark-dev list

You can then pick a model name (slug) and run the model by using the command:

credmark-dev run <Specify Slug> -b <Specify block number>  -i <Specify Input>

so for example

credmark-dev run cmk.circulating-supply -b 14000000  -i “{}”

Tip: you can run the command

credmark-dev list --manifests

to see the input data format required for each model. It will also show the output formats.

Develop a Model

A model is essentially a python code file which implements the model class by subclassing from a base class. See some examples here.

Steps

  1. Create a folder in the models/contrib folder that will hold all of your models, for example models/contrib/my_models. You can add models directly there or create subfolders as desired. Do not work in another contributer's folder.
  2. Create a python file, for example model_foo.py (again it can have any name as long as it ends in .py) in the folder you created in step 1.
  3. Ensure your model class inherits from the base Model class credmark.model.Model. Also, use decorator @credmark.model.describe to define the metadata for your model. Example:
import credmark.model

@credmark.model.describe(slug='contrib.echo',
                   version='1.0',
                   display_name='Echo',
                   description="A test model to echo the message property sent in input.",
                   developer='Credmark',
                   input=EchoDto,
                   output=EchoDto)
class EchoModel(credmark.model.Model):
    def run(self, input: EchoDto) -> EchoDto:
        return input

The model class implements a run(self, input) method, which takes input data (as a dict or DTO (Data Transfer Object)) and returns a result dict or DTO (see later section DTO), with various properties and values, potentially nested with other JSON-compatible data structures.

A special DTO EmptyInput, which is an empty DTO, is created for model with no input required. The model input is EmptyInput by default if input is not specified in the decorator @credmark.model.describe. If the model takes non-empty input, modeller can specify input=dict or create a DTO for more structured data.

A model can optionally implement a init(self) method which will be called when the instance is initialized and the self.context is available.

Models can call other python code, in imported python files (in your models folder or below) or from packages, as needed. You may not import code from other model folders. One thing to keep in mind is that different instances of a model may or may not be run in the same python execution so do not make use of global or class variables unless they are meant to be shared across model instances.

A model instance has access to the following instance variables:

  • self.context - A context which holds state and provides functionality. Details on the Model Context are below.
  • self.logger - Python logger instance for logging to stderr(optional) A model should never write/print to stdout.

Please find more detailed examples here.

Constraints

  • Model slugs MUST start with "contrib." and the rest of the string can contain letters (upper and lowercase), numbers, and hyphens. In general, use a hyphen between words. Slugs must be unique in a case-insensitive manner across all models running within Credmark.
  • Input variables and Output data fields should use camel-cased names.

DTO

For the DTOs (Data Transfer Objects) we use the python module pydantic to define and validate the data. We have aliased pydantic's BaseModel as DTO and Field as DTOField to avoid confusion with Credmark models but all the functionality of pydantic is available.

The DTO used in the example above, for both the input and output, looks like this:

from credmark.types.dto import DTO, DTOField

class EchoDto(DTO):
    message: str = DTOField('Hello', description='A message')

The credmark-model-framework defines many common data objects as DTOs and fields such as Address, Contract, Token, Position, Portfolio etc. They can be found here

  • Example 1: Use Address in input/ouput DTOs

The input data, e.g. {"poolAddress":"0x..."}, is converted to Address type. The property .checksum to get its checksum address.

from credmark.types import Address

class PoolAddress(DTO):
    poolAddress: Address = DTOField(..., description='Address of Pool')

@credmark.model.describe(...
                         input=PoolAddress)
def run(self, input: PoolAddress):
    address = input.poolAddress.checksum
  • Example 2: Use Address to auto-convert to checksum address.
from credmark.types import Address

def run(self, input):
    address = Address(wallet_adress)
    address.checksum
  • Example 3: Pre-defined financial DTO to define input. Use it as object in the run(self, input)
from credmark.types import Portfolio

"""
# Portfolio is defined in the framework
# class Portfolio(DTO):
#    positions: List[Position] = DTOField([], description='List of positions')

class PortfolioSummary(DTO):
    num_tokens: int = DTOField(..., description='Number of different tokens')
"""

@credmark.model.describe(slug='contrib.type-test-1',
                         version='1.0',
                         display_name='Test Model',
                         description='A Test Model',
                         input=Portfolio,
                         output=PortfolioSummary)
class TestModel(credmark.model.Model):

    def run(self, input: Portfolio) -> PortfolioSummary:
        return PortfolioSummary(num_tokens=len(input.positions))

We strongly encourage you to create DTOs and/or make use of the common objects, either as your top-level DTO or as sub-objects and in lists etc. as needed.

You may use credmark-dev describe {model_slug} to show the input/output schema and examples for specific model(s). For example

credmark-dev describe aave.token-asset-historical

(...omit the output header)

Loaded models:

 - slug: aave.token-asset-historical
 - version: 1.0
 - tags: None
 - display_name: Aave V2 token liquidity
 - description: Aave V2 token liquidity at a given block number
 - developer:
 - input schema:
   Token(object)
     └─address(string)
 - input example:
   #01: {'symbol': 'USDC'}
   #02: {'symbol': 'USDC', 'decimals': 6}
   #03: {'address': '0x1F98431c8aD98523631AE4a59f267346ea31F984'}
   #04: {'address': '0x1F98431c8aD98523631AE4a59f267346ea31F984', 'abi': '(Optional) contract abi JSON string'}
 - output schema:
   BlockSeries(object)
     └─series(List[BlockSeriesRow])
         └─blockNumber(integer)
         └─blockTimestamp(integer)
         └─sampleTimestamp(integer)
         └─output(object)
 - output example:
   #01: {'series': [{'blockNumber': 'integer', 'blockTimestamp': 'integer', 'sampleTimestamp': 'integer', 'output': 'object'}]}
 - class: models.credmark.protocols.aave.aave_v2.AaveV2GetTokenAssetHistorical

Submit a Model

If you are a contributor external to credmark, you should create your folder in [credmark-models-py/models/contrib].

You should create and keep your models under this folder. Note that we have applied additional conditions for model slug names under this folder. Slug name must start with contrib.<model-name>, so for example: Slug = ‘contrib.sample-model.

If you are a contributor external to credmark, you should create your folder in [credmark-models-py/models/contrib].

You should create and keep your models under this folder. Note that we have applied additional conditions for model slug names under this folder. Slug name must start with contrib.<model-name>, so for example: Slug = ‘contrib.sample-model.

Once your model is ready to submit, simply create a pull request on the github repo and let us know in our Discord.

Model Library

See a list of the existing models in the repository.

API Gateway

The Credmark Framework provides access to remote models and access to on-chain data via Credmark API Gateway.

Interactive HTTP requests

If you go to the popup in the top right of the window you can now choose between the different model groups:

For each group, you will get the docs for all the models within the group and you are able to run them interactively. Note that not all models are fully documented yet.

Model documentation

If you're just looking for model documentation (without all the API details), please refer to the Credmark Model Documentation. This documentation contains the description of the models as well as example requests and responses and is searchable (by name and slug).

Framework Command Documentation

Help command

All the commands accept -h parameter for help, e.g.:

>credmark-dev -h

usage: credmark-dev [-h] [--log_level LOG_LEVEL] [--model_path MODEL_PATH] [--manifest_file MANIFEST_FILE]
                    {list,list-models,models,deployed-models,describe,describe-models,man,run,run-model,build,build-manifest,clean,remove-manifest} ...

Credmark developer tool

optional arguments:
  -h, --help            show this help message and exit
  --log_level LOG_LEVEL
                        Log level: DEBUG, INFO, WARNING, ERROR, CRITICAL
  --model_path MODEL_PATH
                        Semicolon separated paths to the model folders (or parent) or model python file. Defaults to models folder.
  --manifest_file MANIFEST_FILE
                        Name of the built manifest file. Defaults to models.json. [Not required during development]

Commands:
  Supported commands

  {list,list-models,models,deployed-models,describe,describe-models,man,run,run-model,build,build-manifest,clean,remove-manifest}
                        additional help
    list (list-models)  List models in this repo
    models (deployed-models)
                        List models deployed on server
    describe (describe-models, man)
                        Show documentation for local and deployed models
    run (run-model)     Run a model
    build (build-manifest)
                        Build model manifest [Not required during development]
    clean (remove-manifest)
                        Clean model manifest

runcommand

Below -h command shows the details of options available for run commands.

>credmark-dev run -h

usage: credmark-dev run [-h] [-b BLOCK_NUMBER] [-c CHAIN_ID] [-i INPUT] [-v MODEL_VERSION] [--provider_url_map PROVIDER_URL_MAP] [--api_url API_URL] model-slug

positional arguments:
  model-slug            Slug for the model to run.

optional arguments:
  -h, --help            show this help message and exit
  -b BLOCK_NUMBER, --block_number BLOCK_NUMBER
                        Block number used for the context of the model run. If not specified, it is set to the latest block of the chain.
  -c CHAIN_ID, --chain_id CHAIN_ID
                        [OPTIONAL] The chain ID. Defaults to 1.
  -i INPUT, --input INPUT
                        [OPTIONAL] Input JSON or if value is "-" it will read input JSON from stdin.
  -v MODEL_VERSION, --model_version MODEL_VERSION
                        [OPTIONAL] Version of the model to run. Defaults to latest.
  --provider_url_map PROVIDER_URL_MAP
                        [OPTIONAL] JSON object of chain id to Web3 provider HTTP URL. Overrides settings in env vars.
  --api_url API_URL     [OPTIONAL] Credmark API url. Defaults to the standard API gateway. You do not normally need to set this.
  --api_url API_URL     [OPTIONAL] Credmark API url

To call any model we can specify the output by providing below parameters (they're not necessarily required):

  • -b or –block_number : to define against which block number the model should run. If not specified, it uses the "latest" block from our ledger db.
  • -i or –input : to provide input for a model in a predefined structure.(you can run command credmark-dev list --manifests to see the input format required for each model. See example below). If not provided it will default to “{}”. Model-slug: Name of the model (slug) to call the model.

Note: if chain ID is not mentioned explicitly in the parameter, it defaults to 1. If the model is using web 3 instance then chain id (and blockchain) will be picked from the .env file we defined during setup (refer to “configure environment variable” section). If the model is using Credmark database then, by default, it will refer to the Ethereum blockchain.

See the example below. Here, we are running the model “cmk.circulating-supply” at block_number 14000000.

>Credmark-dev run -b 14000000 cmk.circulating-supply -i "{}"

{"slug": "cmk.circulating-supply", "version": "1.0", "output": {"result": 28314402605762084044696668}, "dependencies": {"cmk.total-supply": {"1.0": 1}, "cmk.circulating-supply": {"1.0": 1}}}

listcommand

Below -h command shows the details of options available for list commands.

>credmark-dev list -h

usage: credmark-dev list [-h] [--manifests] [--json]

optional arguments:
  -h, --help   show this help message and exit
  --manifests
  --json

Note: You can also run list-models command alternatively.

Example below shows simple output (list of all models and their version) of list command:

>credmark-dev list -h

Loaded models:

 - var: ['1.0']
 - cmk.total-supply: ['1.0']
 - cmk.circulating-supply: ['1.0']
 - xcmk.total-supply: ['1.0']
[...]

You can also get the list result in different formats using --json or --manifest.

Note: the commands build and clean does not need to be used.

Credmark Model Framework Core Components

In the following you will find the key components of every model.

Model Class

Credmark uses a simple base class called ‘Model’ class to set up a model. The actual code can be found here.

All Models should import this class import credmark.model and can override the run() method. See examples here.

The @credmark.model.describe() decorator provides a simple interface to define the model properties such as slug, version, display_name, description, developer, input, output etc so that it can be used easily by consumers and other models.

See example here.

Model Context

Each model runs with a particular context, including the name of the blockchain, block number, and a configured web3 instance (among other things). The context can be passed along when the model calls other models. The context’s web3 instance can be used to make RPC calls. The ModelContext() Class sets up the context for the model to run and can be accessed from a model as self.context. The base code can be found here. It provides an interface for models to run other models, call contracts, get ledger data, use a web3 instance etc.

It also enforces deterministic behavior for Models. The key utilities in ModelContext are

  • web3
  • contract
  • ledger
  • block number
  • historical utility

Methods

def run_model(name: str,
          input: Union[dict, DTO] = EmptyInput(),
          return_type: Union[Type[dict], Type[DTO], None],
          block_number: Union[int, None] = None,
          version: Union[str, None] = None)

A model can call other models and use their results. run_model() calls the specified model and returns the results as a dict or DTO (if return_type is specified) (or raises an error if the called model was unavailable or had an error.)

For example:

price = context.run_model('price', token, return_type=credmark.types.Price)

Web3

context.web3 will return a configured web3 instance with the default block set to the block number of context. The web3 providers are determined from the environment variables as described in the configuration section above. Currently users will need to use their own alchemy account (or other web3 provider) to access web3 functionality.

Address

Address class is inherited from str to help with web3 address conversion. It's highly recommended to use it instead of a baremetal address.

✔️: Address("0x7d2768dE32b0b80b7a3454c06BdAc94A69DDc7A9").checksum # checksum version to be used

❌: Address("0x7d2768dE32b0b80b7a3454c06BdAc94A69DDc7A9") # lower case version

❌:"0x7d2768de32b0b80b7a3454c06bdac94a69ddc7a9" # lower case version

Example:

    from credmark.types import (Address, Contract)

    contract = Contract(
        # lending pool address
        address=Address("0x7d2768dE32b0b80b7a3454c06BdAc94A69DDc7A9").checksum,
        abi=AAVE_V2_TOKEN_CONTRACT_ABI
    )

Contract

Credmark simplified the process of getting web3 instances of any contract from any chain. So you don't need to find and hardcode chain specific attributes and functions within these chains to run your models.

The model context exposes the “context.contracts” property which can be used to get contracts by metadata or address. The contracts are instances of the Contract class which are configured and use the web3 instance at specified block number and specified chain id along with additional data based on constructor_args.

Example code for contact class can be found here.

Currently below parameters as argument are supported to fetched using Contracts:

  • name: name of the contract
  • address: address of the contract
  • deploy_tx_hash: transaction hash at which contract was deployed
  • Constructor_args
  • protocol: protocol name
  • product: product name
  • abi_hash
  • abi

Contract functions are accessible using the contract.functions property.

Tip: the contract object returned from contract class can be used to fetch any specific web3 attributes of the contract and call contract functions. As well it can be used as a DTO (see details below) so it can be returned as part of the output of a model.

Ledger

Credmark allows access to in-house blockchain ledger data via ledger interface (context.ledger), so that any model can fetch/use ledger data if required. This is done via Ledger class which currently supports below functions:

  • get_transactions
  • get_traces
  • get_logs
  • get_contracts
  • get_blocks
  • get_receipts
  • get_erc20_tokens
  • get_erc20_transfers

Please refer here for the code of the Ledger class.

Block number

The context.block_number holds the block number for which a model is running. Models only have access to data at (by default) or before this block number (by instantiating a new context). In other words models cannot see into the future and ledger queries etc. will restrict access to data by this block number. As a subclass of int, the block_number class allows the provided block numbers to be treated as integers and hence enables arithmetic operations on block numbers. It also allows you to fetch the corresponding datetime and timestamp properties for the block number. This can be super useful in case we want to run any model iteratively for a certain block-interval or time-interval backwards from the block number provided in the context.

Example code for the block-number class can be found here.

Historical Utility

The historical utility, available at context.historical (see here), allows you to run a model over a series of blocks for any defined range and interval.

Block ranges can be specified by blocks (either a window from current block or a start and end block) or by time (a window from the current block’s time or start and end time.) Times can be specified different units, i.e. year, month, week, day, hour, minute and second.

See historical_example.py on how to use this class.

Data Transfer Object (DTO)

Input and output data for models are json-serializable objects of arbitrary depth and complexity. Objects can have 0 or more keys whose values can be null, number, string, array, or another object.

Although you can use dictionaries for model input and output data in your python code, we strongly encourage the use of DTOs (Data Transfer Objects.)

DTOs are classes with typed properties which will serialize and deserialize to and from JSON. They also automatically produce a JSON-schema that is used to document the input and output of a model. Each model may have their own DTOs or may share or inherit a DTO from another model that you have developed.

To create a DTO, simply subclass the DTO base class and use DTOFields to annotate your properties. Under the hood, the Credmark Model Framework uses the pydantic python module (DTO is simply an alias for pydantic BaseModel and DTOField an alias for Field) so almost anything that works with pydantic will work for your DTO.

Please see the pydantic docs for more information.

Additional Useful Modules

We also have some built-in reusable type classes available under Credmark.types.

We have created and grouped together different classes to manage input and output types to be used by models. These types include some standard blockchain and financial data structures as well as some standard input and output objects for Credmark models.

models

  1. ledger.py : DTOs and data used by the ledger models
  2. series.py: DTOs for the series models

data

1. Address: this class is a subclass of string and holds ablockchain address.

The address can be provided in lower case, upper case or checksum hex format. This class will normalize the address into lower case. Note that It can be used as a normal string but it also has a "checksum" property which returns a web3 ChecksumAddress.

See address_examples.py on how to use this class.

2. Account(s): Account simply holds an address. Accounts is a list of account instances which allows iteration through each account.

See iteration_example on how to use this class.

3. Contract: a contract is a subclass of Account which has a name, deployed transaction hash, abi, protocol name etc..

Object instantiation of this class will load all information available for the contract (against contract address provided as input) in our database and you can access whatever information you want from the object.

See contact_example.py on how to use this class.

4. Token: Token is a specific kind of contract; hence the Token class inherits from Contract class.

This class allows you to load token information with an address or symbol as well as get its price in USD Currently this class supports data load for erc20 token but we will support erc721 as well soon.

See token_example.py on how to use this class. Token_data.py lists all erc20 tokens currently supported.

5. Price and TokenPairPrice: these classes can be used to hold a price or a price of a token against a reference token (such as CMK-BTC, CMK-ETH etc.)

6. Position: this class holds a Token and an amount It can calculate its value based on the token price in USD. You can also access the scaled amount property if you need the scaled amount of the erc20 token.

Token_data.py lists all erc20 tokens currently supported.

7. Portfolio: This class holds a list of positions. So, it can be used to calculate all positions within a wallet.

Error handling

When running a model, the top level framework code will catch any exceptions and output error JSON to stdout and exit with an error code.

Models can raise a credmark.model.ModelRunError (or other Exception) to terminate a run.

Models that run another model will terminate if the requested model has an error.

Models should never write to stdout. They should use a logger to write to stderr.

About

The Credmark Models Repository

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • Python 94.0%
  • Shell 6.0%