-
Notifications
You must be signed in to change notification settings - Fork 39
csp.Struct API
csp.Struct
is a native, first-class CSP type that should be used for struct-like data ( key, values ). csp.Struct
is implemented as a high performant C++ object and CSP C++ adapters are able to create / access them efficiently from C++.
In general its recommended to use csp.Struct
for any object-like data required on your timeseries rather than plain old python object unless there is good reason not to.
csp.Struct
types need to declare all their fields as annotated fields, for example:
class MyData(csp.Struct):
a: int
b: str = 'default'
c: list
def some_method(self):
return a * 2
>>> MyData()
MyData( a='<unset>', b='default', c='<unset>' )
The variables a
, b
, c
here define the struct members (similar to __slots__
on python objects, structs can not have any other attributes set on them other than the ones listed here).
Defaults: Note that members can be defined with default values, as b
is here.
Unset fields: Note that struct fields can be "unset". Fields can be checked for existence with hasattr(o, field)
and can be removed with del o.field
.
Methods: Note that you can define methods on structs just like any other python object.
While building your graph, if you have an edge that represents a csp.Struct
type you can access a member of that struct at graph time. What this means is that when you do edge.field
in your graph code, you will get a new edge that will tick with the value of that field. CSP will implicitly inject a csp.node
that will extract that field whenever the struct timeseries ticks. Note that if the struct ticks and the field is unset, the field's edge will not tick. Here's an example of this in practice:
import csp
from datetime import datetime
class Trade(csp.Struct):
price: float
size: int
@csp.graph
def my_graph():
trades = csp.curve(Trade,
[(datetime(2020, 1, 1), Trade(price=100.01, size=200)),
(datetime(2020, 1, 1, 0, 0, 1), Trade(price=100.01, size=300))]
)
sizes = trades.size
cumqty = csp.accum(sizes)
csp.print('trades', trades)
csp.print('cumqty', cumqty)
csp.run( my_graph, starttime = datetime( 2020, 1, 1 ))
>>> 2020-01-01 00:00:00 trades:Trade( price=100.01, size=200 )
>>> 2020-01-01 00:00:00 cumqty:200
>>> 2020-01-01 00:00:01 trades:Trade( price=100.01, size=300 )
>>> 2020-01-01 00:00:01 cumqty:500
trades
is defined as a timeseries of Trade
objects. On line 13 we access the size
field of the trades
timeseries, then accumulate the sizes to get cumqty
edge.
List fields in a csp.Struct
can be specified as three different types - untyped Python list, typed regular list and typed FastList
.
A Python list field keeps its value as a Python object, it is not typed. It can be created by annotating the field as list
.
A regular list field is typed and internally implemented as a subclass of the Python list class, so it behaves exactly like Python list and passes the isinstance
check for list. It can be created by annotating the field as typing.List[T]
or [T]
.
A FastList
field is typed and a more efficient implementation, implementing all Python list operations natively in C++. However, it doesn't pass isinstance
check for list. It can be created by annotating the field as csp.impl.types.typing_utils.FastList[T]
.
Example of using Python list field:
import csp
class A(csp.Struct):
a: list
s = A(a = [1, 'x'])
s.a.append(True)
print(f"Using Python list field: value {s.a}, type {type(s.a)}, is Python list: {isinstance(s.a, list)}")
>>> Using Python list field: value [1, 'x', True], type <class 'list'>, is Python list: True
Example of using regular list field:
from typing import List
import csp
class A(csp.Struct):
a: List[int]
s = A(a = [1, 2])
s.a.append(3)
print(f"Using list field: value {s.a}, type {type(s.a)}, is Python list: {isinstance(s.a, list)}")
>>> Using list field: value [1, 2, 3], type <class '_cspimpl.PyStructList'>, is Python list: True
Example of using FastList
field:
import csp
from csp.impl.types.typing_utils import FastList
class A(csp.Struct):
a: FastList[int]
s = A(a = [1, 2])
s.a.append(3)
print(f"Using FastList field: value {s.a}, type {type(s.a)}, is Python list: {isinstance(s.a, list)}")
>>> Using FastList field: value [1, 2, 3], type <class '_cspimpl.PyStructFastList'>, is Python list: False
-
clear(self)
clear all fields on the struct -
collectts(self, **kwargs)
:kwargs
expects key/values of struct fields and time series to populate the struct. This will return an Edge representing a ticking struct created from all the ticking inputs provided. Structs will only be generated from inputs that actively ticked in the given engine cycle (seefromts
to create struct from all valid inputs) -
copy(self)
: return a shallow copy of the struct -
copy_from(self, rhs)
: copy data fromrhs
into this instance.rhs
must be of the same type or a derived type. Copy will include unsetting fields unset in therhs
. -
update_from(self, rhs)
: copy only the set fields from therhs
into this instance -
update(self, **kwargs)
: in this instance, set the provided fields with the provided values -
fromts(self, trigger=None, /, **kwargs)
: similar tocollectts
above,fromts
will create a ticking Struct timeseries from the valid values of all the provided inputs whenever any of them tick.trigger
is an optional position-only argument which is used as a trigger timeseries for when to convert inputs into a struct tick. By default any input tick will generate a new struct of valid inputs -
from_dict(self, dict)
: convert a regular python dict to an instance of the struct -
metadata(self)
: returns the struct's metadata as a dictionary of key : type pairs -
to_dict(self)
: convert struct instance to a python dictionary -
to_json(self, callback=lambda x: x)
: convert struct instance to a json string, callback is invoked for any values encountered when processing the struct that are not basic Python types, datetime types, tuples, lists, dicts, csp.Structs, or csp.Enums. The callback should convert the unhandled type to a combination of the known types. -
all_fields_set(self)
: returnsTrue
if all the fields on the struct are set. Note that this will not recursively check sub-struct fields
csp.Struct
types may inherit from each other, but multiple inheritance is not supported. Composition is usually a good choice in absence of multiple inheritance.
This wiki is autogenerated. To made updates, open a PR against the original source file in docs/wiki
.
Get Started (Tutorials)
Concepts
- CSP Node
- CSP Graph
- Historical Buffers
- Execution Modes
- Adapters
- Feedback and Delayed Edge
- Common Mistakes
How-to guides
- Use Statistical Nodes
- Create Dynamic Baskets
- Write Adapters:
- Profile CSP Code
References
- API Reference
- Glossary of Terms
- Examples
Developer Guide