#' Unnest a JSON column
#'
#' A wrapper around [tidyr::unnest()] that extracts its data from a JSON column.
#' The inverse of [json_nest()].
#'
#' @param data A data frame, a data frame extension (e.g. a tibble), or a lazy
#' data frame (e.g. from dbplyr or dtplyr).
#' @inheritParams rlang::args_dots_used
#' @inheritParams tidyr::unnest
#'
#' @return An object of the same type as `data`
#' @export
#' @examples
#' tibble(a = 1, b = '[{ "c": 2 }, { "c": 3 }]') %>%
#' json_unnest(b)
json_unnest <- function(data, cols, ..., names_sep = NULL, names_repair = "check_unique") {
check_dots_used()
UseMethod("json_unnest")
}
#' @export
json_unnest.data.frame <- function(data, cols, ..., keep_empty = FALSE, ptype = NULL,
names_sep = NULL, names_repair = "check_unique") {
if (missing(cols)) {
abort("The `cols` argument must be provided")
}
data %>%
mutate(across({{ cols }}, ~ map(., ~ jsonlite::fromJSON(.) %>% as_tibble()))) %>%
unnest(
{{ cols }},
keep_empty = keep_empty,
ptype = ptype,
names_sep = names_sep,
names_repair = names_repair
)
}
# @export
# json_unnest.tbl_lazy <- function(data, cols, ..., keep_empty = FALSE, ptype = NULL,
# names_sep = NULL, names_repair = "check_unique") {
# select string_agg(quotename(k) + case t
# when 0 then ' nchar(1)' -- javascript null
# when 1 then ' nvarchar(max)' -- javascript string
# when 2 then ' float' -- javascript number
# when 3 then ' bit' -- javascript boolean
# else ' nvarchar(max) as json' -- javascript array or object
# end, ', ') within group (order by k)
# from (
# select j2.[key], max(j2.[type])
# from test
# cross apply openjson(case when json_col like '{%}' then '[' + json_col + ']' else json_col end) as j1
# cross apply openjson(j1.value) as j2
# group by j2.[key]
# ) as kt(k, t)
# }
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.