Addict dictionary enhanced with ability to track changes -- currently only supports field additions
Project description
addict-tracking-changes
Introduction
Originally, this repository was a fork of addict by Mats Julian Olsen. Overtime, it has substatially diverged in functionality and codebase that it seemed to make sense to breakout as its own repository.
The original addict: provides an alternative and succient interface to manipulate a dictionary. This is especially useful when dealing with heavily nested dictionaries. As example (taken from https://github.com/mewwts/addict) a dictionary created using standard python dictionary interface looks as follows:
body = {
'query': {
'filtered': {
'query': {
'match': {'description': 'addictive'}
},
'filter': {
'term': {'created_by': 'Mats'}
}
}
}
}`
can be summarized to following three lines:
body = Dict()
body.query.filtered.query.match.description = 'addictive'
body.query.filtered.filter.term.created_by = 'Mats'
This repo builds on original addict and adds contraptions to track key additions in the dictionary. This features comes in quite handy in building reactive webapps where one has to respond to all the changes made on the browser. Addict-tracking-changes is the underpinning data-structure in ofjustpy: a python based webframework build from justpy
The functions relevant to tracking changed history are:
get_changed_history
and clear_changed_history
.
The get_changed_history
returns an iterator of XPath style paths like /a/b/c/e
(see Demo example).
Usage and examples
Installation
This project is not on PyPI. Its a simple package with no third party dependency. Simply clone from github and include the source directory in your PYTHONPATH.
Demo example
from addict import Dict
body = Dict()
body.query.filtered.query.match.description = 'addictive'
body.query.filtered.filter.term.created_by = 'Mats'
for changed_path in body.get_changed_history():
#<work with changed_path>
body.clear_changed_history()
Behaviour when values are instances of container types
addict works as expected when the values of keys are simple data types (such as string, int, float, etc.). However, for container types such as dict, list, tuples, etc. the behaviour is somewhat differs.
- dicts are treated as opaque object just like simple data types
from addict import Dict
mydict = Dict()
mydict.a.b.c = {'kk': 1}
mydict.a.b.e = {'dd': 1}
for _ in mydict.get_changed_history():
print(_)
will print
/a/b/c
/a/b/e
and not
/a/b/cc/kk
/a/b/e/dd
- lists
are seen as container, i.e.,
get_changed_history
will report path for each element of the list
from addict import Dict
mydict = Dict()
mydict.a.b = [1, [1]]
mydict.a.c = [2, [2, [3]]]
get_changed_history will report following paths:
/a/b/0,
/a/b/1/0,
/a/c/0,
/a/c/1/0,
/a/c/1/1/"
- tuple, namedtuple, sets tuple behave same as dict and are treated as opaque data structure
Known bugs and Caveats
- Only tracks field additions. Deletions and updates are not tracked.
- freeze doesn't guards against deletions
- building dict from another dict as shown in following expression wont' work
cjs_cfg = Dict(other_dict, track_changes=True)
instead use
cjs_cfg = Dict(track_changes = True)
with open("other_dict.pickle", "rb") as fh:
x = pickle.load(fh)
for _ in oj.dictWalker(x):
oj.dnew(cjs_cfg, _[0], _[1])
APIs
| API | Description | |--------------------------------------------------------+-------------------------------------------------------| | Dict(*args, **kwargs) | Initializes a new Dict object | | to_dict(self) | Converts the Dict object to a regular dictionary | | freeze(self, shouldFreeze=True) | Freezes the Dict object, making it immutable | | unfreeze(self) | Unfreezes the Dict object, making it mutable again | | get_changed_history(self, prefix="", path_guards=None) | Returns an iterator with the changed history of keys | | clear_changed_history(self) | Clears the changed history of the Dict object | | set_tracker(self, track_changes=False) | Sets or resets the change tracker for the Dict object | |--------------------------------------------------------+-------------------------------------------------------|
EndNotes
-
Docs (in readthedocs format): https://monallabs-org.github.io/addict-tracking-changes/#introduction, https://webworks.monallabs.in/addict-tracking-changes
-
Developed By: webworks.monallabs.in
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
Built Distribution
File details
Details for the file addict-tracking-changes-1.0.0.tar.gz
.
File metadata
- Download URL: addict-tracking-changes-1.0.0.tar.gz
- Upload date:
- Size: 48.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: python-requests/2.30.0
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | a0f5a611f2952a7bfb56f0cdb4ebb2e3e8d1f90fdbd9c0c79191037ac12e2fdb |
|
MD5 | 5be02ccbfa82b7be76ff6dbbfc34aaf8 |
|
BLAKE2b-256 | a48866c34bbb368d7d6c5661105f001cbde8b87e9e6832a5214f8b4078cf9cc6 |
File details
Details for the file addict_tracking_changes-1.0.0-py2.py3-none-any.whl
.
File metadata
- Download URL: addict_tracking_changes-1.0.0-py2.py3-none-any.whl
- Upload date:
- Size: 8.4 kB
- Tags: Python 2, Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: python-requests/2.30.0
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | c4bc37cecb488a5ff552f280a0bb57441d079d4c6e1e143b2005d34189b5ce24 |
|
MD5 | 87e883c80b596b4122a11c8de2eae148 |
|
BLAKE2b-256 | 1b622a2491029eef0873a661372c14434fb4e53b19a8b1ee3e39c501932cdc54 |