Skip to content

Latest commit

 

History

History
167 lines (149 loc) · 6.43 KB

Patterns JSON and DSV Output.md

File metadata and controls

167 lines (149 loc) · 6.43 KB
layout title nav_order parent permalink
default
JSON and DSV Output
5
Design Patterns
/patterns/json-sql-output

Using JSON, rather than a record set, for encoding the returned data may also be more efficient due to the reduced number of required API calls. While SQLite API is usually fast, each returned value still costs several API calls. (Under certain circumstances, however, there might be a high SQLite-independent overhead for each API call.) Another important consideration is whether there are any potential side effects of data conversion between numeric and textual formats. Let us show a few examples.

Consider a modified query from the Surrogate Variables section:

WITH
    delimiters(delimiter) AS (VALUES ('/')),
    strings(string_id, string) AS (
        VALUES
            ('abc', 'C:/Winows/System32/drivers/etc/'),
            ('def', 'C:/Users/Public/Desktop')
    ),
    folders AS (
        SELECT string_id, "terms"."key" AS term_id, "terms"."value" AS term
        FROM
            delimiters, strings,
            json_each('["' || replace(trim(string, delimiter), delimiter, '", "') || '"]') AS terms
        ORDER BY string_id, term_id
    ),
    json_folders AS (
        SELECT string_id, json_group_array(term) AS path_json
        FROM folders
        GROUP BY string_id
    )
SELECT * FROM json_folders;

which outputs:

string_id path_json
abc ["C:","Winows","System32","drivers","etc"]
def ["C:","Users","Public","Desktop"]

This query has a new section, json_folders, at the end, which uses json_group_array to collect folders belonging to the same path. Note that the ordering clause is added to the folders section to ensure that the order of individual folders within JSON arrays after grouping reflects their positions in original paths.


Similarly, the following modified query:

WITH
    folders AS (
        SELECT
            json_extract(dirs.value, '$.bin_id') AS bin_id,
            json_extract(dirs.value, '$.prefix') AS prefix,
            json_extract(dirs.value, '$.name')   AS name
        FROM
            json_each(
                '['                                                                                    ||
                    '{"bin_id": "239", "prefix": "C:/Winows/System32/drivers/etc", "name": "hosts"},'  ||
                    '{"bin_id": "876", "prefix": "C:/Users/Public/Desktop",        "name": "pic"  },'  ||
                    '{"bin_id": "374", "prefix": "C:/Users/Default/Music",         "name": "drum" }'   ||
                ']'
            ) AS dirs
    ),
    json_fs_objects AS (
        SELECT
            json_group_array(json_object('bin_id', bin_id, 'prefix', prefix, 'name', name)) AS fs_objects
        FROM folders
    )
SELECT * FROM json_fs_objects;

return a scalar string containing a set of records in the JSON format:

[
    {"bin_id": "239", "name": "hosts", "prefix": "C:/Winows/System32/drivers/etc"},
    {"bin_id": "876", "name": "pic",   "prefix": "C:/Users/Public/Desktop"       },
    {"bin_id": "374", "name": "drum",  "prefix": "C:/Users/Default/Music"        }
]

JSON objects are a bit too verbose when returning a record set, but there are a few other options. For example:

WITH
    folders AS (
        SELECT
            json_extract(dirs.value, '$.bin_id') AS bin_id,
            json_extract(dirs.value, '$.prefix') AS prefix,
            json_extract(dirs.value, '$.name')   AS name
        FROM
            json_each(
                '['                                                                                    ||
                    '{"bin_id": "239", "prefix": "C:/Winows/System32/drivers/etc", "name": "hosts"},'  ||
                    '{"bin_id": "876", "prefix": "C:/Users/Public/Desktop",        "name": "pic"  },'  ||
                    '{"bin_id": "374", "prefix": "C:/Users/Default/Music",         "name": "drum" }'   ||
                ']'
            ) AS dirs
    ),
    tsv_fs_objects AS (
        SELECT group_concat(bin_id || x'09' || prefix || x'09' || name, x'0A') AS fs_objects
        FROM folders
    )
SELECT * FROM tsv_fs_objects;

or

WITH
    folders AS (
        SELECT
            json_extract(dirs.value, '$.bin_id') AS bin_id,
            json_extract(dirs.value, '$.prefix') AS prefix,
            json_extract(dirs.value, '$.name')   AS name
        FROM
            json_each(
                '['                                                                                    ||
                    '{"bin_id": "239", "prefix": "C:/Winows/System32/drivers/etc", "name": "hosts"},'  ||
                    '{"bin_id": "876", "prefix": "C:/Users/Public/Desktop",        "name": "pic"  },'  ||
                    '{"bin_id": "374", "prefix": "C:/Users/Default/Music",         "name": "drum" }'   ||
                ']'
            ) AS dirs
    ),
    tsv_fs_objects AS (
        SELECT
            group_concat(printf('%s' || x'09' || '%s' || x'09' || '%s', bin_id, prefix, name), x'0A') AS fs_objects
        FROM folders
    )
SELECT * FROM tsv_fs_objects;

or

WITH
    folders AS (
        SELECT
            json_extract(dirs.value, '$.bin_id') AS bin_id,
            json_extract(dirs.value, '$.prefix') AS prefix,
            json_extract(dirs.value, '$.name')   AS name
        FROM
            json_each(
                '['                                                                                    ||
                    '{"bin_id": "239", "prefix": "C:/Winows/System32/drivers/etc", "name": "hosts"},'  ||
                    '{"bin_id": "876", "prefix": "C:/Users/Public/Desktop",        "name": "pic"  },'  ||
                    '{"bin_id": "374", "prefix": "C:/Users/Default/Music",         "name": "drum" }'   ||
                ']'
            ) AS dirs
    ),
    records AS (
            SELECT 
                'bin_id' || x'09' || 'prefix' || x'09' || 'name' || x'0A' ||
                'str'    || x'09' || 'str'    || x'09' || 'str' AS record
        UNION ALL
            SELECT bin_id || x'09' || prefix || x'09' || name AS record
            FROM folders
    ),
    tsv_fs_objects AS (
        SELECT group_concat(record, x'0A') AS fs_objects
        FROM records
    )
SELECT * FROM tsv_fs_objects;

return a scalar string containing a set of records in the tab-separated value format. The last query also adds a table header.