Passing a row type dynamically to jsonb_populate_recordset

1.3k Views Asked by At

I'm working on some code to unwind data packed into a JSONB with a regular format for passthrough to another system.

-- Postgres 11.4.
-- The format of the JSON is knowable in advance, it's specified in the record.
-- I've got a custom type for the JSON format defined.

Here's the stripped down passthrough table:

CREATE TABLE IF NOT EXISTS data.domo_passthrough (
    id           uuid NOT NULL DEFAULT extensions.gen_random_uuid (),
    type_name    text NOT NULL DEFAULT NULL, 
    report_data  jsonb NOT NULL DEFAULT '{}',

CONSTRAINT domo_passthrough_id_pkey
    PRIMARY KEY (id)
);

The report_data field holds the packed data, the type_name field includes the name of the format, like this one:

create type data.table_stats_type as (
    table_name text, 
    record_count integer,
    table_number integer)

This code works properly to unpack the data into the three columns defined in the type:

select expanded_data.*
  from domo_passthrough,
       jsonb_populate_recordset(null::table_stats_type, report_data) as expanded_data;

What I'd like to do, if possible, is to pass in the type_name as an argument to jsonb_populate_recordset:

select expanded_data.*
  from domo_passthrough,
       jsonb_populate_recordset(null::type_name, report_data) as expanded_data;

This does not work as null::type_name returns an error:

ERROR:  type "type_name" does not exist
LINE 3:        jsonb_populate_recordset(null::type_name, report_data..

Fair. The type_name is a text, not a literal. I'm hoping that there is a tidy way to do what I'm after. (I understand that it's on me at that point to not screw up the type name assignments in the data.) Possibilities:

-- Some kind of ::casting operation, but I can't figure out what type I'd use between null:: and type_name. I've found the pg_typeof function, but don't see a way to apply it to build in code.

-- Dynamic SQL in a function. This seems like it should be possible, but I'm hoping for something simpler.

-- Some other feature or option I'm unaware of.

RETURN SETOF Function?

Thanks for the comment, I figured I'd try writing a function...and I got stumped. It's always tricky to get dynamic results out of SQL. I'm not complaining about that...but this is one time I'd like to have a generalized tool. It sounds like RETURN SETOF RECORD ought to work, but I haven't succeeded. I've look through a fair few threads and the docs, but I'm still stumped. Here's a PG/pgSQL function as I couldn't get the pure SQL version to work:

DROP FUNCTION IF EXISTS api.domo_passthrough_expand (text, jsonb);
CREATE FUNCTION api.domo_passthrough_expand(type_name text, packed_data jsonb) 
    RETURNS SETOF record AS
$BODY$
BEGIN

    RETURN QUERY EXECUTE FORMAT
   'select *
      from jsonb_populate_recordset(null::%1$I, %2$I) as expanded_data;',
           type_name,
           packed_data;
END;
$BODY$
LANGUAGE plpgsql;

ALTER FUNCTION api.domo_passthrough_expand(text, jsonb) OWNER TO user_bender;

select expanded.* 

  from domo_passthrough,
        domo_passthrough_expand(domo_passthrough.type_name,domo_passthrough.report_data)

ERROR:  a column definition list is required for functions returning "record"
LINE 4:      domo_passthrough_expand(domo_passthrough.type_name,domo...

Is there a way to do my expansion generically with the help of a function? If there isn't, I can approach things another way, but I'd prefer to have it all Postgres-side.

0

There are 0 best solutions below