Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
P
Postgres FD Implementation
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Abuhujair Javed
Postgres FD Implementation
Commits
8f2f180f
Commit
8f2f180f
authored
Oct 11, 2006
by
Tom Lane
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Code review for LIKE INCLUDING CONSTRAINTS patch --- improve comments,
don't cheat on the raw-vs-cooked status of a constraint.
parent
3f166479
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
142 additions
and
106 deletions
+142
-106
src/backend/commands/tablecmds.c
src/backend/commands/tablecmds.c
+114
-83
src/backend/parser/analyze.c
src/backend/parser/analyze.c
+16
-11
src/include/nodes/parsenodes.h
src/include/nodes/parsenodes.h
+12
-12
No files found.
src/backend/commands/tablecmds.c
View file @
8f2f180f
...
@@ -8,7 +8,7 @@
...
@@ -8,7 +8,7 @@
*
*
*
*
* IDENTIFICATION
* IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/commands/tablecmds.c,v 1.20
4 2006/10/06 17:13:59 petere
Exp $
* $PostgreSQL: pgsql/src/backend/commands/tablecmds.c,v 1.20
5 2006/10/11 16:42:58 tgl
Exp $
*
*
*-------------------------------------------------------------------------
*-------------------------------------------------------------------------
*/
*/
...
@@ -163,6 +163,8 @@ static List *MergeAttributes(List *schema, List *supers, bool istemp,
...
@@ -163,6 +163,8 @@ static List *MergeAttributes(List *schema, List *supers, bool istemp,
List
**
supOids
,
List
**
supconstr
,
int
*
supOidCount
);
List
**
supOids
,
List
**
supconstr
,
int
*
supOidCount
);
static
void
MergeConstraintsIntoExisting
(
Relation
child_rel
,
Relation
parent_rel
);
static
void
MergeConstraintsIntoExisting
(
Relation
child_rel
,
Relation
parent_rel
);
static
void
MergeAttributesIntoExisting
(
Relation
child_rel
,
Relation
parent_rel
);
static
void
MergeAttributesIntoExisting
(
Relation
child_rel
,
Relation
parent_rel
);
static
void
add_nonduplicate_constraint
(
Constraint
*
cdef
,
ConstrCheck
*
check
,
int
*
ncheck
);
static
bool
change_varattnos_walker
(
Node
*
node
,
const
AttrNumber
*
newattno
);
static
bool
change_varattnos_walker
(
Node
*
node
,
const
AttrNumber
*
newattno
);
static
void
StoreCatalogInheritance
(
Oid
relationId
,
List
*
supers
);
static
void
StoreCatalogInheritance
(
Oid
relationId
,
List
*
supers
);
static
void
StoreCatalogInheritance1
(
Oid
relationId
,
Oid
parentOid
,
static
void
StoreCatalogInheritance1
(
Oid
relationId
,
Oid
parentOid
,
...
@@ -285,7 +287,6 @@ DefineRelation(CreateStmt *stmt, char relkind)
...
@@ -285,7 +287,6 @@ DefineRelation(CreateStmt *stmt, char relkind)
List
*
rawDefaults
;
List
*
rawDefaults
;
Datum
reloptions
;
Datum
reloptions
;
ListCell
*
listptr
;
ListCell
*
listptr
;
int
i
;
AttrNumber
attnum
;
AttrNumber
attnum
;
/*
/*
...
@@ -378,49 +379,35 @@ DefineRelation(CreateStmt *stmt, char relkind)
...
@@ -378,49 +379,35 @@ DefineRelation(CreateStmt *stmt, char relkind)
localHasOids
=
interpretOidsOption
(
stmt
->
options
);
localHasOids
=
interpretOidsOption
(
stmt
->
options
);
descriptor
->
tdhasoid
=
(
localHasOids
||
parentOidCount
>
0
);
descriptor
->
tdhasoid
=
(
localHasOids
||
parentOidCount
>
0
);
if
(
old_constraints
!=
NIL
)
if
(
old_constraints
||
stmt
->
constraints
)
{
{
ConstrCheck
*
check
=
(
ConstrCheck
*
)
ConstrCheck
*
check
;
palloc0
(
list_length
(
old_constraints
)
*
sizeof
(
ConstrCheck
));
int
ncheck
=
0
;
int
ncheck
=
0
;
/* make array that's certainly big enough */
check
=
(
ConstrCheck
*
)
palloc
((
list_length
(
old_constraints
)
+
list_length
(
stmt
->
constraints
))
*
sizeof
(
ConstrCheck
));
/* deal with constraints from MergeAttributes */
foreach
(
listptr
,
old_constraints
)
foreach
(
listptr
,
old_constraints
)
{
{
Constraint
*
cdef
=
(
Constraint
*
)
lfirst
(
listptr
);
Constraint
*
cdef
=
(
Constraint
*
)
lfirst
(
listptr
);
bool
dup
=
false
;
if
(
cdef
->
contype
!=
CONSTR_CHECK
)
continue
;
Assert
(
cdef
->
name
!=
NULL
);
Assert
(
cdef
->
raw_expr
==
NULL
&&
cdef
->
cooked_expr
!=
NULL
);
if
(
cdef
->
contype
==
CONSTR_CHECK
)
add_nonduplicate_constraint
(
cdef
,
check
,
&
ncheck
);
}
/*
/*
* In multiple-inheritance situations, it's possible to inherit
* analyze.c might have passed some precooked constraints too,
* the same grandparent constraint through multiple parents.
* due to LIKE tab INCLUDING CONSTRAINTS
* Hence, discard inherited constraints that match as to both name
* and expression. Otherwise, gripe if the names conflict.
*/
*/
for
(
i
=
0
;
i
<
ncheck
;
i
++
)
foreach
(
listptr
,
stmt
->
constraints
)
{
if
(
strcmp
(
check
[
i
].
ccname
,
cdef
->
name
)
!=
0
)
continue
;
if
(
strcmp
(
check
[
i
].
ccbin
,
cdef
->
cooked_expr
)
==
0
)
{
dup
=
true
;
break
;
}
ereport
(
ERROR
,
(
errcode
(
ERRCODE_DUPLICATE_OBJECT
),
errmsg
(
"duplicate check constraint name
\"
%s
\"
"
,
cdef
->
name
)));
}
if
(
!
dup
)
{
{
check
[
ncheck
].
ccname
=
cdef
->
name
;
Constraint
*
cdef
=
(
Constraint
*
)
lfirst
(
listptr
)
;
check
[
ncheck
].
ccbin
=
pstrdup
(
cdef
->
cooked_expr
);
ncheck
++
;
if
(
cdef
->
contype
==
CONSTR_CHECK
&&
cdef
->
cooked_expr
!=
NULL
)
}
add_nonduplicate_constraint
(
cdef
,
check
,
&
ncheck
);
}
}
/* if we found any, insert 'em into the descriptor */
if
(
ncheck
>
0
)
if
(
ncheck
>
0
)
{
{
if
(
descriptor
->
constr
==
NULL
)
if
(
descriptor
->
constr
==
NULL
)
...
@@ -1118,66 +1105,57 @@ MergeAttributes(List *schema, List *supers, bool istemp,
...
@@ -1118,66 +1105,57 @@ MergeAttributes(List *schema, List *supers, bool istemp,
return
schema
;
return
schema
;
}
}
/*
/*
* Varattnos of pg_constraint.conbin must be rewritten when subclasses inherit
* In multiple-inheritance situations, it's possible to inherit
* constraints from parent classes, since the inherited attributes could
* the same grandparent constraint through multiple parents.
* be given different column numbers in multiple-inheritance cases.
* Hence, we want to discard inherited constraints that match as to
*
* both name and expression. Otherwise, gripe if there are conflicting
* Note that the passed node tree is modified in place!
* names. Nonconflicting constraints are added to the array check[]
*
* of length *ncheck ... caller must ensure there is room!
* This function is used elsewhere such as in analyze.c
*
*/
*/
static
void
void
add_nonduplicate_constraint
(
Constraint
*
cdef
,
ConstrCheck
*
check
,
int
*
ncheck
)
change_varattnos_of_a_node
(
Node
*
node
,
const
AttrNumber
*
newattno
)
{
{
change_varattnos_walker
(
node
,
newattno
);
int
i
;
}
/* Generate a map for change_varattnos_of_a_node from two tupledesc's. */
AttrNumber
*
/* Should only see precooked constraints here */
varattnos_map
(
TupleDesc
old
,
TupleDesc
new
)
Assert
(
cdef
->
contype
==
CONSTR_CHECK
);
{
Assert
(
cdef
->
name
!=
NULL
);
int
i
,
Assert
(
cdef
->
raw_expr
==
NULL
&&
cdef
->
cooked_expr
!=
NULL
);
j
;
AttrNumber
*
attmap
=
palloc0
(
sizeof
(
AttrNumber
)
*
old
->
natts
);
for
(
i
=
1
;
i
<=
old
->
natts
;
i
++
)
for
(
i
=
0
;
i
<
*
ncheck
;
i
++
)
{
if
(
old
->
attrs
[
i
-
1
]
->
attisdropped
)
{
{
attmap
[
i
-
1
]
=
0
;
if
(
strcmp
(
check
[
i
].
ccname
,
cdef
->
name
)
!=
0
)
continue
;
continue
;
if
(
strcmp
(
check
[
i
].
ccbin
,
cdef
->
cooked_expr
)
==
0
)
return
;
/* duplicate constraint, so ignore it */
ereport
(
ERROR
,
(
errcode
(
ERRCODE_DUPLICATE_OBJECT
),
errmsg
(
"duplicate check constraint name
\"
%s
\"
"
,
cdef
->
name
)));
}
}
for
(
j
=
1
;
j
<=
new
->
natts
;
j
++
)
/* No match on name, so add it to array */
if
(
!
strcmp
(
NameStr
(
old
->
attrs
[
i
-
1
]
->
attname
),
NameStr
(
new
->
attrs
[
j
-
1
]
->
attname
)))
check
[
*
ncheck
].
ccname
=
cdef
->
name
;
attmap
[
i
-
1
]
=
j
;
check
[
*
ncheck
].
ccbin
=
pstrdup
(
cdef
->
cooked_expr
);
}
(
*
ncheck
)
++
;
return
attmap
;
}
}
/*
/*
* Generate a map for change_varattnos_of_a_node from a tupledesc and a list of
* Replace varattno values in an expression tree according to the given
* ColumnDefs
* map array, that is, varattno N is replaced by newattno[N-1]. It is
* caller's responsibility to ensure that the array is long enough to
* define values for all user varattnos present in the tree. System column
* attnos remain unchanged.
*
* Note that the passed node tree is modified in-place!
*/
*/
AttrNumber
*
void
varattnos_map_schema
(
TupleDesc
old
,
List
*
schema
)
change_varattnos_of_a_node
(
Node
*
node
,
const
AttrNumber
*
newattno
)
{
{
int
i
;
/* no setup needed, so away we go */
AttrNumber
*
attmap
=
palloc0
(
sizeof
(
AttrNumber
)
*
old
->
natts
);
(
void
)
change_varattnos_walker
(
node
,
newattno
);
for
(
i
=
1
;
i
<=
old
->
natts
;
i
++
)
{
if
(
old
->
attrs
[
i
-
1
]
->
attisdropped
)
{
attmap
[
i
-
1
]
=
0
;
continue
;
}
attmap
[
i
-
1
]
=
findAttrByName
(
NameStr
(
old
->
attrs
[
i
-
1
]
->
attname
),
schema
);
}
return
attmap
;
}
}
static
bool
static
bool
...
@@ -1206,6 +1184,59 @@ change_varattnos_walker(Node *node, const AttrNumber *newattno)
...
@@ -1206,6 +1184,59 @@ change_varattnos_walker(Node *node, const AttrNumber *newattno)
(
void
*
)
newattno
);
(
void
*
)
newattno
);
}
}
/*
* Generate a map for change_varattnos_of_a_node from old and new TupleDesc's,
* matching according to column name.
*/
AttrNumber
*
varattnos_map
(
TupleDesc
old
,
TupleDesc
new
)
{
AttrNumber
*
attmap
;
int
i
,
j
;
attmap
=
(
AttrNumber
*
)
palloc0
(
sizeof
(
AttrNumber
)
*
old
->
natts
);
for
(
i
=
1
;
i
<=
old
->
natts
;
i
++
)
{
if
(
old
->
attrs
[
i
-
1
]
->
attisdropped
)
continue
;
/* leave the entry as zero */
for
(
j
=
1
;
j
<=
new
->
natts
;
j
++
)
{
if
(
strcmp
(
NameStr
(
old
->
attrs
[
i
-
1
]
->
attname
),
NameStr
(
new
->
attrs
[
j
-
1
]
->
attname
))
==
0
)
{
attmap
[
i
-
1
]
=
j
;
break
;
}
}
}
return
attmap
;
}
/*
* Generate a map for change_varattnos_of_a_node from a TupleDesc and a list
* of ColumnDefs
*/
AttrNumber
*
varattnos_map_schema
(
TupleDesc
old
,
List
*
schema
)
{
AttrNumber
*
attmap
;
int
i
;
attmap
=
(
AttrNumber
*
)
palloc0
(
sizeof
(
AttrNumber
)
*
old
->
natts
);
for
(
i
=
1
;
i
<=
old
->
natts
;
i
++
)
{
if
(
old
->
attrs
[
i
-
1
]
->
attisdropped
)
continue
;
/* leave the entry as zero */
attmap
[
i
-
1
]
=
findAttrByName
(
NameStr
(
old
->
attrs
[
i
-
1
]
->
attname
),
schema
);
}
return
attmap
;
}
/*
/*
* StoreCatalogInheritance
* StoreCatalogInheritance
* Updates the system catalogs with proper inheritance information.
* Updates the system catalogs with proper inheritance information.
...
...
src/backend/parser/analyze.c
View file @
8f2f180f
...
@@ -6,7 +6,7 @@
...
@@ -6,7 +6,7 @@
* Portions Copyright (c) 1996-2006, PostgreSQL Global Development Group
* Portions Copyright (c) 1996-2006, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California
* Portions Copyright (c) 1994, Regents of the University of California
*
*
* $PostgreSQL: pgsql/src/backend/parser/analyze.c,v 1.35
2 2006/10/04 00:29:55 momjian
Exp $
* $PostgreSQL: pgsql/src/backend/parser/analyze.c,v 1.35
3 2006/10/11 16:42:59 tgl
Exp $
*
*
*-------------------------------------------------------------------------
*-------------------------------------------------------------------------
*/
*/
...
@@ -1286,12 +1286,10 @@ transformInhRelation(ParseState *pstate, CreateStmtContext *cxt,
...
@@ -1286,12 +1286,10 @@ transformInhRelation(ParseState *pstate, CreateStmtContext *cxt,
InhRelation
*
inhRelation
)
InhRelation
*
inhRelation
)
{
{
AttrNumber
parent_attno
;
AttrNumber
parent_attno
;
Relation
relation
;
Relation
relation
;
TupleDesc
tupleDesc
;
TupleDesc
tupleDesc
;
TupleConstr
*
constr
;
TupleConstr
*
constr
;
AclResult
aclresult
;
AclResult
aclresult
;
bool
including_defaults
=
false
;
bool
including_defaults
=
false
;
bool
including_constraints
=
false
;
bool
including_constraints
=
false
;
bool
including_indexes
=
false
;
bool
including_indexes
=
false
;
...
@@ -1342,15 +1340,18 @@ transformInhRelation(ParseState *pstate, CreateStmtContext *cxt,
...
@@ -1342,15 +1340,18 @@ transformInhRelation(ParseState *pstate, CreateStmtContext *cxt,
including_indexes
=
false
;
including_indexes
=
false
;
break
;
break
;
default:
default:
elog
(
ERROR
,
"unrecognized CREATE TABLE LIKE option: %d"
,
option
);
elog
(
ERROR
,
"unrecognized CREATE TABLE LIKE option: %d"
,
option
);
}
}
}
}
if
(
including_indexes
)
if
(
including_indexes
)
elog
(
ERROR
,
"TODO"
);
ereport
(
ERROR
,
(
errcode
(
ERRCODE_FEATURE_NOT_SUPPORTED
),
errmsg
(
"LIKE INCLUDING INDEXES is not implemented"
)));
/*
/*
* Insert the
inherit
ed attributes into the cxt for the new table
* Insert the
copi
ed attributes into the cxt for the new table
* definition.
* definition.
*/
*/
for
(
parent_attno
=
1
;
parent_attno
<=
tupleDesc
->
natts
;
for
(
parent_attno
=
1
;
parent_attno
<=
tupleDesc
->
natts
;
...
@@ -1367,7 +1368,7 @@ transformInhRelation(ParseState *pstate, CreateStmtContext *cxt,
...
@@ -1367,7 +1368,7 @@ transformInhRelation(ParseState *pstate, CreateStmtContext *cxt,
continue
;
continue
;
/*
/*
* Create a new
inherited column
.
* Create a new
column, which is marked as NOT inherited
.
*
*
* For constraints, ONLY the NOT NULL constraint is inherited by the
* For constraints, ONLY the NOT NULL constraint is inherited by the
* new column definition per SQL99.
* new column definition per SQL99.
...
@@ -1389,7 +1390,7 @@ transformInhRelation(ParseState *pstate, CreateStmtContext *cxt,
...
@@ -1389,7 +1390,7 @@ transformInhRelation(ParseState *pstate, CreateStmtContext *cxt,
cxt
->
columns
=
lappend
(
cxt
->
columns
,
def
);
cxt
->
columns
=
lappend
(
cxt
->
columns
,
def
);
/*
/*
* Copy default
if any,
and the default has been requested
* Copy default
, if present
and the default has been requested
*/
*/
if
(
attribute
->
atthasdef
&&
including_defaults
)
if
(
attribute
->
atthasdef
&&
including_defaults
)
{
{
...
@@ -1419,10 +1420,14 @@ transformInhRelation(ParseState *pstate, CreateStmtContext *cxt,
...
@@ -1419,10 +1420,14 @@ transformInhRelation(ParseState *pstate, CreateStmtContext *cxt,
}
}
}
}
/*
* Copy CHECK constraints if requested, being careful to adjust
* attribute numbers
*/
if
(
including_constraints
&&
tupleDesc
->
constr
)
if
(
including_constraints
&&
tupleDesc
->
constr
)
{
{
int
ccnum
;
AttrNumber
*
attmap
=
varattnos_map_schema
(
tupleDesc
,
cxt
->
columns
);
AttrNumber
*
attmap
=
varattnos_map_schema
(
tupleDesc
,
cxt
->
columns
);
int
ccnum
;
for
(
ccnum
=
0
;
ccnum
<
tupleDesc
->
constr
->
num_check
;
ccnum
++
)
for
(
ccnum
=
0
;
ccnum
<
tupleDesc
->
constr
->
num_check
;
ccnum
++
)
{
{
...
@@ -1435,8 +1440,8 @@ transformInhRelation(ParseState *pstate, CreateStmtContext *cxt,
...
@@ -1435,8 +1440,8 @@ transformInhRelation(ParseState *pstate, CreateStmtContext *cxt,
n
->
contype
=
CONSTR_CHECK
;
n
->
contype
=
CONSTR_CHECK
;
n
->
name
=
pstrdup
(
ccname
);
n
->
name
=
pstrdup
(
ccname
);
n
->
raw_expr
=
ccbin_node
;
n
->
raw_expr
=
NULL
;
n
->
cooked_expr
=
NULL
;
n
->
cooked_expr
=
nodeToString
(
ccbin_node
)
;
n
->
indexspace
=
NULL
;
n
->
indexspace
=
NULL
;
cxt
->
ckconstraints
=
lappend
(
cxt
->
ckconstraints
,
(
Node
*
)
n
);
cxt
->
ckconstraints
=
lappend
(
cxt
->
ckconstraints
,
(
Node
*
)
n
);
}
}
...
...
src/include/nodes/parsenodes.h
View file @
8f2f180f
...
@@ -7,7 +7,7 @@
...
@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2006, PostgreSQL Global Development Group
* Portions Copyright (c) 1996-2006, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California
* Portions Copyright (c) 1994, Regents of the University of California
*
*
* $PostgreSQL: pgsql/src/include/nodes/parsenodes.h,v 1.33
1 2006/10/04 00:30:09 momjian
Exp $
* $PostgreSQL: pgsql/src/include/nodes/parsenodes.h,v 1.33
2 2006/10/11 16:42:59 tgl
Exp $
*
*
*-------------------------------------------------------------------------
*-------------------------------------------------------------------------
*/
*/
...
@@ -414,9 +414,19 @@ typedef struct InhRelation
...
@@ -414,9 +414,19 @@ typedef struct InhRelation
{
{
NodeTag
type
;
NodeTag
type
;
RangeVar
*
relation
;
RangeVar
*
relation
;
List
*
options
;
List
*
options
;
/* integer List of CreateStmtLikeOption */
}
InhRelation
;
}
InhRelation
;
typedef
enum
CreateStmtLikeOption
{
CREATE_TABLE_LIKE_INCLUDING_DEFAULTS
,
CREATE_TABLE_LIKE_EXCLUDING_DEFAULTS
,
CREATE_TABLE_LIKE_INCLUDING_CONSTRAINTS
,
CREATE_TABLE_LIKE_EXCLUDING_CONSTRAINTS
,
CREATE_TABLE_LIKE_INCLUDING_INDEXES
,
CREATE_TABLE_LIKE_EXCLUDING_INDEXES
}
CreateStmtLikeOption
;
/*
/*
* IndexElem - index parameters (used in CREATE INDEX)
* IndexElem - index parameters (used in CREATE INDEX)
*
*
...
@@ -1055,16 +1065,6 @@ typedef struct CreateStmt
...
@@ -1055,16 +1065,6 @@ typedef struct CreateStmt
char
*
tablespacename
;
/* table space to use, or NULL */
char
*
tablespacename
;
/* table space to use, or NULL */
}
CreateStmt
;
}
CreateStmt
;
typedef
enum
CreateStmtLikeOption
{
CREATE_TABLE_LIKE_INCLUDING_DEFAULTS
,
CREATE_TABLE_LIKE_EXCLUDING_DEFAULTS
,
CREATE_TABLE_LIKE_INCLUDING_CONSTRAINTS
,
CREATE_TABLE_LIKE_EXCLUDING_CONSTRAINTS
,
CREATE_TABLE_LIKE_INCLUDING_INDEXES
,
CREATE_TABLE_LIKE_EXCLUDING_INDEXES
}
CreateStmtLikeOption
;
/* ----------
/* ----------
* Definitions for plain (non-FOREIGN KEY) constraints in CreateStmt
* Definitions for plain (non-FOREIGN KEY) constraints in CreateStmt
*
*
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment