Python callable parsing#
There are currently several methods that implement complex translation of python code to OneTick’s expressions:
Translation#
The main feature of these methods is that, unlike most of the other onetick.py.Source
methods,
using callables for them is not resulting in these callables being called in python, but rather
these callables being translated from python code to OneTick expressions.
Therefore, these callables will not be called on each tick and they will not be called even once.
Methods onetick.py.Operation.apply()
and onetick.py.Source.apply()
work the same
and return a OneTick’s CASE
expression as a new onetick.py.Column
object,
but the first parameter of the callable for onetick.py.Operation.apply()
represents the current column
and for onetick.py.Source.apply()
it represents the whole source object.
>>> t = otp.Tick(A=1)
>>> t['A'].apply(lambda x: x + 1 if x >= 0 else x - 1)
Column(case((A) >= (0), 1, (A) + (1), (A) - (1)), <class 'int'>)
>>> t.apply(lambda x: x['A'] + 1 if x['A'] >= 0 else x['A'] - 1)
Column(case((A) >= (0), 1, (A) + (1), (A) - (1)), <class 'int'>)
Using functions for these methods works the same.
>>> t = otp.Tick(A=1)
>>> def fun(x):
... if x >= 0:
... return x + 1
... return x - 1
>>> t['A'].apply(fun)
Column(case((A) >= (0), 1, (A) + (1), (A) - (1)), <class 'int'>)
>>> def fun(x):
... if x['A'] >= 0:
... return x['A'] + 1
... return x['A'] - 1
>>> t.apply(fun)
Column(case((A) >= (0), 1, (A) + (1), (A) - (1)), <class 'int'>)
Method onetick.py.Source.script()
works differently and in this case
python’s callable is translated to OneTick’s per-tick script language and
this script is used with SCRIPT
event processor and method returns new onetick.py.Source
object.
Also the first argument of the callable represents the special input tick object.
>>> t = otp.Ticks(A=[-1, 1])
>>> def fun(tick):
... tick['B'] = 0
... if tick['A'] >= 0:
... tick['B'] = 1
>>> t = t.script(fun)
>>> otp.run(t)
Time A B
0 2003-12-01 00:00:00.000 -1 0
1 2003-12-01 00:00:00.001 1 1
Apply methods#
onetick.py.Operation.apply()
and onetick.py.Source.apply()
For apply methods the main idea is that the logic of the python callable
should be convertable to OneTick’s CASE
expression.
Python’s lambda-expressions have the same semantic capabilites as OneTick’s CASE
expressions,
so there are no limitations for them.
For functions some restrictions should be taken into consideration.
Only these operators are supported:
if
statementreturn
statement
>>> t = otp.Ticks(A=[-1, 1])
>>> def fun(x):
... if x['A'] >= 0:
... return x['A'] + 1
... return x['A'] - 1
>>> t['X'] = t.apply(fun)
>>> otp.run(t)
Time A X
0 2003-12-01 00:00:00.000 -1 -2
1 2003-12-01 00:00:00.001 1 2
Also, the python code translation is very flexible and allows some of the python operators and methods to be translated into simpler ones or to be executed in python and then be translated to OneTick.
Simple
for
statement can be replaced with it’s duplicated body:
>>> t = otp.Tick(A=1)
>>> def fun(x):
... for i in [999, 55, 1]:
... if x['A'] > i:
... return i
... return -333
>>> t.apply(fun)
Column(case((A) > (999), 1, 999, case((A) > (55), 1, 55, case((A) > (1), 1, 1, -333))), <class 'int'>)
If the first parameter is propagated to inner python callables, the code of these callables will also be translated (they won’t be called):
>>> t = otp.Tick(A=1)
>>> def inner_fun(x):
... if x['A'] >= 10:
... return 10
... return 0
>>> t.apply(lambda x: inner_fun(x) if x['A'] > 0 else -10)
Column(case((A) > (0), 1, case((A) >= (10), 1, 10, 0), -10), <class 'int'>)
All other inner python callables will be called once and their result will be inserted in OneTick expression:
>>> t = otp.Tick(A=1)
>>> t.apply(lambda x: x['A'] + sum([1, 2, 3, 4, 5]))
Column((A) + (15), <class 'int'>)
Per-tick script#
For onetick.py.Source.script()
method more python operators may also be used in the function:
Adding new and modifying existing columns#
It can be done by modifying first parameter of the function. This parameter represents input tick.
>>> t = otp.Tick(A=1)
>>> def fun(tick):
... tick['A'] += 1
... tick['B'] = 'B'
>>> t = t.script(fun)
>>> otp.run(t)
Time A B
0 2003-12-01 2 B
Filtering ticks#
By default, all input ticks are returned in script.
To filter tick out you can use return False
statement.
If some return
statements are specified, then all ticks
are filtered out by default, and the user is expected to control
all cases where ticks should be propagated or not.
>>> t = otp.Ticks(A=[1, -1])
>>> def fun(tick):
... if tick['A'] < 0:
... return False
... return True
>>> t = t.script(fun)
>>> otp.run(t)
Time A
0 2003-12-01 1
Propagating ticks#
yield
statement allows to propagate tick more than once.
>>> t = otp.Tick(A=1)
>>> def fun(tick):
... yield
... yield
... return False
>>> t = t.script(fun)
>>> otp.run(t)
Time A
0 2003-12-01 1
1 2003-12-01 1
Local variables#
Simple local variables are redefined on each arrived input tick. Static local variables are defined once and their values are saved between the arrival of input ticks.
>>> t = otp.Ticks(A=[0, 1])
>>> def fun(tick):
... a = 1234
... b = otp.static(0)
... a = a + 1
... b = b + 1
... tick['A'] = a * 2
... tick['B'] = b
>>> t = t.script(fun)
>>> otp.run(t)
Time A B
0 2003-12-01 00:00:00.000 2470 1
1 2003-12-01 00:00:00.001 2470 2
Python function calls#
Simple function calls results are also inserted into resulting OneTick code.
>>> t = otp.Tick(A=1)
>>> def fun(tick):
... tick['A'] = sum([1, 2, 3, 4, 5])
>>> t = t.script(fun)
>>> otp.run(t)
Time A
0 2003-12-01 15
Python for
statements#
Simple for
statements are also translated to several copies of the body of this statement.
>>> t = otp.Tick(A=1)
>>> def fun(tick):
... for i in [1, 2, 3, 4, 5]:
... tick['A'] += i
>>> t = t.script(fun)
>>> otp.run(t)
Time A
0 2003-12-01 16
Looping with while
statement#
>>> t = otp.Tick(A=1)
>>> def fun(tick):
... tick['X'] = 0
... while tick['X'] < 5:
... tick['X'] += 1
>>> t = t.script(fun)
>>> otp.run(t)
Time A X
0 2003-12-01 1 5
Tick sequences#
You can iterate over tick sequences inside per-tick script. These sequences should be created outside of the per-tick script.
>>> t = otp.Tick(A=1)
>>> t.state_vars['list'] = otp.state.tick_list(otp.eval(otp.Ticks(X=[1, 2, 3, 4, 5])))
>>> def fun(tick):
... for t in tick.state_vars['list']:
... tick['A'] += t['X']
>>> t = t.script(fun)
>>> otp.run(t)
Time A
0 2003-12-01 16
Tick descriptor fields#
It’s possible to iterate over tick descriptor fields in per-tick script, get their names and types.
>>> t = otp.Tick(A=1, B=2)
>>> def fun(tick):
... tick['NAMES'] = ''
... for field in otp.tick_descriptor_fields():
... tick['NAMES'] += field.get_name() + ','
>>> t = t.script(fun)
>>> otp.run(t)
Time A B NAMES
0 2003-12-01 1 2 A,B,
- class TickDescriptorFields[source]#
Class for declaring tick descriptor fields in per-tick script. Can only be iterated, doesn’t have methods and parameters.
Examples
>>> t = otp.Tick(A=1) >>> def fun(tick): ... for field in otp.tick_descriptor_fields(): ... tick['NAME'] = field.get_name() >>> t = t.script(fun) >>> otp.run(t) Time A NAME 0 2003-12-01 1 A
See also
- class TickDescriptorField(name, **_)[source]#
Tick descriptor field object. Can be accessed only while iterating over
otp.tick_descriptor_fields
in per-tick script.Examples
>>> t = otp.Tick(A=2, B='B', C=1.2345) >>> def fun(tick): ... tick['NAMES'] = '' ... tick['TYPES'] = '' ... tick['SIZES'] = '' ... for field in otp.tick_descriptor_fields(): ... tick['NAMES'] += field.get_name() + ',' ... tick['TYPES'] += field.get_type() + ',' ... tick['SIZES'] += field.get_size().apply(str) + ',' >>> t = t.script(fun) >>> otp.run(t) Time A B C NAMES TYPES SIZES 0 2003-12-01 2 B 1.2345 A,B,C, long,string,double, 8,64,8,
Tick objects#
Tick objects can be created inside per-tick script. They can be copied or modified, also some methods are available.
>>> t = otp.Tick(A=1)
>>> def fun(tick):
... t = otp.dynamic_tick()
... t['A'] = 12345
... tick.copy_tick(t)
>>> t = t.script(fun)
>>> otp.run(t)
Time A
0 2003-12-01 12345
- tick_list_tick()[source]#
Can be used only in per-tick script function to define a tick list tick local variable.
Tick list ticks can be used with some methods of tick lists
onetick.py.state.tick_list
.Note
Note that
onetick.py.static
value is returned. You should not define tick variable as static manually.Examples
>>> def fun(tick): ... t = otp.tick_list_tick() ... tick.state_vars['LIST'].push_back(t)
See also
- tick_set_tick()[source]#
Can be used only in per-tick script function to define a tick set tick local variable.
Tick set ticks can be used with some methods of tick sets
onetick.py.state.tick_set
.Note
Note that
onetick.py.static
value is returned. You should not define tick variable as static manually.Examples
>>> def fun(tick): ... t = otp.tick_set_tick() ... if tick.state_vars['SET'].find(t, -1): ... tick['RES'] = '-1'
See also
- tick_deque_tick()[source]#
Can be used only in per-tick script function to define a tick deque tick local variable.
Tick deque ticks can be used with some methods of tick deques
onetick.py.state.tick_deque
.Note
Note that
onetick.py.static
value is returned. You should not define tick variable as static manually.Examples
>>> def fun(tick): ... t = otp.tick_deque_tick() ... tick.state_vars['DEQUE'].get_tick(0, t)
See also
- dynamic_tick()[source]#
Can be used only in per-tick script function to define a dynamic tick local variable.
Dynamic ticks can be used with some methods of all tick sequences.
Note
Note that
onetick.py.static
value is returned. You should not define tick variable as static manually.Examples
>>> def fun(tick): ... t = otp.dynamic_tick() ... t['X'] = tick['SUM']