Raku package for translation of JSON specs or JSON-like data structures into other formats.
It is envisioned this package to have translators to multiple formats. For example:
- DONE HTML
- DONE JSON
- DONE R
- DONE WL
- TODO Plain text
- TODO Python
- TODO Mermaid-JS
- TODO Julia
- TODO SQL
The main motivation for making the package is to have convenient way of making tables while doing Literate programming with Raku using:
- Computational Markdown documents, [AAp4]
- Jupyter notebooks, [BDp1]
- Mathematica notebooks, [AAp4]
The use of JSON came to focus, since when working Large Language Model (LLM) functions, [AAp3], very often it is requested from LLMs to produce output in JSON format, [AA1, AA2].
The package "Data::Reshapers", [AAp1], would complement nicely "Data::Translators" and vice versa. The package "Data::TypeSystem", [AAp2], is used for "translation decisions" and for conversions into more regular datasets.
The package "Mathematica::Serializer", [AAp5], has very similar mission -- it is for translating Raku data structures into Mathematica (aka Wolfram Language or WL) code.
In order to utilize "Data::Translators" while doing Literate programming with:
- Computational Markdown files, then use the code chunk argument
results=asis
. - Jupyter notebooks, then use the code cell magic spec
%%html
or%% > html
.
One can find concrete examples for:
- Computational Markdown files, in the raw source code of this README
- Jupyter notebooks, in the magics examples notebook at [BDp1]
Remark: The provided converters are made for communication purposes, so they might not be very performant. I have used or tested them with datasets that have less than 5000 rows.
Package installations from both sources use zef installer (which should be bundled with the "standard" Rakudo installation file.)
To install the package from Zef ecosystem use the shell command:
zef install Data::Translators
To install the package from the GitHub repository use the shell command:
zef install https://github.com/antononcube/Raku-JSON-Translators.git
Here is a "main use case" example:
- Get a dataset that is an array of hashes
- Filter or sample the records
- Make an HTML table with those records
The HTML table outputs can be used to present datasets nicely in:
- Markdown documents
- Jupyter notebooks
Here we get the Titanic dataset and sample it:
use Data::Reshapers;
use Data::TypeSystem;
use Data::Translators;
my $tbl = get-titanic-dataset.pick(3);
# ({id => 258, passengerAge => 40, passengerClass => 1st, passengerSex => female, passengerSurvival => survived} {id => 1211, passengerAge => 40, passengerClass => 3rd, passengerSex => male, passengerSurvival => died} {id => 1227, passengerAge => 20, passengerClass => 3rd, passengerSex => male, passengerSurvival => died})
Here is the corresponding dataset type:
deduce-type($tbl);
# Vector(Assoc(Atom((Str)), Atom((Str)), 5), 3)
Here is the corresponding HTML table:
$tbl ==> data-translation
passengerClass | passengerSurvival | id | passengerAge | passengerSex |
---|---|---|---|---|
1st | survived | 258 | 40 | female |
3rd | died | 1211 | 40 | male |
3rd | died | 1227 | 20 | male |
We can specify field names and HTML table attributes:
$tbl ==> data-translation(field-names => <id passengerSurvival>, table-attributes => 'id="info-table" class="table table-bordered table-hover" text-align="center"');
id | passengerSurvival |
---|---|
258 | survived |
1211 | died |
1227 | died |
Here is how the transposed dataset is tabulated:
$tbl ==> transpose() ==> data-translation;
id |
|
---|---|
passengerSurvival |
|
passengerClass |
|
passengerSex |
|
passengerAge |
|
Here is a JSON string translation to HTML:
my $json1 = q:to/END/;
{
"sample": [
{"name": "json2html", "desc": "coverts json 2 html table format", "lang": "python"},
{"name": "testing", "desc": "clubbing same keys of array of objects", "lang": "python"}
]
}
END
data-translation($json1);
sample |
|
---|
Here is a more involved data example:
data-translation(cross-tabulate(get-titanic-dataset, 'passengerSex', 'passengerSurvival'))
male |
| ||||
---|---|---|---|---|---|
female |
|
Compare the HTML table above with the following plain text table:
to-pretty-table(cross-tabulate(get-titanic-dataset, 'passengerSex', 'passengerSurvival'))
# +--------+----------+------+
# | | survived | died |
# +--------+----------+------+
# | female | 339 | 127 |
# | male | 161 | 682 |
# +--------+----------+------+
Here is the R code version of the Titanic data sample:
$tbl ==> data-translation(target => 'R', field-names => <id passengerClass passengerSex passengerAge passengerSurvival>)
data.frame(`passengerClass` = c("1st", "3rd", "3rd"),
`passengerSurvival` = c("survived", "died", "died"),
`id` = c("258", "1211", "1227"),
`passengerAge` = c("40", "40", "20"),
`passengerSex` = c("female", "male", "male"))
Here is the R code version of the contingency table:
data-translation(cross-tabulate(get-titanic-dataset, 'passengerSex', 'passengerSurvival'), target => 'R')
list("male"=list("died"=682, "survived"=161), "female"=list("survived"=339, "died"=127))
Here is the WL code version of the contingency table:
data-translation(cross-tabulate(get-titanic-dataset, 'passengerSex', 'passengerSurvival'), target => 'WL')
Association["female"->Association["survived"->339, "died"->127], "male"->Association["survived"->161, "died"->682]]
In order to obtain datasets or more regular datasets the function to-dataset
can be used.
Here a rugged dataset is made regular and converted to an HTML table:
my @tbl2 = get-titanic-dataset.pick(6);
@tbl2 = @tbl2.map({ $_.pick((1..5).pick).Hash });
@tbl2 ==> to-dataset(missing-value=>'・') ==> data-translation
passengerClass | passengerAge | passengerSurvival | passengerSex | id |
---|---|---|---|---|
3rd | 30 | survived | ・ | 881 |
3rd | 10 | survived | female | 1258 |
2nd | ・ | ・ | ・ | ・ |
3rd | 10 | died | female | 808 |
・ | ・ | ・ | male | ・ |
1st | 20 | survived | female | ・ |
Here a hash is transformed into dataset with columns <Key Value>
and then converted into an HTML table:
{ 4 => 'a', 5 => 'b', 8 => 'c'} ==> to-dataset() ==> data-translation
Value | Key |
---|---|
c | 8 |
b | 5 |
a | 4 |
- The "need" for this package became evident while working on the notebooks/articles [AA1, AA2].
- Initially, I translated plain text tables into HTML.
- Using LLMs or
md-interpret
provided by "Markdown::Grammar".
- Using LLMs or
- I considered re-using the code behind
to-pretty-table
provided by "Data::Reshapers", [AAp1].- This is "too much work" and I wanted a lighter weight package.
- Having a solution for the more general problem translating JSON to HTML seemed a much better and easier option.
- For example, I hoped that someone has already solved that problem for Raku.
- Since I did not find Raku packages for the translation I wanted, I looked for solutions into the Python ecosystem.
- ... And found "json2html".
- Using ChatGPT-4.0 I translated the only class of that package from Python into Raku.
- The obtained translation could be executed with relatively minor changes.
- I further refactored and enhanced the HTML translator to fit my most frequent Raku workflows.
- The ingestion of JSON strings is done with the package "JSON::Fast".
- Hence the conversion to JSON "comes for free" using
to-json
from that package.
- Hence the conversion to JSON "comes for free" using
- The initial versions of the package did not have the "umbrella" function
data-translation
.- Only the "lower level" functions
json-to-html
andjson-to-r
were provided.
- Only the "lower level" functions
- The "lower level" functions, or shortcuts, can be used:
to-html
,to-r
,to-wl
.
The package provides a Command Line Interface (CLI) script. Here is its usage message:
data-translation --help
# Usage:
# data-translation <data> [-t|--target=<Str>] [--encode] [--escape] [--field-names=<Str>] -- Convert data into another format.
#
# <data> Data to convert.
# -t|--target=<Str> Target to convert to, one of <JSON HTML R>. [default: 'HTML']
# --encode Whether to encode or not. [default: False]
# --escape Whether to escape or not. [default: False]
# --field-names=<Str> Field names to use for Map objects, separated with ';'. [default: '']
Here is an example application (to this file):
data-translation ./resources/professionals.json --field-names='data;id;name;age;profession'
data |
|
---|
[AA1] Anton Antonov, "Workflows with LLM functions", (2023), RakuForPrediction at WordPress.
[AA2] Anton Antonov, "TLDR LLM solutions for software manuals", (2023), RakuForPrediction at WordPress.
[AAp1] Anton Antonov, Data::Reshapers Raku package, (2021-2023), GitHub/antononcube.
[AAp2] Anton Antonov, Data::TypeSystem Raku package, (2023), GitHub/antononcube.
[AAp3] Anton Antonov, LLM::Functions Raku package, (2023), GitHub/antononcube.
[AAp4] Anton Antonov, Text::CodeProcessing Raku package, (2021-2023), GitHub/antononcube.
[AAp5] Anton Antonov, Mathematica::Serializer Raku package, (2021-2022), GitHub/antononcube.
[BDp1] Brian Duggan, Jupyter:Kernel Raku package, (2017-2023), GitHub/bduggan.
[VMp1] Varun Malhotra, json2html Python package, (2013-2021), GitHub/softvar.