Re: Referential integrity (foreign keys) across multiple tables

From: "Aaron Bono" <postgresql(at)aranya(dot)com>
To: "Richard Jones" <rich(at)annexia(dot)org>
Cc: pgsql-sql(at)postgresql(dot)org
Subject: Re: Referential integrity (foreign keys) across multiple tables
Date: 2006-07-24 17:26:15
Message-ID: bf05e51c0607241026y5404fb9do86a2af7f1cde74ac@mail.gmail.com
Views: Raw Message | Whole Thread | Download mbox | Resend email
Thread:
Lists: pgsql-sql

On 7/22/06, Richard Jones <rich(at)annexia(dot)org> wrote:
>
>
> Simplified schema:
>
> create table hosts (
> id serial primary key,
> hostname text not null
> );
>
> create table pages (
> id serial primary key,
> hostid int not null references hosts (id),
> url text not null,
> unique (hostid, url)
> );
>
> create table page_contents (
> pageid int not null references pages (id),
> section text not null
> );
>
> (There are many hosts, many pages per host, and many page_contents
> sections per page).
>
> Now I want to add a column to page_contents, say called link_name,
> which is going to reference the pages.url column for the particular
> host that this page belongs to.
>
> Something like:
>
> alter table page_contents add link_name text;
> alter table page_contents
> add constraint foo foreign key (p.hostid, link_name)
> references pages (hostid, url)
> where p.id = pageid;

The easiest, and not necessarily elegant, way to acomplish this is to create
linkid rather than link_name and make it a foreign key to pages.id. Then
add a trigger that checks to make sure the pages you link to from
page_contents to pages is for the same host. If not, raise an exception.

Another option is to do this:

create table page_contents (
hostid int not null,
url text not null,
linkhostid int,
linkurl text,
section text not null,
foreign key (hostid, url) references pages (hostid, url),
foreign key (linkhostid, linkurl) references pages (hostid, url)
);

Or if you really want to restructure things:

create table hosts (
id serial primary key,
hostname text not null
);

create table pages (
id serial primary key,
url text not null,
unique (url)
);

create table page_contents (
pageid int not null references pages (id),
hostsid int not null references hosts (id),
linkpageid int references pages(id),
section text not null
);

That should give you some options to play with.

As a side comment, you should also name your primary key columns more
meaningfully. Use hosts.hostsid and pages.pagesid, not hosts.id and
pages.id. When you begin writing large queries, the use of the column name
id all over the place will make your queries more prone to error, harder to
read and harder to write.

==================================================================
Aaron Bono
Aranya Software Technologies, Inc.
http://www.aranya.com
==================================================================

In response to

Responses

Browse pgsql-sql by date

  From Date Subject
Next Message Bruno Wolff III 2006-07-24 17:51:48 Re: Referential integrity (foreign keys) across multiple tables
Previous Message Aaron Bono 2006-07-24 15:56:13 Re: System catalog table privileges