openGauss ERROR: inserted partition key does not map to any table partition Call getNextException to see other errors in the batch.

发表于:2023年11月22日 17点40分 0 阅读 1评论 3点赞

opengauss系数据库insert失败因缺少匹配分区表,提示inserted partition key does not map to any table partition ,在oracle中一样存在该问题,建议月末或年末提前查询下个月的分区是否存在,匹配oracle的dba_tab_partitions,在postgresq 11后查询pg_partitioned_table,而在openGauss中查询pg_partition.

create table test_part(id int,orderno number, sal number, sdate date not null)
partition by range(sdate)
(
  partition p09 values less than(to_date('2023-10-1','yyyy-mm-dd')),
  partition p10 values less than(to_date('2023-11-1','yyyy-mm-dd')),
  partition p11 values less than(to_date('2023-12-1','yyyy-mm-dd')),
  partition p12 values less than(to_date('2024-01-1','yyyy-mm-dd'))
);

create table test_part2(id int,orderno number, sal number, sdate date not null)
partition by range(sdate)
(
  partition p09 values less than(to_date('2023-10-1','yyyy-mm-dd')),
  partition p10 values less than(to_date('2023-11-1','yyyy-mm-dd')),
  partition p11 values less than(to_date('2023-12-1','yyyy-mm-dd')),
  partition p12 values less than(to_date('2024-01-1','yyyy-mm-dd')),
  partition pmax values less than (maxvalue)
);

create table test_part3(id int,orderno number, sal number, sdate date not null)
partition by hash(id)
partitions 3
;


create table test_part4(id int,orderno number, sal number, sdate date not null,sn number not null)
partition by range(sn)
(
  partition p09 values less than( date_part('epoch', DATE_TRUNC('month', now()) + '0 month')),
  partition p10 values less than( date_part('epoch', DATE_TRUNC('month', now()) + '1 month')),
  partition p11 values less than( date_part('epoch', DATE_TRUNC('month', now()) + '2 month')),
  partition p12 values less than( date_part('epoch', DATE_TRUNC('month', now()) + '3 month'))
);

SELECT parentid::regclass,relname,partstrategy, boundaries,array_to_string (boundaries,',') FROM pg_partition p where parttype='p'
parentid
relname
partstrategy
boundaries
array_to_string
test_part
p12
r
{“2024-01-01 00:00:00”}
2024-01-01 00:00:00
test_part
p11
r
{“2023-12-01 00:00:00”}
2023-12-01 00:00:00
test_part
p10
r
{“2023-11-01 00:00:00”}
2023-11-01 00:00:00
test_part
p09
r
{“2023-10-01 00:00:00”}
2023-10-01 00:00:00
test_part2
pmax
r
{NULL}
test_part2
p12
r
{“2024-01-01 00:00:00”}
2024-01-01 00:00:00
test_part2
p11
r
{“2023-12-01 00:00:00”}
2023-12-01 00:00:00
test_part2
p10
r
{“2023-11-01 00:00:00”}
2023-11-01 00:00:00
test_part2
p09
r
{“2023-10-01 00:00:00”}
2023-10-01 00:00:00
test_part3
p2
h
{2}
2
test_part3
p1
h
{1}
1
test_part3
p0
h
{0}
0
test_part4
p12
r
{1704038400}
1704038400
test_part4
p11
r
{1701360000}
1701360000
test_part4
p10
r
{1698768000}
1698768000
test_part4
p09
r
{1696089600}
1696089600
# range partition
SELECT parentid::regclass,max(boundaries) tt FROM pg_partition p where   parttype='p' and partstrategy='r'   group by parentid
parentid
tt
test_part
{“2024-01-01 00:00:00”}
test_part2
{NULL}
test_part4
{1704038400}

date to epoch

select date_part('epoch', now())::numeric c1,date_part('epoch', now()) c2,to_char(date_part('epoch', now())) c3,round(date_part('epoch', now()),0) c4,extract(epoch from now()) c5;
c1
c2
c3
c4
c5
1696865525.88698
1.696865525886984E9
1696865525.88698411
1696865526
1.696865525886984E9
 

epoch to date

select to_char(date_part('epoch', timestamp '2023-11-01'::timestamp without time zone)) tt;
tt
----------
1698796800

