防止类型为json'的无效输入语法在Postgres

时间:2017-08-31 01:06:54

标签: json postgresql plpgsql

我有一个包含JSON和计划文本的文本列。我想将其转换为JSON,然后选择一个特定的属性。例如:

user_data
_________
{"user": {"name": "jim"}}
{"user": {"name": "sally"}}
some random data string

我试过了:

select user_data::json#>'{user,name}' from users

我明白了:

ERROR:  invalid input syntax for type json
DETAIL:  Token "some" is invalid.
CONTEXT:  JSON user_data, line 1: some...

有可能阻止这种情况吗?

2 个答案:

答案 0 :(得分:6)

使用此功能:

create or replace function is_json(text)
returns boolean language plpgsql immutable as $$
begin
    perform $1::json;
    return true;
exception
    when invalid_text_representation then 
        return false;
end $$;

测试:

with users(user_data) as (
values
    ('{"user": {"name": "jim"}}'),
    ('not json'),
    ('{"user": {"name": "sally"}}'),
    ('also not json')
)

select user_data::json#>'{user,name}' as name
from users
where is_json(user_data);

  name   
---------
 "jim"
 "sally"
(2 rows)

答案 1 :(得分:4)

如果要跳过包含无效JSON的行,则必须先测试,如果文本是有效的JSON。您可以通过创建一个尝试解析该值的函数来执行此操作,并捕获无效JSON值的异常。

CREATE OR REPLACE FUNCTION is_json(input_text varchar) RETURNS boolean AS $$
  DECLARE
    maybe_json json;
  BEGIN
    BEGIN
      maybe_json := input_text;
    EXCEPTION WHEN others THEN
      RETURN FALSE;
    END;

    RETURN TRUE;
  END;
$$ LANGUAGE plpgsql IMMUTABLE;

如果有,可以使用is_jsonCASE子句中的WHERE函数缩小有效值。

-- this can eliminate invalid values
SELECT user_data::json #> '{user,name}'
FROM users WHERE is_json(user_data);

-- or this if you want to fill will NULLs
SELECT
  CASE
    WHEN is_json(user_data)
      THEN user_data::json #> '{user,name}'
    ELSE
      NULL
  END
FROM users;