On Wed, 5 Jun 2024 at 22:57, Paul Jungwirth <p...@illuminatedcomputing.com> 
wrote:
>
> On Thu, May 9, 2024 at 5:44 PM Matthias van de Meent 
> <boekewurm+postg...@gmail.com> wrote:
>  > Additionally, because I can't create my own non-constraint-backing
>  > unique GIST indexes, I can't pre-create my unique constraints
>  > CONCURRENTLY as one could do for the non-temporal case
>
> We talked about this a bit at pgconf.dev. I would like to implement it, since 
> I agree it is an
> important workflow to support. Here are some thoughts about what would need 
> to be done.
>
> First we could take a small step: allow non-temporal UNIQUE GiST indexes. 
> This is possible according
> to [1], but in the past we had no way of knowing which strategy number an 
> opclass was using for
> equality. With the stratnum support proc introduced by 6db4598fcb (reverted 
> for v17), we could
> change amcanunique to true for the GiST AM handler. If the index's opclasses 
> had that sproc and it
> gave non-zero for RTEqualStrategyNumber, we would have a reliable "definition 
> of uniqueness". UNIQUE
> GiST indexes would raise an error if they detected a duplicate record.

Cool.

> But that is just regular non-temporal indexes. To avoid a long table lock 
> you'd need a way to build
> the index that is not just unique, but also does exclusion based on &&.  We 
> could borrow syntax from
> SQL:2011 and allow `CREATE INDEX idx ON t (id, valid_at WITHOUT OVERLAPS)`. 
> But since CREATE INDEX
> is a lower-level concept than a constraint, it'd be better to do something 
> more general. You can
> already give opclasses for each indexed column. How about allowing operators 
> as well? For instance
> `CREATE UNIQUE INDEX idx ON t (id WITH =, valid_at WITH &&)`? Then the index 
> would know to enforce
> those rules.

I think this looks fine. I'd like it even better if we could default
to the equality operator that's used by the type's default btree
opclass in this syntax; that'd make CREATE UNIQUE INDEX much less
awkward for e.g. hash indexes.

> This is the same data we store today in pg_constraint.conexclops. So that 
> would get
> moved/copied to pg_index (probably moved).

I'd keep the pg_constraint.conexclops around: People are inevitably
going to want to keep the current exclusion constraints' handling of
duplicate empty ranges, which is different from expectations we see
for UNIQUE INDEX's handling.

> Then when you add the constraint, what is the syntax? Today when you say 
> PRIMARY KEY/UNIQUE USING
> INDEX, you don't give the column names. So how do we know it's WITHOUT 
> OVERLAPS? I guess if the
> underlying index has (foo WITH = [, bar WITH =], baz WITH &&) we just assume 
> the user wants WITHOUT
> OVERLAPS, and otherwise they want a regular PK/UQ constraint?

Presumably you would know this based on the pg_index.indisunique flag?

> In addition this workflow only works if you can CREATE INDEX CONCURRENTLY. 
> I'm not sure yet if we'll
> have problems there. I noticed that for REINDEX at least, there were plans in 
> 2012 to support
> exclusion-constraint indexes,[2] but when the patch was committed in 2019 
> they had been dropped,
> with plans to add support eventually.[3] Today they are still not supported. 
> Maybe whatever caused
> problems for REINDEX isn't an issue for just INDEX, but it would take more 
> research to find out.

I don't quite see where exclusion constraints get into the picture?
Isn't this about unique indexes, not exclusion constraints? I
understand exclusion constraints are backed by indexes, but that
doesn't have to make it a unique index, right? I mean, currently, you
can write an exclusion constraint that makes sure that all rows with a
certain prefix have the same suffix columns (given a btree-esque index
type with <> -operator support), which seems exactly opposite of what
unique indexes should do.

Kind regards,

Matthias van de Meent
Neon (https://neon.tech)


Reply via email to