SELECT TIMESTAMP 'epoch' + 1698768000 * INTERVAL '1 second' tt;
tt
--------------------
2023-10-31 16:00:00.0

select to_timestamp(1698768000) c1,to_timestamp(1698768000)::timestamp without time zone c2;
c1                     c2
--------------------   --------------------
2023-11-01 00:00:00.0  2023-11-01 00:00:00.0
 

Tote:
当前是+8时区

select  DATE_TRUNC('month', now()), 
DATE_TRUNC('month', now()::timestamp with time zone),
DATE_TRUNC('month', now()) + '3 month', 
DATE_TRUNC('month', now()::timestamp with time zone) + '3 month', 
DATE_TRUNC('month', now()::timestamp without time zone) + '3 month', 
to_char(date_part('epoch', DATE_TRUNC('month', now()) + '3 month')),
extract(epoch from now()) ,
TIMESTAMP 'epoch' + 1704038400 * INTERVAL '1 second';
2023-10-01 00:00:00.0
2023-10-01 00:00:00.0
2023-12-31 16:00:00.0
2023-12-31 16:00:00.0
2023-12-31 16:00:00.0
1704038400
1.696865270298389E9
2023-12-31 16:00:00.0

add partition

alter table test_part add partition p13 values less than (to_date('2024-02-1','yyyy-mm-dd'))

Note:
在opengauss中对于声明式分区,range partition增加分区语法和oracle一样.

同事写了一份查询预分区是否存在的方法, 因为opengauss目前没有postgresql中的pg_partitioned_table(pg v11+),这里pg与og分别 :

# postgresql

select nspname,relname,max(part_range) max_part_range,
case when max(part_range)>date_trunc('year',now()+interval'1y') then 't' else 'f' end  has_next_year_part   --  next year
from (select nsp.nspname,c.relname,(regexp_matches(pg_get_expr(pc.relpartbound,pc.oid),'\((''[0-9]{4}-[0-9]{2}-[0-9]{2}.*?'')\)','g'))[1]::timestamp part_range
  from pg_class c, pg_partitioned_table p , pg_partition_tree(c.oid) t ,pg_class pc,pg_namespace nsp
  where c.relkind = 'p' and c.relispartition = 'f' and c.oid=p.partrelid and p.partstrat='r' and t.level=1 and t.relid=pc.oid and c.relnamespace=nsp.oid) r
group by nspname,relname  ;

 nspname |      relname      |   max_part_range    | has_next_year_part
---------+-------------------+---------------------+--------------------
 public  | test_check_part_v | 2024-03-01 00:00:00 | t
 public  | test_check_part   | 2023-12-01 00:00:00 | f

# opengauss/MogDB

MogDB=#select t.nspname,t.relname,t.max_part,t.max_part_time,
case when t.max_part_time>date_trunc('month',now()+interval'1month') then 't' else 'f' end  has_next_month_part  ---next month
from 
( select nsp.nspname,c.relname,max(boundaries[1]) max_part,
  max(case when boundaries[1] ~ '([0-9]{4}-[0-9]{2}-[0-9]{2}.*?)' 
   then boundaries[1]::timestamp 
when boundaries[1] ~ '(^[0-9]+$)' and boundaries[1]>946684800::bigint and boundaries[1]<4102444800::bigint 
   then to_timestamp(boundaries[1]::bigint) end)  max_part_time
from pg_partition p , pg_class c ,pg_namespace nsp 
where p.parentid=c.oid and c.relnamespace=nsp.oid 
  group by nsp.nspname,c.relname) t 
where t.max_part_time is not null;
 nspname |  relname   |      max_part       |     max_part_time      | has_next_month_part 
---------+------------+---------------------+------------------------+---------------------
 public  | test_part  | 2024-01-01 00:00:00 | 2024-01-01 00:00:00+08 | t
 public  | test_part2 | 2023-10-01 00:00:00 | 2023-10-01 00:00:00+08 | f
 public  | test_part4 | 1704038400          | 2024-01-01 00:00:00+08 | t

注:本例中分两种情况判断分区是否存在,一个是分区范围用时间格式表示的,另一个是分区范围用epoch(自1970年1月1日00:00:00 UTC以来的秒数)格式表示的,且把epoch限制在946684800~4102444800之间(2000年-2100年)

{{c.name}} {{c.create_time|simymdhm}} {{c.like_num}}
{{c.content}}