Thread: table constraint + INSERT

table constraint + INSERT

From
"Risto Tamme"
Date:
<p><font size="2">Hello<br /><br /> I use PostgreSQL in my program and I found a strange behavior, at least for me.<br
/><br/> I have a simple table with constraint<br /><br /> CREATE TABLE "PART"<br /> (<br />   "P_PARTKEY" int4 NOT
NULL,<br/>   "P_RETAILPRICE" numeric,<br />   CONSTRAINT "PART_PRIMARY" PRIMARY KEY ("P_PARTKEY"),<br />   CONSTRAINT
"PART_check"CHECK ("P_RETAILPRICE" = (90000 + "P_PARTKEY" / 10 + "P_PARTKEY" / 100)<br /> );<br /><br /> And I try to
inserta row:<br /> INSERT INTO "PART" ("P_PARTKEY","P_RETAILPRICE") VALUES(999,90109.89);<br /><br /> but it fails:
ERROR: new row for relation "PART" violates check constraint "PART_check"<br /><br /> When you check using your head or
pocketcalculator then this INSERT seems to be correct. Is it some floating point mystery?<br /> Is there some trick?<br
/>I'm using version 8.0.1 for Windows.<br /><br /> With thanks, Risto<br /><br />  </font> 

Re: table constraint + INSERT

From
"Dirk Jagdmann"
Date:
>  I have a simple table with constraint
>
>  CREATE TABLE "PART"
>  (
>    "P_PARTKEY" int4 NOT NULL,
>    "P_RETAILPRICE" numeric,
>    CONSTRAINT "PART_PRIMARY" PRIMARY KEY ("P_PARTKEY"),
>    CONSTRAINT "PART_check" CHECK ("P_RETAILPRICE" = (90000 + "P_PARTKEY" /
> 10 + "P_PARTKEY" / 100)
>  );
>
>  And I try to insert a row:
>  INSERT INTO "PART" ("P_PARTKEY","P_RETAILPRICE") VALUES(999,90109.89);
>
>  but it fails: ERROR:  new row for relation "PART" violates check constraint
> "PART_check"
>
>  When you check using your head or pocket calculator then this INSERT seems
> to be correct. Is it some floating point mystery?
>  Is there some trick?

Postgres is likely doing integer arithmetic:

test=# select 90000+999/10+999/100;?column?
----------   90108
(1 row)

So you have to cast your check constraint to numeric types:

CREATE TABLE PART
( P_PARTKEY int4 NOT NULL, P_RETAILPRICE numeric, CONSTRAINT PART_PRIMARY PRIMARY KEY (P_PARTKEY), CONSTRAINT
PART_checkCHECK (P_RETAILPRICE = (90000 + P_PARTKEY::numeric / 10 
);

However if this would be your real SQL Schema I'd recommend using a
view to calculate the R_RETAILPRICE column:

CREATE TABLE PART
( P_PARTKEY int4 NOT NULL, CONSTRAINT PART_PRIMARY PRIMARY KEY (P_PARTKEY)
);
create view PARTV as
select P_PARTKEY, 90000 + P_PARTKEY::numeric / 10 + P_PARTKEY::numeric / 100 as
from PART;

--
---> Dirk Jagdmann
----> http://cubic.org/~doj
-----> http://llg.cubic.org


Re: table constraint + INSERT

From
"Dirk Jagdmann"
Date:
Too bad, some code got truncated...

CREATE TABLE PART
( P_PARTKEY int4 NOT NULL, P_RETAILPRICE numeric, CONSTRAINT PART_PRIMARY PRIMARY KEY (P_PARTKEY), CONSTRAINT
PART_checkCHECK (P_RETAILPRICE = (90000 + 
P_PARTKEY::numeric / 10 + P_PARTKEY::numeric / 100
);

and the second code should read:

create view PARTV as
select P_PARTKEY, 90000 + P_PARTKEY::numeric / 10 + P_PARTKEY::numeric
/ 100 as P_RETAILPRICE
from PART;

--
---> Dirk Jagdmann
----> http://cubic.org/~doj
-----> http://llg.cubic.org