A generic, unlimited level json normalizer which handles arrays at any location. Outputs a normalised set of data which can be imported into a relational table
- This outputs relational data from pretty much and analytics focuse json
- It'll create new entities for data which should be split out into a new table
- It'll link those entries by a set list of ID keys you specify
- All your join keys are yours, so you won't get random breakages if you switch tools etc
If you're reporting within a relational Database, or have something in the pipeline which doesn't like structured data, this tool will do the heavy lifting to relationalise the json for you. This will save you and your teams time doing the work of unnesting objects and pulling arrays into new tables so you can query them without the weird mappings that forced unnesting offers
- Free software: MIT license
- Documentation: https://super-json-normalize.readthedocs.io.
This works well as a transform step for your ETL (or even ELT)
- Extract your data
- Iterate through each entry
- Run
super_json_normalize.normalize_record(<your_dict_here>)
on the entry - pipe the output arrays of entries to your needed location
- TODO
- initial dumb unnesting with
id
as the Primary key - unnesting with configurable Primary Keys
- multiple Primary keys for parent
- multi-layerd primary keys (eg entities inside entities)
This works by pulling out arrays into their own entities.
for the following example payload properties
for a house (in directory /samples/property_data/):
//for the file property_data.json
{
"id": "ID001",
"address": {
"street_address": "123 Fake street",
"suburb": "Fakeland",
"state": "VIC",
"country": "Australia"
},
"inspection_times": [
{"id": "IID001", "description":"First inspection date on Sunday"},
{"id": "IID002", "description":"Second inspection date on Tuesday"},
{"id": "IID003", "description":"Final inpection date on Friday"}
]
}
The following will occur when we run super_json_normalize.normalize_record()
on the loaded json data:
- the
address
object will be unnested - a new entitiy called
inspection_times
will be created, and theinspection_times
array will be removed from the root entity - the parent
id
from the root will be pulled into theinspection_times
entity so we can join the data
this would end up with:
[
{
"name": "properties", // `properties` entity
"data": [
{
"id": "ID001", // primary key
"address_street_address": "123 Fake street",
"address_suburb": "Fakeland",
"address_state": "VIC",
"address_country": "Australia"
}
]
},
{
"name": "inspection_times", // `inspection_times` entity
"data": [
{
"id": "IID001",
"description": "First inspection date on Sunday",
"properties_id": "ID001" // parent key (foreign key) inherited from the root
},
{
"id": "IID002",
"description": "Second inspection date on Tuesday",
"properties_id": "ID001" // parent key (foreign key) inherited from the root
},
{
"id": "IID003",
"description": "Final inpection date on Friday",
"properties_id": "ID001" // parent key (foreign key) inherited from the root
}
]
}
]
This is useful for preparing data for a relational db or systems requiring relational data.
Example to dump each entity to json from the sample above from /samples/
This one is the property_data
example:
import super_json_normalize
import json
#load the data from file
my_sample_data = super_json_normalize.load_json("./samples/property_data.json")
#now create the entities in a json file
my_output_data = super_json_normalize.normalize_record(my_sample_data, parent_name="properties")
#finally, export the entities in each individual file
super_json_normalize.export_records(my_normalized_data, path="./export_data/", format="jsonl")
###
# creates files in ./export_data/ directory
# properties.jsonl
# inspection_times.jsonl
### properties.jsonl
# {"id": "ID001", "address_street_address": "123 Fake street", "address_suburb": "Fakeland", "address_state": "VIC", "address_country": "Australia"}
### inspection_times.jsonl
# {"id": "IID001", "description": "First inspection date on Sunday", "properties_id": "ID001"}
# {"id": "IID002", "description": "Second inspection date on Tuesday", "properties_id": "ID001"}
# {"id": "IID003", "description": "Final inpection date on Friday", "properties_id": "ID001"}
This package was created with Cookiecutter_ and the audreyr/cookiecutter-pypackage
_ project template.
.. _Cookiecutter: https://github.com/audreyr/cookiecutter
.. _audreyr/cookiecutter-pypackage
: https://github.com/audreyr/cookiecutter-pypackage