Assorted utility functions to support working with SQLAlchemy.
Project description
Latest release 20190830: @json_column: small docstring improvement.
Assorted utility functions to support working with SQLAlchemy.
Function auto_session(func)
Decorator to run a function in a session if one is not presupplied.
The function func
runs within a transaction,
nested if the session already exists.
Function find_json_field(column_value, field_name, *, infill=False)
Descend a JSONable Python object column_value
to field_name
.
Return column_value
(possibly infilled), final_field
, final_field_name
.
This supports database row columns which are JSON columns.
Parameters:
column_value
: the original value of the columnfield_name
: the field within the column to locateinfill
: optional keyword parameter, defaultFalse
. If true,column_value
and its innards will be filled in asdict
s to allow deferencing thefield_name
.
The field_name
is a str
consisting of a period ('.'
) separated sequence of field parts.
Each field part becomes a key to index the column mapping.
These keys are split into the leading field parts
and the final field part,
which is returned as final_field_name
above.
The final_field
return value above
is the mapping within which final_field_value
may lie
and where final_field_value
may be set.
Note: it may not be present.
If a leading key is missing and infill
is true
the corresponding part of the column_value
is set to an empty dictionary
in order to allow deferencing the leading key.
This includes the case when column_value
itself is None
,
which is why the column_value
is part of the return.
If a leading key is missing and infill
is false
this function will raise a KeyError
for the portion of the field_name
which failed.
Examples:
>>> find_json_field({'a':{'b':{}}}, 'a.b')
({'a': {'b': {}}}, {'b': {}}, 'b')
>>> find_json_field({'a':{}}, 'a.b')
({'a': {}}, {}, 'b')
>>> find_json_field({'a':{'b':{}}}, 'a.b.c.d')
Traceback (most recent call last):
...
KeyError: 'a.b.c'
>>> find_json_field({'a':{'b':{}}}, 'a.b.c.d', infill=True)
({'a': {'b': {'c': {}}}}, {}, 'd')
>>> find_json_field(None, 'a.b.c.d')
Traceback (most recent call last):
...
KeyError: 'a'
>>> find_json_field(None, 'a.b.c.d', infill=True)
({'a': {'b': {'c': {}}}}, {}, 'd')
Function get_json_field(column_value, field_name, *, default=None)
Return the value of field_name
from column_value
or a defaault if the field is not present.
Parameters:
column_value
: the original value of the columnfield_name
: the field within the column to locatedefault
: default value to return if the field is not present, default:None
Examples:
>>> get_json_field({'a': 1}, 'a')
1
>>> get_json_field({'b': 1}, 'a')
>>> get_json_field({'a': {}}, 'a.b')
>>> get_json_field({'a': {'b': 2}}, 'a.b')
2
Function json_column(*da, **dkw)
Class decorator to declare a virtual column name on a table where the value resides inside a JSON column of the table.
Parameters:
cls
: the class to annotateattr
: the virtual column name to present as a row attributejson_field_name
: the field within the JSON column used to store this value, default the same asattr
json_column_name
: the name of the associated JSON column, default'info'
default
: the default value returned by the getter if the field is not present, defaultNone
Example use:
Base = declarative_base()
...
@json_column('virtual_name', 'json.field.name')
class TableClass(Base):
...
This annotates the class with a .virtual_name
property
which can be accessed or set,
accessing or modifying the associated JSON column
(in this instance, the column info
,
accessing info['json']['field']['name']
).
Function log_level(*da, **dkw)
Decorator to run func
at the specified logging level
, default logging.DEBUG
.
Class ORM
MRO: cs.resources.MultiOpenMixin
A convenience base class for an ORM class.
This defines a .Base
attribute which is a new DeclarativeBase
and provides various Session related convenience methods.
It is also a MultiOpenMixin
subclass
supporting nested open/close sequences and use as a context manager.
Subclasses must define the following:
.Session
: a factory in their own__init__
, typicallyself.Session=sessionmaker(bind=engine)
.startup
and.shutdown
methods to support theMultiOpenMixin
, even if these justpass
Function orm_auto_session(method)
Decorator to run a method in a session derived from self.orm
if a session is not presupplied.
Intended to assist classes with a .orm
attribute.
Function push_log_level(level)
Temporarily set the level of the default SQLAlchemy logger to level
.
Yields the logger.
NOTE: this is not MT safe - competing Threads can mix log levels up.
Function set_json_field(column_value, field_name, value, *, infill=False)
Set a new value
for field_name
of column_value
.
Return the new column_value
.
Parameters:
column_value
: the original value of the columnfield_name
: the field within the column to locatevalue
: the value to store asfield_name
infill
: optional keyword parameter, defaultFalse
. If true,column_value
and its innards will be filled in asdict
s to allow deferencing thefield_name
.
As with find_json_field
,
a true infill
may modify column_value
to provide field_name
which is why this function returns the new column_value
.
Examples:
>>> set_json_field({'a': 2}, 'a', 3)
{'a': 3}
>>> set_json_field({'a': 2, 'b': {'c': 5}}, 'b.c', 4)
{'a': 2, 'b': {'c': 4}}
>>> set_json_field({'a': 2}, 'b.c', 4)
Traceback (most recent call last):
...
KeyError: 'b'
>>> set_json_field({'a': 2}, 'b.c', 4, infill=True)
{'a': 2, 'b': {'c': 4}}
>>> set_json_field(None, 'b.c', 4, infill=True)
{'b': {'c': 4}}
Function with_session(func, *a, orm=None, session=None, **kw)
Call func(*a,session=session,**kw)
, creating a session if required.
The function func
runs within a transaction,
nested if the session already exists.
This is the inner mechanism of @auto_session
and
ORM.auto_session_method
.
Parameters:
func
: the function to calla
: the positional parametersorm
: optional ORM class with a.session()
context manager method such as theORM
base class supplied by this module.session
: optional existing ORM session
One of orm
or session
must be not None
; if session
is None
then one is made from orm.session()
and used as
a context manager.
The session
is also passed to func
as
the keyword parameter session
to support nested calls.
Release Log
Release 20190830: @json_column: small docstring improvement.
Release 20190829: Bugfix @json_column setter: mark the column as modified for the ORM. New push_log_level context manager and @log_level decorator to temporarily change the SQLAlchemy logging handler level.
Release 20190812:
Make ORM a MultiOpenMixin.
get_json_field: use forgotten default
parameter.
Other minor changes.
Release 20190526: Support for virtual columns mapped to a JSON column interior value: New functions find_json_field, get_json_field, set_json_field. New decorator @json_column for declaritive_base tables.
Release 20190517: Make ORM._Session private session factory the public ORM.Session factory for external use. with_session: preexisting sessions still trigger a session.begin_nested, removes flush/commit tension elsewhere.
Release 20190403: Rename @ORM.orm_auto_session to @ORM.auto_session. New @orm_auto_session decorator for methods of objects with a .orm attribute.
Release 20190319.1: Initial release. ORM base class, @auto_session decorator.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Hashes for cs.sqlalchemy_utils-20190830.tar.gz
Algorithm | Hash digest | |
---|---|---|
SHA256 | 0eb8ac36a0284962a00aa2fc760d19e6036a2606121f5875551775c1d7ff8e76 |
|
MD5 | 144da6d3d582c9615a5f0844f038f10e |
|
BLAKE2b-256 | 5bca25d45cc1db917c4343ab3792e993010165c259e1570c0167b046597fc6b2 |