Simple query language to extract tables from JSON.
Project description
json_tabulator
A simple query language for extracting tables from JSON-like objects.
Working with tabular data is much easier than working with nested documents. json_tabulator helps to extract tables from JSON-like objects in a simple, declarative manner. All further processing is left to the many powerful tools that exist for working with tables, such as Spark or Pandas.
Installation
Install from pypi:
pip install json_tabulator
Quickstart
The json_tabulator module provides tools to extract a JSON document into a set of related tables. Let's start with a simple document
data = {
'id': 'doc-1',
'table': [
{'id': 1, 'name': 'row-1'},
{'id': 2, 'name': 'row-2'}
]
}
The document consists of a document-level value id as well as a nested sub-table table. We want to extract it into a single table, with the global value folded into the table.
To do this, we write a query that defines the conversion into a table like this:
from json_tabulator import tabulate
query = tabulate({
'document_id': 'id',
'row_id': 'table[*].id',
'row_name': 'table[*].name'
})
rows = query.get_rows(data)
This returns an iterator of rows, where each row is a dict {<column_name>: <value>}:
>>> list(rows)
[
{'document_id': 'doc-1', 'row_id': 1, 'row_name': 'row-1'},
{'document_id': 'doc-1', 'row_id': 2, 'row_name': 'row-2'}
]
Path Syntax
The syntax for path expressions is very similar to a subset of JSON Path. A path consists of an optional root element '$' followed by a path that specifies what is to be extracted. The child operator . and subscripts [1], ['a'], [*] can be used for arrays or dicts.
Dict key
Can be any string. Key values can be quoted with single or double quotes. Within quoted strings, the quote character must be doubled to escape it. For example, "say \"hello\"" is interpreted as say "hello".
Keys must be quoted if
- they contain any of the characters
*$.'"[](), or if - they start with a digit
- they are used in a subscript, e.g.
$["child"]is valid, but$[child]is not.
Subscripts
Subscripts are entered as [...] without a leading .. Allowed values in a subscript are
- Non-negative numbers representing array indices, e.g.
$[123] - Quoted dict keys, e.g.
$['a'] - Wildcards, e.g.
$[*]
Wildcard *
An asterisk * is interpreted as a wildcard. Iterates over dict values or array items. Note that wildcards must be entered explicitly, there is no implicit iteration over arrays. Use either $[*] or $.*.
Note that in contrast to JSON path, the wildcard can be used for dicts and arrays regardless of whether subscript is used.
Functions
Functions are written in parentheses. Currently there are two functions available:
(key)returns the index that corresponds to the preceding wildcard(path)returns the full path up to the preceding wildcard
Both functions must be placed directly after a wildcard and must be at the end of the path. For example *.(key) is valid, but a.(key) and *.(key).b are not.
The output of (path) is unique for all rows extracted from the document.
Data Extraction
Query Semantics
Values for all attributes in a query are combined into individual rows. Attributes from different parts of the document are combined by "joining" on the lowest common ancestor.
For this reason, all wildcards for all attributes must lie on a common path. Violating this condition would lead to implicit cross joins and the associated data blow-up.
For example, the paths $.a[*] and $.b[*] cannot be combined because the wildcards are not on the same path. On the other hand, $.a and $.b[*].c can be combined.
Queries are analysed and compiled independently of the data to be queried.
If you think you need to get a combination of attributes that is not allowed, think again. If you still think so just run multiple queries and do the join afterwards.
Returned values
All extracted values are returned as-is with no further conversion. Returned values are not limited to atomic types, dicts and lists are also allowed.
By default, all requested attributes are returned, the value None is used if an attribute cannot be found in the data. When the omit_missing_attributes flag is set in tabulate, each row only contains keys that are found in the data, allowing for downstream error-handling.
Related Projects
- jsontable has the same purpose but is not maintained.
- jsonpath-ng and other jsonpath implementation are much more flexible but more cumbersome to extract nested tables.
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
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file json_tabulator-0.4.0.tar.gz.
File metadata
- Download URL: json_tabulator-0.4.0.tar.gz
- Upload date:
- Size: 6.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.8.5 CPython/3.9.7 Darwin/22.5.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
670a6ec7c959a6e22587eb192a63e2d10b34fabc87ccdaac31f44fd6a5af8196
|
|
| MD5 |
c3472f6144157668dc8abf1176d218b1
|
|
| BLAKE2b-256 |
5f9bd796b43d4ab1ce9ae2296190d9d6f7e1e92cbe22e3fc01c4692d7b476a71
|
File details
Details for the file json_tabulator-0.4.0-py3-none-any.whl.
File metadata
- Download URL: json_tabulator-0.4.0-py3-none-any.whl
- Upload date:
- Size: 7.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.8.5 CPython/3.9.7 Darwin/22.5.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b4666a31bed550195512b8716da847b66c66371d1a26405591dbd367ad56433a
|
|
| MD5 |
d65ec29a36428b5b4d8f69e3593108c1
|
|
| BLAKE2b-256 |
27f6440131224066f29391c362a6ddade4f8abb8f813afd10a4fa48c82175c0e
|