Skip to main content

SQLAlchemy hierarchical key/value helper

Project description

metalchemy: SQLAlchemy hierarchical key/value helper

The metalchemy package provides helpers for your SQLAlchemy models to add dynamic properties.


pip install metalchemy


metalchemy usage example:

from sqlalchemy.ext.declarative import declarative_base

Base = declarative_base()

import metalchemy

metalchemy_attributes = metalchemy.initialize(Base)

Session = sessionmaker(bind=engine)
sess = Session()

class MyModel(Base)

    meta = metalchemy_attributes.Metadata()

my_object = MyModel()
my_object.meta.some.value = 'some value'

assert  my_object.meta.some.value.get_value() == 'some value'

In order to give a class metadata capabilities, add a single class attribute to it which is an instance of <Metadata>:

class HasMetadata(object):
    meta = metadata.Metadata()

Any instance of this class will now have its metadata accessible via the meta attribute.

Such meta attributes allow free reading and assigning of attributes, with no limits on the depth of the attributes. i.e., is always available for reading and writing, but also, without any setup beforehand.

Assigning to a metadata property is simple and obvious: = 42 = 'qux'

Any metadata attribute is also implicitly an array. It is possible to assign and read from any index:[0] = 42[1] = 'baz'[1].bar = 'qux'[1].spam[2] = 'eggs'[1][2] = 'xyzzy'

Any non-indexed attribute is implicitly converted to an index of zero.

All metadata values are converted to unicode strings on assignment. Assigned values are automatically added to the SQLAlchemy session, but not committed, so remember to execute session.commit(). Furthermore, the methods FieldWrapper.append, FieldWrapper.iteritems and FieldWrapper.__iter__ are supported as well for direct iteration.

Reading the value back requires using get_value method:  # returns 42  # returns u'qux'

Internals: The hierarchical structure of the fields is stored in an adjacency list (represented by <_Fields>), which is unique for a single class. All instances of a class share this same tree. An <_Object> maps the class name to this tree, and is set up to have the entire tree load at once when it is needed.

The <Metadata> instance assigned to a container class will load the <_Object> (and implicitly, the field hierarchy) on access and return a wrapped root node.

<FieldWrappers> wrap each <_Field>, performing two functions: - They allow accessing fields that have no concrete <_Field> instance yet, creating these as necessary - They actually access the values list of a specific container class instance.

Values are stored in a flat list, which is loaded entirely for the container class instane when it is first needed.

Array support is handled by having two states of FieldWrappers: regular and indexed.

A regular wrapper wraps an unindexed attribute. It handles reading and writing of array elements on its attribute (FieldWrapper.__getitem__ and FieldWrapper.__setitem__), and defers attribute access to its own zeroth index.

An indexed wrapper wraps an indexed attribute. This wrapper handles attribute access by returning a wrapper for that child attribute. Indexed wrappers can also be indexed, but this is internally done by deferring the secondary index to a hidden child attribute. i.e. accessing[0][1][2] is internally handled as accessing[0].<hidden>[1].<hidden>[2].


If you have questions, bug reports, suggestions, etc. please create an issue on the GitHub project page.


This software is licensed under the MIT license

See License

© 2014 Paylogic International.



  • Initial public release

Project details

Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

metalchemy-1.0.0.tar.gz (10.4 kB view hashes)

Uploaded Source

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page