Commit 3752e85b authored by Tom Lane's avatar Tom Lane

Determine the set of constraints applied to a domain at executor

startup, not in the parser; this allows ALTER DOMAIN to work correctly
with domain constraint operations stored in rules.  Rod Taylor;
code review by Tom Lane.
parent 464598b6
...@@ -9,7 +9,7 @@ ...@@ -9,7 +9,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/commands/copy.c,v 1.188 2003/01/10 22:03:27 petere Exp $ * $Header: /cvsroot/pgsql/src/backend/commands/copy.c,v 1.189 2003/02/03 21:15:43 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -853,7 +853,7 @@ CopyFrom(Relation rel, List *attnumlist, bool binary, bool oids, ...@@ -853,7 +853,7 @@ CopyFrom(Relation rel, List *attnumlist, bool binary, bool oids,
} }
} }
/* If it's a domain type, get info on domain constraints */ /* If it's a domain type, set up to check domain constraints */
if (get_typtype(attr[i]->atttypid) == 'd') if (get_typtype(attr[i]->atttypid) == 'd')
{ {
Param *prm; Param *prm;
...@@ -863,25 +863,23 @@ CopyFrom(Relation rel, List *attnumlist, bool binary, bool oids, ...@@ -863,25 +863,23 @@ CopyFrom(Relation rel, List *attnumlist, bool binary, bool oids,
* Easiest way to do this is to use parse_coerce.c to set up * Easiest way to do this is to use parse_coerce.c to set up
* an expression that checks the constraints. (At present, * an expression that checks the constraints. (At present,
* the expression might contain a length-coercion-function call * the expression might contain a length-coercion-function call
* and/or ConstraintTest nodes.) The bottom of the expression * and/or CoerceToDomain nodes.) The bottom of the expression
* is a Param node so that we can fill in the actual datum during * is a Param node so that we can fill in the actual datum during
* the data input loop. * the data input loop.
*/ */
prm = makeNode(Param); prm = makeNode(Param);
prm->paramkind = PARAM_EXEC; prm->paramkind = PARAM_EXEC;
prm->paramid = 0; prm->paramid = 0;
prm->paramtype = attr[i]->atttypid; prm->paramtype = getBaseType(attr[i]->atttypid);
node = coerce_type_constraints((Node *) prm, attr[i]->atttypid, node = coerce_to_domain((Node *) prm,
COERCE_IMPLICIT_CAST); prm->paramtype,
attr[i]->atttypid,
COERCE_IMPLICIT_CAST);
/* check whether any constraints actually found */ constraintexprs[i] = ExecPrepareExpr((Expr *) node,
if (node != (Node *) prm) estate);
{ hasConstraints = true;
constraintexprs[i] = ExecPrepareExpr((Expr *) node,
estate);
hasConstraints = true;
}
} }
} }
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/commands/typecmds.c,v 1.29 2003/01/08 22:06:23 tgl Exp $ * $Header: /cvsroot/pgsql/src/backend/commands/typecmds.c,v 1.30 2003/02/03 21:15:43 tgl Exp $
* *
* DESCRIPTION * DESCRIPTION
* The "DefineFoo" routines take the parse tree and pick out the * The "DefineFoo" routines take the parse tree and pick out the
...@@ -46,8 +46,10 @@ ...@@ -46,8 +46,10 @@
#include "commands/typecmds.h" #include "commands/typecmds.h"
#include "executor/executor.h" #include "executor/executor.h"
#include "miscadmin.h" #include "miscadmin.h"
#include "nodes/execnodes.h"
#include "nodes/nodes.h" #include "nodes/nodes.h"
#include "optimizer/clauses.h" #include "optimizer/clauses.h"
#include "optimizer/planmain.h"
#include "optimizer/var.h" #include "optimizer/var.h"
#include "parser/parse_coerce.h" #include "parser/parse_coerce.h"
#include "parser/parse_expr.h" #include "parser/parse_expr.h"
...@@ -1555,7 +1557,7 @@ domainAddConstraint(Oid domainOid, Oid domainNamespace, Oid baseTypeOid, ...@@ -1555,7 +1557,7 @@ domainAddConstraint(Oid domainOid, Oid domainNamespace, Oid baseTypeOid,
char *ccsrc; char *ccsrc;
char *ccbin; char *ccbin;
ParseState *pstate; ParseState *pstate;
ConstraintTestValue *domVal; CoerceToDomainValue *domVal;
/* /*
* Assign or validate constraint name * Assign or validate constraint name
...@@ -1582,13 +1584,13 @@ domainAddConstraint(Oid domainOid, Oid domainNamespace, Oid baseTypeOid, ...@@ -1582,13 +1584,13 @@ domainAddConstraint(Oid domainOid, Oid domainNamespace, Oid baseTypeOid,
pstate = make_parsestate(NULL); pstate = make_parsestate(NULL);
/* /*
* Set up a ConstraintTestValue to represent the occurrence of VALUE * Set up a CoerceToDomainValue to represent the occurrence of VALUE
* in the expression. Note that it will appear to have the type of the * in the expression. Note that it will appear to have the type of the
* base type, not the domain. This seems correct since within the * base type, not the domain. This seems correct since within the
* check expression, we should not assume the input value can be considered * check expression, we should not assume the input value can be considered
* a member of the domain. * a member of the domain.
*/ */
domVal = makeNode(ConstraintTestValue); domVal = makeNode(CoerceToDomainValue);
domVal->typeId = baseTypeOid; domVal->typeId = baseTypeOid;
domVal->typeMod = typMod; domVal->typeMod = typMod;
...@@ -1669,6 +1671,125 @@ domainAddConstraint(Oid domainOid, Oid domainNamespace, Oid baseTypeOid, ...@@ -1669,6 +1671,125 @@ domainAddConstraint(Oid domainOid, Oid domainNamespace, Oid baseTypeOid,
return ccbin; return ccbin;
} }
/*
* GetDomainConstraints - get a list of the current constraints of domain
*
* Returns a possibly-empty list of DomainConstraintState nodes.
*
* This is called by the executor during plan startup for a CoerceToDomain
* expression node. The given constraints will be checked for each value
* passed through the node.
*/
List *
GetDomainConstraints(Oid typeOid)
{
List *result = NIL;
bool notNull = false;
Relation conRel;
conRel = heap_openr(ConstraintRelationName, AccessShareLock);
for (;;)
{
HeapTuple tup;
HeapTuple conTup;
Form_pg_type typTup;
ScanKeyData key[1];
SysScanDesc scan;
tup = SearchSysCache(TYPEOID,
ObjectIdGetDatum(typeOid),
0, 0, 0);
if (!HeapTupleIsValid(tup))
elog(ERROR, "GetDomainConstraints: failed to lookup type %u",
typeOid);
typTup = (Form_pg_type) GETSTRUCT(tup);
/* Test for NOT NULL Constraint */
if (typTup->typnotnull)
notNull = true;
/* Look for CHECK Constraints on this domain */
ScanKeyEntryInitialize(&key[0], 0x0,
Anum_pg_constraint_contypid, F_OIDEQ,
ObjectIdGetDatum(typeOid));
scan = systable_beginscan(conRel, ConstraintTypidIndex, true,
SnapshotNow, 1, key);
while (HeapTupleIsValid(conTup = systable_getnext(scan)))
{
Form_pg_constraint c = (Form_pg_constraint) GETSTRUCT(conTup);
Datum val;
bool isNull;
Expr *check_expr;
DomainConstraintState *r;
/* Ignore non-CHECK constraints (presently, shouldn't be any) */
if (c->contype != CONSTRAINT_CHECK)
continue;
/* Not expecting conbin to be NULL, but we'll test for it anyway */
val = fastgetattr(conTup, Anum_pg_constraint_conbin,
conRel->rd_att, &isNull);
if (isNull)
elog(ERROR, "GetDomainConstraints: domain %s constraint %s has NULL conbin",
NameStr(typTup->typname), NameStr(c->conname));
check_expr = (Expr *)
stringToNode(DatumGetCString(DirectFunctionCall1(textout,
val)));
/* ExecInitExpr assumes we already fixed opfuncids */
fix_opfuncids((Node *) check_expr);
r = makeNode(DomainConstraintState);
r->constrainttype = DOM_CONSTRAINT_CHECK;
r->name = pstrdup(NameStr(c->conname));
r->check_expr = ExecInitExpr(check_expr, NULL);
/*
* use lcons() here because constraints of lower domains should
* be applied earlier.
*/
result = lcons(r, result);
}
systable_endscan(scan);
if (typTup->typtype != 'd')
{
/* Not a domain, so done */
ReleaseSysCache(tup);
break;
}
/* else loop to next domain in stack */
typeOid = typTup->typbasetype;
ReleaseSysCache(tup);
}
heap_close(conRel, AccessShareLock);
/*
* Only need to add one NOT NULL check regardless of how many domains
* in the stack request it.
*/
if (notNull)
{
DomainConstraintState *r = makeNode(DomainConstraintState);
r->constrainttype = DOM_CONSTRAINT_NOTNULL;
r->name = pstrdup("NOT NULL");
r->check_expr = NULL;
/* lcons to apply the nullness check FIRST */
result = lcons(r, result);
}
return result;
}
/* /*
* ALTER DOMAIN .. OWNER TO * ALTER DOMAIN .. OWNER TO
* *
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/executor/execQual.c,v 1.123 2003/01/12 04:03:34 tgl Exp $ * $Header: /cvsroot/pgsql/src/backend/executor/execQual.c,v 1.124 2003/02/03 21:15:43 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -36,6 +36,7 @@ ...@@ -36,6 +36,7 @@
#include "access/heapam.h" #include "access/heapam.h"
#include "catalog/pg_type.h" #include "catalog/pg_type.h"
#include "commands/typecmds.h"
#include "executor/execdebug.h" #include "executor/execdebug.h"
#include "executor/functions.h" #include "executor/functions.h"
#include "executor/nodeSubplan.h" #include "executor/nodeSubplan.h"
...@@ -80,10 +81,10 @@ static Datum ExecEvalNullTest(GenericExprState *nstate, ...@@ -80,10 +81,10 @@ static Datum ExecEvalNullTest(GenericExprState *nstate,
static Datum ExecEvalBooleanTest(GenericExprState *bstate, static Datum ExecEvalBooleanTest(GenericExprState *bstate,
ExprContext *econtext, ExprContext *econtext,
bool *isNull, ExprDoneCond *isDone); bool *isNull, ExprDoneCond *isDone);
static Datum ExecEvalConstraintTest(ConstraintTestState *cstate, static Datum ExecEvalCoerceToDomain(CoerceToDomainState *cstate,
ExprContext *econtext, ExprContext *econtext,
bool *isNull, ExprDoneCond *isDone); bool *isNull, ExprDoneCond *isDone);
static Datum ExecEvalConstraintTestValue(ConstraintTestValue *conVal, static Datum ExecEvalCoerceToDomainValue(CoerceToDomainValue *conVal,
ExprContext *econtext, bool *isNull); ExprContext *econtext, bool *isNull);
static Datum ExecEvalFieldSelect(GenericExprState *fstate, static Datum ExecEvalFieldSelect(GenericExprState *fstate,
ExprContext *econtext, ExprContext *econtext,
...@@ -1559,32 +1560,37 @@ ExecEvalBooleanTest(GenericExprState *bstate, ...@@ -1559,32 +1560,37 @@ ExecEvalBooleanTest(GenericExprState *bstate,
} }
/* /*
* ExecEvalConstraintTest * ExecEvalCoerceToDomain
* *
* Test the constraint against the data provided. If the data fits * Test the provided data against the domain constraint(s). If the data
* within the constraint specifications, pass it through (return the * passes the constraint specifications, pass it through (return the
* datum) otherwise throw an error. * datum) otherwise throw an error.
*/ */
static Datum static Datum
ExecEvalConstraintTest(ConstraintTestState *cstate, ExprContext *econtext, ExecEvalCoerceToDomain(CoerceToDomainState *cstate, ExprContext *econtext,
bool *isNull, ExprDoneCond *isDone) bool *isNull, ExprDoneCond *isDone)
{ {
ConstraintTest *constraint = (ConstraintTest *) cstate->xprstate.expr; CoerceToDomain *ctest = (CoerceToDomain *) cstate->xprstate.expr;
Datum result; Datum result;
List *l;
result = ExecEvalExpr(cstate->arg, econtext, isNull, isDone); result = ExecEvalExpr(cstate->arg, econtext, isNull, isDone);
if (isDone && *isDone == ExprEndResult) if (isDone && *isDone == ExprEndResult)
return result; /* nothing to check */ return result; /* nothing to check */
switch (constraint->testtype) foreach(l, cstate->constraints)
{ {
case CONSTR_TEST_NOTNULL: DomainConstraintState *con = (DomainConstraintState *) lfirst(l);
if (*isNull)
elog(ERROR, "Domain %s does not allow NULL values", switch (con->constrainttype)
constraint->domname); {
break; case DOM_CONSTRAINT_NOTNULL:
case CONSTR_TEST_CHECK: if (*isNull)
elog(ERROR, "Domain %s does not allow NULL values",
format_type_be(ctest->resulttype));
break;
case DOM_CONSTRAINT_CHECK:
{ {
Datum conResult; Datum conResult;
bool conIsNull; bool conIsNull;
...@@ -1592,7 +1598,7 @@ ExecEvalConstraintTest(ConstraintTestState *cstate, ExprContext *econtext, ...@@ -1592,7 +1598,7 @@ ExecEvalConstraintTest(ConstraintTestState *cstate, ExprContext *econtext,
bool save_isNull; bool save_isNull;
/* /*
* Set up value to be returned by ConstraintTestValue nodes. * Set up value to be returned by CoerceToDomainValue nodes.
* We must save and restore prior setting of econtext's * We must save and restore prior setting of econtext's
* domainValue fields, in case this node is itself within * domainValue fields, in case this node is itself within
* a check expression for another domain. * a check expression for another domain.
...@@ -1603,35 +1609,37 @@ ExecEvalConstraintTest(ConstraintTestState *cstate, ExprContext *econtext, ...@@ -1603,35 +1609,37 @@ ExecEvalConstraintTest(ConstraintTestState *cstate, ExprContext *econtext,
econtext->domainValue_datum = result; econtext->domainValue_datum = result;
econtext->domainValue_isNull = *isNull; econtext->domainValue_isNull = *isNull;
conResult = ExecEvalExpr(cstate->check_expr, conResult = ExecEvalExpr(con->check_expr,
econtext, &conIsNull, NULL); econtext, &conIsNull, NULL);
if (!conIsNull && if (!conIsNull &&
!DatumGetBool(conResult)) !DatumGetBool(conResult))
elog(ERROR, "ExecEvalConstraintTest: Domain %s constraint %s failed", elog(ERROR, "ExecEvalCoerceToDomain: Domain %s constraint %s failed",
constraint->domname, constraint->name); format_type_be(ctest->resulttype), con->name);
econtext->domainValue_datum = save_datum; econtext->domainValue_datum = save_datum;
econtext->domainValue_isNull = save_isNull; econtext->domainValue_isNull = save_isNull;
break;
} }
break; default:
default: elog(ERROR, "ExecEvalCoerceToDomain: Constraint type unknown");
elog(ERROR, "ExecEvalConstraintTest: Constraint type unknown"); break;
break; }
} }
/* If all has gone well (constraint did not fail) return the datum */ /* If all has gone well (constraints did not fail) return the datum */
return result; return result;
} }
/* /*
* ExecEvalConstraintTestValue * ExecEvalCoerceToDomainValue
* *
* Return the value stored by constraintTest. * Return the value stored by CoerceToDomain.
*/ */
static Datum static Datum
ExecEvalConstraintTestValue(ConstraintTestValue *conVal, ExprContext *econtext, ExecEvalCoerceToDomainValue(CoerceToDomainValue *conVal,
bool *isNull) ExprContext *econtext, bool *isNull)
{ {
*isNull = econtext->domainValue_isNull; *isNull = econtext->domainValue_isNull;
return econtext->domainValue_datum; return econtext->domainValue_datum;
...@@ -1830,14 +1838,14 @@ ExecEvalExpr(ExprState *expression, ...@@ -1830,14 +1838,14 @@ ExecEvalExpr(ExprState *expression,
isNull, isNull,
isDone); isDone);
break; break;
case T_ConstraintTest: case T_CoerceToDomain:
retDatum = ExecEvalConstraintTest((ConstraintTestState *) expression, retDatum = ExecEvalCoerceToDomain((CoerceToDomainState *) expression,
econtext, econtext,
isNull, isNull,
isDone); isDone);
break; break;
case T_ConstraintTestValue: case T_CoerceToDomainValue:
retDatum = ExecEvalConstraintTestValue((ConstraintTestValue *) expr, retDatum = ExecEvalCoerceToDomainValue((CoerceToDomainValue *) expr,
econtext, econtext,
isNull); isNull);
break; break;
...@@ -1915,7 +1923,7 @@ ExecInitExpr(Expr *node, PlanState *parent) ...@@ -1915,7 +1923,7 @@ ExecInitExpr(Expr *node, PlanState *parent)
case T_Var: case T_Var:
case T_Const: case T_Const:
case T_Param: case T_Param:
case T_ConstraintTestValue: case T_CoerceToDomainValue:
/* No special setup needed for these node types */ /* No special setup needed for these node types */
state = (ExprState *) makeNode(ExprState); state = (ExprState *) makeNode(ExprState);
break; break;
...@@ -2092,13 +2100,13 @@ ExecInitExpr(Expr *node, PlanState *parent) ...@@ -2092,13 +2100,13 @@ ExecInitExpr(Expr *node, PlanState *parent)
state = (ExprState *) gstate; state = (ExprState *) gstate;
} }
break; break;
case T_ConstraintTest: case T_CoerceToDomain:
{ {
ConstraintTest *ctest = (ConstraintTest *) node; CoerceToDomain *ctest = (CoerceToDomain *) node;
ConstraintTestState *cstate = makeNode(ConstraintTestState); CoerceToDomainState *cstate = makeNode(CoerceToDomainState);
cstate->arg = ExecInitExpr(ctest->arg, parent); cstate->arg = ExecInitExpr(ctest->arg, parent);
cstate->check_expr = ExecInitExpr(ctest->check_expr, parent); cstate->constraints = GetDomainConstraints(ctest->resulttype);
state = (ExprState *) cstate; state = (ExprState *) cstate;
} }
break; break;
......
...@@ -15,7 +15,7 @@ ...@@ -15,7 +15,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/nodes/copyfuncs.c,v 1.238 2003/01/23 23:38:56 petere Exp $ * $Header: /cvsroot/pgsql/src/backend/nodes/copyfuncs.c,v 1.239 2003/02/03 21:15:43 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -933,29 +933,28 @@ _copyBooleanTest(BooleanTest *from) ...@@ -933,29 +933,28 @@ _copyBooleanTest(BooleanTest *from)
} }
/* /*
* _copyConstraintTest * _copyCoerceToDomain
*/ */
static ConstraintTest * static CoerceToDomain *
_copyConstraintTest(ConstraintTest *from) _copyCoerceToDomain(CoerceToDomain *from)
{ {
ConstraintTest *newnode = makeNode(ConstraintTest); CoerceToDomain *newnode = makeNode(CoerceToDomain);
COPY_NODE_FIELD(arg); COPY_NODE_FIELD(arg);
COPY_SCALAR_FIELD(testtype); COPY_SCALAR_FIELD(resulttype);
COPY_STRING_FIELD(name); COPY_SCALAR_FIELD(resulttypmod);
COPY_STRING_FIELD(domname); COPY_SCALAR_FIELD(coercionformat);
COPY_NODE_FIELD(check_expr);
return newnode; return newnode;
} }
/* /*
* _copyConstraintTestValue * _copyCoerceToDomainValue
*/ */
static ConstraintTestValue * static CoerceToDomainValue *
_copyConstraintTestValue(ConstraintTestValue *from) _copyCoerceToDomainValue(CoerceToDomainValue *from)
{ {
ConstraintTestValue *newnode = makeNode(ConstraintTestValue); CoerceToDomainValue *newnode = makeNode(CoerceToDomainValue);
COPY_SCALAR_FIELD(typeId); COPY_SCALAR_FIELD(typeId);
COPY_SCALAR_FIELD(typeMod); COPY_SCALAR_FIELD(typeMod);
...@@ -2476,11 +2475,11 @@ copyObject(void *from) ...@@ -2476,11 +2475,11 @@ copyObject(void *from)
case T_BooleanTest: case T_BooleanTest:
retval = _copyBooleanTest(from); retval = _copyBooleanTest(from);
break; break;
case T_ConstraintTest: case T_CoerceToDomain:
retval = _copyConstraintTest(from); retval = _copyCoerceToDomain(from);
break; break;
case T_ConstraintTestValue: case T_CoerceToDomainValue:
retval = _copyConstraintTestValue(from); retval = _copyCoerceToDomainValue(from);
break; break;
case T_TargetEntry: case T_TargetEntry:
retval = _copyTargetEntry(from); retval = _copyTargetEntry(from);
......
...@@ -18,7 +18,7 @@ ...@@ -18,7 +18,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/nodes/equalfuncs.c,v 1.182 2003/01/23 23:38:56 petere Exp $ * $Header: /cvsroot/pgsql/src/backend/nodes/equalfuncs.c,v 1.183 2003/02/03 21:15:44 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -383,19 +383,25 @@ _equalBooleanTest(BooleanTest *a, BooleanTest *b) ...@@ -383,19 +383,25 @@ _equalBooleanTest(BooleanTest *a, BooleanTest *b)
} }
static bool static bool
_equalConstraintTest(ConstraintTest *a, ConstraintTest *b) _equalCoerceToDomain(CoerceToDomain *a, CoerceToDomain *b)
{ {
COMPARE_NODE_FIELD(arg); COMPARE_NODE_FIELD(arg);
COMPARE_SCALAR_FIELD(testtype); COMPARE_SCALAR_FIELD(resulttype);
COMPARE_STRING_FIELD(name); COMPARE_SCALAR_FIELD(resulttypmod);
COMPARE_STRING_FIELD(domname); /*
COMPARE_NODE_FIELD(check_expr); * Special-case COERCE_DONTCARE, so that pathkeys can build coercion
* nodes that are equal() to both explicit and implicit coercions.
*/
if (a->coercionformat != b->coercionformat &&
a->coercionformat != COERCE_DONTCARE &&
b->coercionformat != COERCE_DONTCARE)
return false;
return true; return true;
} }
static bool static bool
_equalConstraintTestValue(ConstraintTestValue *a, ConstraintTestValue *b) _equalCoerceToDomainValue(CoerceToDomainValue *a, CoerceToDomainValue *b)
{ {
COMPARE_SCALAR_FIELD(typeId); COMPARE_SCALAR_FIELD(typeId);
COMPARE_SCALAR_FIELD(typeMod); COMPARE_SCALAR_FIELD(typeMod);
...@@ -1599,11 +1605,11 @@ equal(void *a, void *b) ...@@ -1599,11 +1605,11 @@ equal(void *a, void *b)
case T_BooleanTest: case T_BooleanTest:
retval = _equalBooleanTest(a, b); retval = _equalBooleanTest(a, b);
break; break;
case T_ConstraintTest: case T_CoerceToDomain:
retval = _equalConstraintTest(a, b); retval = _equalCoerceToDomain(a, b);
break; break;
case T_ConstraintTestValue: case T_CoerceToDomainValue:
retval = _equalConstraintTestValue(a, b); retval = _equalCoerceToDomainValue(a, b);
break; break;
case T_TargetEntry: case T_TargetEntry:
retval = _equalTargetEntry(a, b); retval = _equalTargetEntry(a, b);
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/nodes/outfuncs.c,v 1.194 2003/01/20 18:54:47 tgl Exp $ * $Header: /cvsroot/pgsql/src/backend/nodes/outfuncs.c,v 1.195 2003/02/03 21:15:44 tgl Exp $
* *
* NOTES * NOTES
* Every node type that can appear in stored rules' parsetrees *must* * Every node type that can appear in stored rules' parsetrees *must*
...@@ -745,21 +745,20 @@ _outBooleanTest(StringInfo str, BooleanTest *node) ...@@ -745,21 +745,20 @@ _outBooleanTest(StringInfo str, BooleanTest *node)
} }
static void static void
_outConstraintTest(StringInfo str, ConstraintTest *node) _outCoerceToDomain(StringInfo str, CoerceToDomain *node)
{ {
WRITE_NODE_TYPE("CONSTRAINTTEST"); WRITE_NODE_TYPE("COERCETODOMAIN");
WRITE_NODE_FIELD(arg); WRITE_NODE_FIELD(arg);
WRITE_ENUM_FIELD(testtype, ConstraintTestType); WRITE_OID_FIELD(resulttype);
WRITE_STRING_FIELD(name); WRITE_INT_FIELD(resulttypmod);
WRITE_STRING_FIELD(domname); WRITE_ENUM_FIELD(coercionformat, CoercionForm);
WRITE_NODE_FIELD(check_expr);
} }
static void static void
_outConstraintTestValue(StringInfo str, ConstraintTestValue *node) _outCoerceToDomainValue(StringInfo str, CoerceToDomainValue *node)
{ {
WRITE_NODE_TYPE("CONSTRAINTTESTVALUE"); WRITE_NODE_TYPE("COERCETODOMAINVALUE");
WRITE_OID_FIELD(typeId); WRITE_OID_FIELD(typeId);
WRITE_INT_FIELD(typeMod); WRITE_INT_FIELD(typeMod);
...@@ -1548,11 +1547,11 @@ _outNode(StringInfo str, void *obj) ...@@ -1548,11 +1547,11 @@ _outNode(StringInfo str, void *obj)
case T_BooleanTest: case T_BooleanTest:
_outBooleanTest(str, obj); _outBooleanTest(str, obj);
break; break;
case T_ConstraintTest: case T_CoerceToDomain:
_outConstraintTest(str, obj); _outCoerceToDomain(str, obj);
break; break;
case T_ConstraintTestValue: case T_CoerceToDomainValue:
_outConstraintTestValue(str, obj); _outCoerceToDomainValue(str, obj);
break; break;
case T_TargetEntry: case T_TargetEntry:
_outTargetEntry(str, obj); _outTargetEntry(str, obj);
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/nodes/readfuncs.c,v 1.146 2003/01/10 21:08:11 tgl Exp $ * $Header: /cvsroot/pgsql/src/backend/nodes/readfuncs.c,v 1.147 2003/02/03 21:15:44 tgl Exp $
* *
* NOTES * NOTES
* Path and Plan nodes do not have any readfuncs support, because we * Path and Plan nodes do not have any readfuncs support, because we
...@@ -635,29 +635,28 @@ _readBooleanTest(void) ...@@ -635,29 +635,28 @@ _readBooleanTest(void)
} }
/* /*
* _readConstraintTest * _readCoerceToDomain
*/ */
static ConstraintTest * static CoerceToDomain *
_readConstraintTest(void) _readCoerceToDomain(void)
{ {
READ_LOCALS(ConstraintTest); READ_LOCALS(CoerceToDomain);
READ_NODE_FIELD(arg); READ_NODE_FIELD(arg);
READ_ENUM_FIELD(testtype, ConstraintTestType); READ_OID_FIELD(resulttype);
READ_STRING_FIELD(name); READ_INT_FIELD(resulttypmod);
READ_STRING_FIELD(domname); READ_ENUM_FIELD(coercionformat, CoercionForm);
READ_NODE_FIELD(check_expr);
READ_DONE(); READ_DONE();
} }
/* /*
* _readConstraintTestValue * _readCoerceToDomainValue
*/ */
static ConstraintTestValue * static CoerceToDomainValue *
_readConstraintTestValue(void) _readCoerceToDomainValue(void)
{ {
READ_LOCALS(ConstraintTestValue); READ_LOCALS(CoerceToDomainValue);
READ_OID_FIELD(typeId); READ_OID_FIELD(typeId);
READ_INT_FIELD(typeMod); READ_INT_FIELD(typeMod);
...@@ -900,10 +899,10 @@ parseNodeString(void) ...@@ -900,10 +899,10 @@ parseNodeString(void)
return_value = _readNullTest(); return_value = _readNullTest();
else if (MATCH("BOOLEANTEST", 11)) else if (MATCH("BOOLEANTEST", 11))
return_value = _readBooleanTest(); return_value = _readBooleanTest();
else if (MATCH("CONSTRAINTTEST", 14)) else if (MATCH("COERCETODOMAIN", 14))
return_value = _readConstraintTest(); return_value = _readCoerceToDomain();
else if (MATCH("CONSTRAINTTESTVALUE", 19)) else if (MATCH("COERCETODOMAINVALUE", 19))
return_value = _readConstraintTestValue(); return_value = _readCoerceToDomainValue();
else if (MATCH("TARGETENTRY", 11)) else if (MATCH("TARGETENTRY", 11))
return_value = _readTargetEntry(); return_value = _readTargetEntry();
else if (MATCH("RANGETBLREF", 11)) else if (MATCH("RANGETBLREF", 11))
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/optimizer/path/clausesel.c,v 1.56 2003/01/28 22:13:29 tgl Exp $ * $Header: /cvsroot/pgsql/src/backend/optimizer/path/clausesel.c,v 1.57 2003/02/03 21:15:44 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -540,6 +540,14 @@ clause_selectivity(Query *root, ...@@ -540,6 +540,14 @@ clause_selectivity(Query *root,
varRelid, varRelid,
jointype); jointype);
} }
else if (IsA(clause, CoerceToDomain))
{
/* Not sure this case is needed, but it can't hurt */
s1 = clause_selectivity(root,
(Node *) ((CoerceToDomain *) clause)->arg,
varRelid,
jointype);
}
#ifdef SELECTIVITY_DEBUG #ifdef SELECTIVITY_DEBUG
elog(DEBUG3, "clause_selectivity: s1 %f", s1); elog(DEBUG3, "clause_selectivity: s1 %f", s1);
......
...@@ -15,7 +15,7 @@ ...@@ -15,7 +15,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/optimizer/prep/preptlist.c,v 1.59 2002/12/12 15:49:32 tgl Exp $ * $Header: /cvsroot/pgsql/src/backend/optimizer/prep/preptlist.c,v 1.60 2003/02/03 21:15:44 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -185,9 +185,10 @@ expand_targetlist(List *tlist, int command_type, ...@@ -185,9 +185,10 @@ expand_targetlist(List *tlist, int command_type,
true, /* isnull */ true, /* isnull */
att_tup->attbyval); att_tup->attbyval);
if (!att_tup->attisdropped) if (!att_tup->attisdropped)
new_expr = coerce_type_constraints(new_expr, new_expr = coerce_to_domain(new_expr,
atttype, InvalidOid,
COERCE_IMPLICIT_CAST); atttype,
COERCE_IMPLICIT_CAST);
break; break;
case CMD_UPDATE: case CMD_UPDATE:
/* Insert NULLs for dropped columns */ /* Insert NULLs for dropped columns */
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/optimizer/util/clauses.c,v 1.125 2003/01/20 18:54:54 tgl Exp $ * $Header: /cvsroot/pgsql/src/backend/optimizer/util/clauses.c,v 1.126 2003/02/03 21:15:44 tgl Exp $
* *
* HISTORY * HISTORY
* AUTHOR DATE MAJOR EVENT * AUTHOR DATE MAJOR EVENT
...@@ -2030,7 +2030,7 @@ expression_tree_walker(Node *node, ...@@ -2030,7 +2030,7 @@ expression_tree_walker(Node *node,
case T_Var: case T_Var:
case T_Const: case T_Const:
case T_Param: case T_Param:
case T_ConstraintTestValue: case T_CoerceToDomainValue:
case T_RangeTblRef: case T_RangeTblRef:
/* primitive node types with no subnodes */ /* primitive node types with no subnodes */
break; break;
...@@ -2148,10 +2148,8 @@ expression_tree_walker(Node *node, ...@@ -2148,10 +2148,8 @@ expression_tree_walker(Node *node,
return walker(((NullTest *) node)->arg, context); return walker(((NullTest *) node)->arg, context);
case T_BooleanTest: case T_BooleanTest:
return walker(((BooleanTest *) node)->arg, context); return walker(((BooleanTest *) node)->arg, context);
case T_ConstraintTest: case T_CoerceToDomain:
if (walker(((ConstraintTest *) node)->arg, context)) return walker(((CoerceToDomain *) node)->arg, context);
return true;
return walker(((ConstraintTest *) node)->check_expr, context);
case T_TargetEntry: case T_TargetEntry:
return walker(((TargetEntry *) node)->expr, context); return walker(((TargetEntry *) node)->expr, context);
case T_Query: case T_Query:
...@@ -2374,7 +2372,7 @@ expression_tree_mutator(Node *node, ...@@ -2374,7 +2372,7 @@ expression_tree_mutator(Node *node,
case T_Var: case T_Var:
case T_Const: case T_Const:
case T_Param: case T_Param:
case T_ConstraintTestValue: case T_CoerceToDomainValue:
case T_RangeTblRef: case T_RangeTblRef:
/* primitive node types with no subnodes */ /* primitive node types with no subnodes */
return (Node *) copyObject(node); return (Node *) copyObject(node);
...@@ -2538,14 +2536,13 @@ expression_tree_mutator(Node *node, ...@@ -2538,14 +2536,13 @@ expression_tree_mutator(Node *node,
return (Node *) newnode; return (Node *) newnode;
} }
break; break;
case T_ConstraintTest: case T_CoerceToDomain:
{ {
ConstraintTest *ctest = (ConstraintTest *) node; CoerceToDomain *ctest = (CoerceToDomain *) node;
ConstraintTest *newnode; CoerceToDomain *newnode;
FLATCOPY(newnode, ctest, ConstraintTest); FLATCOPY(newnode, ctest, CoerceToDomain);
MUTATE(newnode->arg, ctest->arg, Expr *); MUTATE(newnode->arg, ctest->arg, Expr *);
MUTATE(newnode->check_expr, ctest->check_expr, Expr *);
return (Node *) newnode; return (Node *) newnode;
} }
break; break;
......
This diff is collapsed.
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/parser/parse_expr.c,v 1.141 2003/01/13 00:18:51 tgl Exp $ * $Header: /cvsroot/pgsql/src/backend/parser/parse_expr.c,v 1.142 2003/02/03 21:15:44 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -674,8 +674,8 @@ transformExpr(ParseState *pstate, Node *expr) ...@@ -674,8 +674,8 @@ transformExpr(ParseState *pstate, Node *expr)
case T_ArrayRef: case T_ArrayRef:
case T_FieldSelect: case T_FieldSelect:
case T_RelabelType: case T_RelabelType:
case T_ConstraintTest: case T_CoerceToDomain:
case T_ConstraintTestValue: case T_CoerceToDomainValue:
{ {
result = (Node *) expr; result = (Node *) expr;
break; break;
...@@ -1017,11 +1017,11 @@ exprType(Node *expr) ...@@ -1017,11 +1017,11 @@ exprType(Node *expr)
case T_BooleanTest: case T_BooleanTest:
type = BOOLOID; type = BOOLOID;
break; break;
case T_ConstraintTest: case T_CoerceToDomain:
type = exprType((Node *) ((ConstraintTest *) expr)->arg); type = ((CoerceToDomain *) expr)->resulttype;
break; break;
case T_ConstraintTestValue: case T_CoerceToDomainValue:
type = ((ConstraintTestValue *) expr)->typeId; type = ((CoerceToDomainValue *) expr)->typeId;
break; break;
case T_RangeVar: case T_RangeVar:
/* /*
...@@ -1117,10 +1117,10 @@ exprTypmod(Node *expr) ...@@ -1117,10 +1117,10 @@ exprTypmod(Node *expr)
return typmod; return typmod;
} }
break; break;
case T_ConstraintTest: case T_CoerceToDomain:
return exprTypmod((Node *) ((ConstraintTest *) expr)->arg); return ((CoerceToDomain *) expr)->resulttypmod;
case T_ConstraintTestValue: case T_CoerceToDomainValue:
return ((ConstraintTestValue *) expr)->typeMod; return ((CoerceToDomainValue *) expr)->typeMod;
default: default:
break; break;
} }
......
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
* back to source text * back to source text
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/utils/adt/ruleutils.c,v 1.133 2003/02/03 15:17:24 momjian Exp $ * $Header: /cvsroot/pgsql/src/backend/utils/adt/ruleutils.c,v 1.134 2003/02/03 21:15:44 tgl Exp $
* *
* This software is copyrighted by Jan Wieck - Hamburg. * This software is copyrighted by Jan Wieck - Hamburg.
* *
...@@ -638,11 +638,11 @@ pg_get_constraintdef(PG_FUNCTION_ARGS) ...@@ -638,11 +638,11 @@ pg_get_constraintdef(PG_FUNCTION_ARGS)
} }
appendStringInfo(&buf, "%s", string); appendStringInfo(&buf, "%s", string);
/* Add ON UPDATE and ON DELETE clauses */ /* Add ON UPDATE and ON DELETE clauses, if needed */
switch (conForm->confupdtype) switch (conForm->confupdtype)
{ {
case FKCONSTR_ACTION_NOACTION: case FKCONSTR_ACTION_NOACTION:
string = ""; string = NULL; /* suppress default */
break; break;
case FKCONSTR_ACTION_RESTRICT: case FKCONSTR_ACTION_RESTRICT:
string = "RESTRICT"; string = "RESTRICT";
...@@ -659,16 +659,16 @@ pg_get_constraintdef(PG_FUNCTION_ARGS) ...@@ -659,16 +659,16 @@ pg_get_constraintdef(PG_FUNCTION_ARGS)
default: default:
elog(ERROR, "pg_get_constraintdef: Unknown confupdtype '%c' for constraint %u", elog(ERROR, "pg_get_constraintdef: Unknown confupdtype '%c' for constraint %u",
conForm->confupdtype, constraintId); conForm->confupdtype, constraintId);
string = ""; /* keep compiler quiet */ string = NULL; /* keep compiler quiet */
break; break;
} }
if (strlen(string) != 0) if (string)
appendStringInfo(&buf, " ON UPDATE %s", string); appendStringInfo(&buf, " ON UPDATE %s", string);
switch (conForm->confdeltype) switch (conForm->confdeltype)
{ {
case FKCONSTR_ACTION_NOACTION: case FKCONSTR_ACTION_NOACTION:
string = ""; string = NULL; /* suppress default */
break; break;
case FKCONSTR_ACTION_RESTRICT: case FKCONSTR_ACTION_RESTRICT:
string = "RESTRICT"; string = "RESTRICT";
...@@ -685,10 +685,10 @@ pg_get_constraintdef(PG_FUNCTION_ARGS) ...@@ -685,10 +685,10 @@ pg_get_constraintdef(PG_FUNCTION_ARGS)
default: default:
elog(ERROR, "pg_get_constraintdef: Unknown confdeltype '%c' for constraint %u", elog(ERROR, "pg_get_constraintdef: Unknown confdeltype '%c' for constraint %u",
conForm->confdeltype, constraintId); conForm->confdeltype, constraintId);
string = ""; /* keep compiler quiet */ string = NULL; /* keep compiler quiet */
break; break;
} }
if (strlen(string) != 0) if (string)
appendStringInfo(&buf, " ON DELETE %s", string); appendStringInfo(&buf, " ON DELETE %s", string);
if (conForm->condeferrable) if (conForm->condeferrable)
...@@ -2252,19 +2252,34 @@ get_rule_expr(Node *node, deparse_context *context, ...@@ -2252,19 +2252,34 @@ get_rule_expr(Node *node, deparse_context *context,
} }
break; break;
case T_ConstraintTest: case T_CoerceToDomain:
{ {
ConstraintTest *ctest = (ConstraintTest *) node; CoerceToDomain *ctest = (CoerceToDomain *) node;
Node *arg = (Node *) ctest->arg;
/* /*
* We assume that the operations of the constraint node * Any implicit coercion at the top level of the argument
* need not be explicitly represented in the output. * is presumably due to the domain's own internal typmod
* coercion, so do not force it to be shown.
*/ */
get_rule_expr((Node *) ctest->arg, context, showimplicit); if (ctest->coercionformat == COERCE_IMPLICIT_CAST &&
!showimplicit)
{
/* don't show the implicit cast */
get_rule_expr(arg, context, false);
}
else
{
appendStringInfoChar(buf, '(');
get_rule_expr(arg, context, false);
appendStringInfo(buf, ")::%s",
format_type_with_typemod(ctest->resulttype,
ctest->resulttypmod));
}
} }
break; break;
case T_ConstraintTestValue: case T_CoerceToDomainValue:
appendStringInfo(buf, "VALUE"); appendStringInfo(buf, "VALUE");
break; break;
...@@ -2444,7 +2459,8 @@ get_agg_expr(Aggref *aggref, deparse_context *context) ...@@ -2444,7 +2459,8 @@ get_agg_expr(Aggref *aggref, deparse_context *context)
* the expression tree has a length-coercion node atop a type-coercion node. * the expression tree has a length-coercion node atop a type-coercion node.
* *
* Note: avoid stripping a length-coercion node, since two successive * Note: avoid stripping a length-coercion node, since two successive
* coercions to different lengths aren't a no-op. * coercions to different lengths aren't a no-op. Also, never strip a
* CoerceToDomain node, even though it might be effectively just RelabelType.
*/ */
static Node * static Node *
strip_type_coercion(Node *expr, Oid resultType) strip_type_coercion(Node *expr, Oid resultType)
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/utils/cache/lsyscache.c,v 1.89 2003/01/15 19:35:44 tgl Exp $ * $Header: /cvsroot/pgsql/src/backend/utils/cache/lsyscache.c,v 1.90 2003/02/03 21:15:44 tgl Exp $
* *
* NOTES * NOTES
* Eventually, the index information should go through here, too. * Eventually, the index information should go through here, too.
...@@ -1012,6 +1012,33 @@ get_typstorage(Oid typid) ...@@ -1012,6 +1012,33 @@ get_typstorage(Oid typid)
return 'p'; return 'p';
} }
/*
* get_typtypmod
*
* Given the type OID, return the typtypmod field (domain's typmod
* for base type)
*/
int32
get_typtypmod(Oid typid)
{
HeapTuple tp;
tp = SearchSysCache(TYPEOID,
ObjectIdGetDatum(typid),
0, 0, 0);
if (HeapTupleIsValid(tp))
{
Form_pg_type typtup = (Form_pg_type) GETSTRUCT(tp);
int32 result;
result = typtup->typtypmod;
ReleaseSysCache(tp);
return result;
}
else
return -1;
}
/* /*
* get_typdefault * get_typdefault
* Given a type OID, return the type's default value, if any. * Given a type OID, return the type's default value, if any.
......
...@@ -37,7 +37,7 @@ ...@@ -37,7 +37,7 @@
* Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $Id: catversion.h,v 1.174 2003/01/28 22:13:36 tgl Exp $ * $Id: catversion.h,v 1.175 2003/02/03 21:15:44 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -53,6 +53,6 @@ ...@@ -53,6 +53,6 @@
*/ */
/* yyyymmddN */ /* yyyymmddN */
#define CATALOG_VERSION_NO 200301281 #define CATALOG_VERSION_NO 200302031
#endif #endif
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $Id: typecmds.h,v 1.4 2003/01/09 18:00:24 tgl Exp $ * $Id: typecmds.h,v 1.5 2003/02/03 21:15:44 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -32,6 +32,8 @@ extern void AlterDomainAddConstraint(List *names, Node *constr); ...@@ -32,6 +32,8 @@ extern void AlterDomainAddConstraint(List *names, Node *constr);
extern void AlterDomainDropConstraint(List *names, const char *constrName, extern void AlterDomainDropConstraint(List *names, const char *constrName,
DropBehavior behavior); DropBehavior behavior);
extern List *GetDomainConstraints(Oid typeOid);
extern void AlterTypeOwner(List *names, AclId newOwnerSysId); extern void AlterTypeOwner(List *names, AclId newOwnerSysId);
#endif /* TYPECMDS_H */ #endif /* TYPECMDS_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $Id: execnodes.h,v 1.92 2003/01/23 05:10:41 tgl Exp $ * $Id: execnodes.h,v 1.93 2003/02/03 21:15:44 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -114,7 +114,7 @@ typedef struct ExprContext ...@@ -114,7 +114,7 @@ typedef struct ExprContext
Datum *ecxt_aggvalues; /* precomputed values for Aggref nodes */ Datum *ecxt_aggvalues; /* precomputed values for Aggref nodes */
bool *ecxt_aggnulls; /* null flags for Aggref nodes */ bool *ecxt_aggnulls; /* null flags for Aggref nodes */
/* Value to substitute for ConstraintTestValue nodes in expression */ /* Value to substitute for CoerceToDomainValue nodes in expression */
Datum domainValue_datum; Datum domainValue_datum;
bool domainValue_isNull; bool domainValue_isNull;
...@@ -539,15 +539,37 @@ typedef struct CaseWhenState ...@@ -539,15 +539,37 @@ typedef struct CaseWhenState
} CaseWhenState; } CaseWhenState;
/* ---------------- /* ----------------
* ConstraintTestState node * CoerceToDomainState node
* ---------------- * ----------------
*/ */
typedef struct ConstraintTestState typedef struct CoerceToDomainState
{ {
ExprState xprstate; ExprState xprstate;
ExprState *arg; /* input expression */ ExprState *arg; /* input expression */
ExprState *check_expr; /* for CHECK test, a boolean expression */ /* Cached list of constraints that need to be checked */
} ConstraintTestState; List *constraints; /* list of DomainConstraintState nodes */
} CoerceToDomainState;
/*
* DomainConstraintState - one item to check during CoerceToDomain
*
* Note: this is just a Node, and not an ExprState, because it has no
* corresponding Expr to link to. Nonetheless it is part of an ExprState
* tree, so we give it a name following the xxxState convention.
*/
typedef enum DomainConstraintType
{
DOM_CONSTRAINT_NOTNULL,
DOM_CONSTRAINT_CHECK
} DomainConstraintType;
typedef struct DomainConstraintState
{
NodeTag type;
DomainConstraintType constrainttype; /* constraint type */
char *name; /* name of constraint (for error msgs) */
ExprState *check_expr; /* for CHECK, a boolean expression */
} DomainConstraintState;
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $Id: nodes.h,v 1.135 2003/01/20 18:55:00 tgl Exp $ * $Id: nodes.h,v 1.136 2003/02/03 21:15:44 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -114,8 +114,8 @@ typedef enum NodeTag ...@@ -114,8 +114,8 @@ typedef enum NodeTag
T_CaseWhen, T_CaseWhen,
T_NullTest, T_NullTest,
T_BooleanTest, T_BooleanTest,
T_ConstraintTest, T_CoerceToDomain,
T_ConstraintTestValue, T_CoerceToDomainValue,
T_TargetEntry, T_TargetEntry,
T_RangeTblRef, T_RangeTblRef,
T_JoinExpr, T_JoinExpr,
...@@ -136,7 +136,8 @@ typedef enum NodeTag ...@@ -136,7 +136,8 @@ typedef enum NodeTag
T_SubPlanState, T_SubPlanState,
T_CaseExprState, T_CaseExprState,
T_CaseWhenState, T_CaseWhenState,
T_ConstraintTestState, T_CoerceToDomainState,
T_DomainConstraintState,
/* /*
* TAGS FOR PLANNER NODES (relation.h) * TAGS FOR PLANNER NODES (relation.h)
......
...@@ -10,7 +10,7 @@ ...@@ -10,7 +10,7 @@
* Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $Id: primnodes.h,v 1.77 2003/01/10 21:08:15 tgl Exp $ * $Id: primnodes.h,v 1.78 2003/02/03 21:15:44 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -578,28 +578,22 @@ typedef struct BooleanTest ...@@ -578,28 +578,22 @@ typedef struct BooleanTest
} BooleanTest; } BooleanTest;
/* /*
* ConstraintTest * CoerceToDomain
* *
* ConstraintTest represents the operation of testing a value to see whether * CoerceToDomain represents the operation of coercing a value to a domain
* it meets a constraint. If so, the input value is returned as the result; * type. At runtime (and not before) the precise set of constraints to be
* if not, an error is raised. * checked will be determined. If the value passes, it is returned as the
* result; if not, an error is raised. Note that this is equivalent to
* RelabelType in the scenario where no constraints are applied.
*/ */
typedef struct CoerceToDomain
typedef enum ConstraintTestType
{
CONSTR_TEST_NOTNULL,
CONSTR_TEST_CHECK
} ConstraintTestType;
typedef struct ConstraintTest
{ {
Expr xpr; Expr xpr;
Expr *arg; /* input expression */ Expr *arg; /* input expression */
ConstraintTestType testtype; /* test type */ Oid resulttype; /* domain type ID (result type) */
char *name; /* name of constraint (for error msgs) */ int32 resulttypmod; /* output typmod (currently always -1) */
char *domname; /* name of domain (for error messages) */ CoercionForm coercionformat; /* how to display this node */
Expr *check_expr; /* for CHECK test, a boolean expression */ } CoerceToDomain;
} ConstraintTest;
/* /*
* Placeholder node for the value to be processed by a domain's check * Placeholder node for the value to be processed by a domain's check
...@@ -610,12 +604,12 @@ typedef struct ConstraintTest ...@@ -610,12 +604,12 @@ typedef struct ConstraintTest
* the domain itself. This is because we shouldn't consider the value to * the domain itself. This is because we shouldn't consider the value to
* be a member of the domain if we haven't yet checked its constraints. * be a member of the domain if we haven't yet checked its constraints.
*/ */
typedef struct ConstraintTestValue typedef struct CoerceToDomainValue
{ {
Expr xpr; Expr xpr;
Oid typeId; /* type for substituted value */ Oid typeId; /* type for substituted value */
int32 typeMod; /* typemod for substituted value */ int32 typeMod; /* typemod for substituted value */
} ConstraintTestValue; } CoerceToDomainValue;
/* /*
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $Id: parse_coerce.h,v 1.48 2002/10/24 22:09:00 tgl Exp $ * $Id: parse_coerce.h,v 1.49 2003/02/03 21:15:44 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -45,8 +45,8 @@ extern bool can_coerce_type(int nargs, Oid *input_typeids, Oid *target_typeids, ...@@ -45,8 +45,8 @@ extern bool can_coerce_type(int nargs, Oid *input_typeids, Oid *target_typeids,
CoercionContext ccontext); CoercionContext ccontext);
extern Node *coerce_type(Node *node, Oid inputTypeId, Oid targetTypeId, extern Node *coerce_type(Node *node, Oid inputTypeId, Oid targetTypeId,
CoercionContext ccontext, CoercionForm cformat); CoercionContext ccontext, CoercionForm cformat);
extern Node *coerce_type_constraints(Node *arg, Oid typeId, extern Node *coerce_to_domain(Node *arg, Oid baseTypeId, Oid typeId,
CoercionForm cformat); CoercionForm cformat);
extern Node *coerce_to_boolean(Node *node, const char *constructName); extern Node *coerce_to_boolean(Node *node, const char *constructName);
......
...@@ -6,7 +6,7 @@ ...@@ -6,7 +6,7 @@
* Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $Id: lsyscache.h,v 1.66 2003/01/15 19:35:47 tgl Exp $ * $Id: lsyscache.h,v 1.67 2003/02/03 21:15:45 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -54,6 +54,7 @@ extern void get_typlenbyval(Oid typid, int16 *typlen, bool *typbyval); ...@@ -54,6 +54,7 @@ extern void get_typlenbyval(Oid typid, int16 *typlen, bool *typbyval);
extern void get_typlenbyvalalign(Oid typid, int16 *typlen, bool *typbyval, extern void get_typlenbyvalalign(Oid typid, int16 *typlen, bool *typbyval,
char *typalign); char *typalign);
extern char get_typstorage(Oid typid); extern char get_typstorage(Oid typid);
extern int32 get_typtypmod(Oid typid);
extern Node *get_typdefault(Oid typid); extern Node *get_typdefault(Oid typid);
extern char get_typtype(Oid typid); extern char get_typtype(Oid typid);
extern Oid get_typ_typrelid(Oid typid); extern Oid get_typ_typrelid(Oid typid);
......
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
* procedural language * procedural language
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/pl/plpgsql/src/pl_exec.c,v 1.77 2003/01/21 22:06:12 tgl Exp $ * $Header: /cvsroot/pgsql/src/pl/plpgsql/src/pl_exec.c,v 1.78 2003/02/03 21:15:45 tgl Exp $
* *
* This software is copyrighted by Jan Wieck - Hamburg. * This software is copyrighted by Jan Wieck - Hamburg.
* *
...@@ -3634,6 +3634,9 @@ exec_simple_check_node(Node *node) ...@@ -3634,6 +3634,9 @@ exec_simple_check_node(Node *node)
case T_BooleanTest: case T_BooleanTest:
return exec_simple_check_node((Node *) ((BooleanTest *) node)->arg); return exec_simple_check_node((Node *) ((BooleanTest *) node)->arg);
case T_CoerceToDomain:
return exec_simple_check_node((Node *) ((CoerceToDomain *) node)->arg);
case T_List: case T_List:
{ {
List *expr = (List *) node; List *expr = (List *) node;
......
...@@ -257,14 +257,49 @@ ERROR: ALTER DOMAIN: Relation "domcontest" attribute "col1" contains values tha ...@@ -257,14 +257,49 @@ ERROR: ALTER DOMAIN: Relation "domcontest" attribute "col1" contains values tha
alter domain con add constraint t check (VALUE < 34); alter domain con add constraint t check (VALUE < 34);
alter domain con add check (VALUE > 0); alter domain con add check (VALUE > 0);
insert into domcontest values (-5); -- fails insert into domcontest values (-5); -- fails
ERROR: ExecEvalConstraintTest: Domain con constraint $1 failed ERROR: ExecEvalCoerceToDomain: Domain con constraint $1 failed
insert into domcontest values (42); -- fails insert into domcontest values (42); -- fails
ERROR: ExecEvalConstraintTest: Domain con constraint t failed ERROR: ExecEvalCoerceToDomain: Domain con constraint t failed
insert into domcontest values (5); insert into domcontest values (5);
alter domain con drop constraint t; alter domain con drop constraint t;
insert into domcontest values (-5); --fails insert into domcontest values (-5); --fails
ERROR: ExecEvalConstraintTest: Domain con constraint $1 failed ERROR: ExecEvalCoerceToDomain: Domain con constraint $1 failed
insert into domcontest values (42); insert into domcontest values (42);
-- Confirm ALTER DOMAIN with RULES.
create table domtab (col1 integer);
create domain dom as integer;
create view domview as select cast(col1 as dom) from domtab;
insert into domtab (col1) values (null);
insert into domtab (col1) values (5);
select * from domview;
col1
------
5
(2 rows)
alter domain dom set not null;
select * from domview; -- fail
ERROR: Domain dom does not allow NULL values
alter domain dom drop not null;
select * from domview;
col1
------
5
(2 rows)
alter domain dom add constraint domchkgt6 check(value > 6);
select * from domview; --fail
ERROR: ExecEvalCoerceToDomain: Domain dom constraint domchkgt6 failed
alter domain dom drop constraint domchkgt6 restrict;
select * from domview;
col1
------
5
(2 rows)
-- cleanup -- cleanup
drop domain ddef1 restrict; drop domain ddef1 restrict;
drop domain ddef2 restrict; drop domain ddef2 restrict;
......
...@@ -224,6 +224,26 @@ alter domain con drop constraint t; ...@@ -224,6 +224,26 @@ alter domain con drop constraint t;
insert into domcontest values (-5); --fails insert into domcontest values (-5); --fails
insert into domcontest values (42); insert into domcontest values (42);
-- Confirm ALTER DOMAIN with RULES.
create table domtab (col1 integer);
create domain dom as integer;
create view domview as select cast(col1 as dom) from domtab;
insert into domtab (col1) values (null);
insert into domtab (col1) values (5);
select * from domview;
alter domain dom set not null;
select * from domview; -- fail
alter domain dom drop not null;
select * from domview;
alter domain dom add constraint domchkgt6 check(value > 6);
select * from domview; --fail
alter domain dom drop constraint domchkgt6 restrict;
select * from domview;
-- cleanup -- cleanup
drop domain ddef1 restrict; drop domain ddef1 restrict;
drop domain ddef2 restrict; drop domain ddef2 restrict;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment