[Date Prev][Date Next] [Thread Prev][Thread Next] [Date Index] [Thread Index]

[PATCH 1/2] [setup] Add debversion PostgreSQL type



This type (derived from TEXT) supports the same comparison and
checking options as dpkg --compare-versions, using the
Dpkg::Version Perl implementation as the basis for the PL/Perl
functions the native PostgreSQL operators are based upon.

Signed-off-by: Roger Leigh <rleigh@debian.org>
---
 setup/version.sql |  378 +++++++++++++++++++++++++++++++++++++++++++++++++++++
 1 files changed, 378 insertions(+), 0 deletions(-)
 create mode 100644 setup/version.sql

diff --git a/setup/version.sql b/setup/version.sql
new file mode 100644
index 0000000..1d2f4e5
--- /dev/null
+++ b/setup/version.sql
@@ -0,0 +1,378 @@
+--- WannaBuild Database Schema for PostgreSQL                        -*- sql -*-
+--- Debian version type and operators
+---
+--- Code derived from Dpkg::Version:
+--- Copyright © Colin Watson <cjwatson@debian.org>
+--- Copyright © Ian Jackson <iwj@debian.org>
+--- Copyright © 2007 by Don Armstrong <don@donarmstrong.com>
+---
+--- PostgreSQL SQL, PL/pgSQL and PL/Perl:
+--- Copyright © 2008 Roger Leigh <rleigh@debian.org>
+---
+--- This program is free software: you can redistribute it and/or modify
+--- it under the terms of the GNU General Public License as published by
+--- the Free Software Foundation, either version 2 of the License, or
+--- (at your option) any later version.
+---
+--- This program is distributed in the hope that it will be useful, but
+--- WITHOUT ANY WARRANTY; without even the implied warranty of
+--- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+--- General Public License for more details.
+---
+--- You should have received a copy of the GNU General Public License
+--- along with this program.  If not, see
+--- <http://www.gnu.org/licenses/>.
+
+SET SESSION plperl.use_strict TO 't';
+
+CREATE DOMAIN debversion AS TEXT;
+COMMENT ON DOMAIN debversion IS 'Debian package version number';
+
+ALTER DOMAIN debversion
+  ADD CONSTRAINT debversion_syntax
+    CHECK (VALUE !~ '[^-+:.0-9a-zA-Z~]');
+
+-- From Dpkg::Version::parseversion
+CREATE OR REPLACE FUNCTION debversion_split (debversion)
+  RETURNS text[] AS $$
+    my $ver = shift;
+    my %verhash;
+    if ($ver =~ /:/)
+    {
+        $ver =~ /^(\d+):(.+)/ or die "bad version number '$ver'";
+        $verhash{epoch} = $1;
+        $ver = $2;
+    }
+    else
+    {
+        $verhash{epoch} = 0;
+    }
+    if ($ver =~ /(.+)-(.*)$/)
+    {
+        $verhash{version} = $1;
+        $verhash{revision} = $2;
+    }
+    else
+    {
+        $verhash{version} = $ver;
+        $verhash{revision} = 0;
+    }
+
+    return [$verhash{'epoch'}, $verhash{'version'}, $verhash{'revision'}];
+$$
+  LANGUAGE plperl
+  IMMUTABLE STRICT;
+COMMENT ON FUNCTION debversion_split (debversion)
+  IS 'Split debian version into epoch, upstream version and revision';
+
+CREATE OR REPLACE FUNCTION debversion_epoch (version debversion)
+  RETURNS text AS $$
+DECLARE
+  split text[];
+BEGIN
+  split := debversion_split(version);
+  RETURN split[1];
+END;
+$$
+  LANGUAGE plpgsql
+  IMMUTABLE STRICT;
+COMMENT ON FUNCTION debversion_epoch (debversion)
+  IS 'Get debian version epoch';
+
+CREATE OR REPLACE FUNCTION debversion_version (version debversion)
+  RETURNS text AS $$
+DECLARE
+  split text[];
+BEGIN
+  split := debversion_split(version);
+  RETURN split[2];
+END;
+$$
+  LANGUAGE plpgsql
+  IMMUTABLE STRICT;
+COMMENT ON FUNCTION debversion_version (debversion)
+  IS 'Get debian version upstream version';
+
+CREATE OR REPLACE FUNCTION debversion_revision (version debversion)
+  RETURNS text AS $$
+DECLARE
+  split text[];
+BEGIN
+  split := debversion_split(version);
+  RETURN split[3];
+END;
+$$
+  LANGUAGE plpgsql
+  IMMUTABLE STRICT;
+COMMENT ON FUNCTION debversion_revision (debversion)
+  IS 'Get debian version revision';
+
+-- From Dpkg::Version::parseversion
+CREATE OR REPLACE FUNCTION debversion_compare_single (version1 text,
+       	  	  	   			      version2 text)
+  RETURNS integer AS $$
+     sub order{
+	  my ($x) = @_;
+	  ##define order(x) ((x) == '~' ? -1 \
+	  #           : cisdigit((x)) ? 0 \
+	  #           : !(x) ? 0 \
+	  #           : cisalpha((x)) ? (x) \
+	  #           : (x) + 256)
+	  # This comparison is out of dpkg's order to avoid
+	  # comparing things to undef and triggering warnings.
+	  if (not defined $x or not length $x) {
+	       return 0;
+	  }
+	  elsif ($x eq '~') {
+	       return -1;
+	  }
+	  elsif ($x =~ /^\d$/) {
+	       return 0;
+	  }
+	  elsif ($x =~ /^[A-Z]$/i) {
+	       return ord($x);
+	  }
+	  else {
+	       return ord($x) + 256;
+	  }
+     }
+
+     sub next_elem(\@){
+	  my $a = shift;
+	  return @{$a} ? shift @{$a} : undef;
+     }
+     my ($val, $ref) = @_;
+     $val = "" if not defined $val;
+     $ref = "" if not defined $ref;
+     my @val = split //,$val;
+     my @ref = split //,$ref;
+     my $vc = next_elem @val;
+     my $rc = next_elem @ref;
+     while (defined $vc or defined $rc) {
+	  my $first_diff = 0;
+	  while ((defined $vc and $vc !~ /^\d$/) or
+		 (defined $rc and $rc !~ /^\d$/)) {
+	       my $vo = order($vc); my $ro = order($rc);
+	       # Unlike dpkg's verrevcmp, we only return 1 or -1 here.
+	       return (($vo - $ro > 0) ? 1 : -1) if $vo != $ro;
+	       $vc = next_elem @val; $rc = next_elem @ref;
+	  }
+	  while (defined $vc and $vc eq '0') {
+	       $vc = next_elem @val;
+	  }
+	  while (defined $rc and $rc eq '0') {
+	       $rc = next_elem @ref;
+	  }
+	  while (defined $vc and $vc =~ /^\d$/ and
+		 defined $rc and $rc =~ /^\d$/) {
+	       $first_diff = ord($vc) - ord($rc) if !$first_diff;
+	       $vc = next_elem @val; $rc = next_elem @ref;
+	  }
+	  return 1 if defined $vc and $vc =~ /^\d$/;
+	  return -1 if defined $rc and $rc =~ /^\d$/;
+	  return (($first_diff  > 0) ? 1 : -1) if $first_diff;
+     }
+     return 0;
+$$
+  LANGUAGE plperl
+  IMMUTABLE STRICT;
+COMMENT ON FUNCTION debversion_compare_single (text, text)
+  IS 'Compare upstream or revision parts of Debian versions';
+
+-- Logic only derived from Dpkg::Version::parseversion
+CREATE OR REPLACE FUNCTION debversion_compare (version1 debversion,
+       	  	  	   		       version2 debversion)
+  RETURNS integer AS $$
+DECLARE
+  split1 text[];
+  split2 text[];
+  result integer;
+BEGIN
+  result := 0;
+  split1 := debversion_split(version1);
+  split2 := debversion_split(version2);
+
+  -- RAISE NOTICE 'Version 1: %', version1;
+  -- RAISE NOTICE 'Version 2: %', version2;
+  -- RAISE NOTICE 'Split 1: %', split1;
+  -- RAISE NOTICE 'Split 2: %', split2;
+
+  IF split1[1] > split2[1] THEN
+    result := 1;
+  ELSIF split1[1] < split2[1] THEN
+    result := -1;
+  ELSE
+    result := debversion_compare_single(split1[2], split2[2]);
+    IF result = 0 THEN
+      result := debversion_compare_single(split1[3], split2[3]);
+    END IF;
+  END IF;
+
+  RETURN result;
+END;
+$$
+  LANGUAGE plpgsql
+  IMMUTABLE STRICT;
+COMMENT ON FUNCTION debversion_compare (debversion, debversion)
+  IS 'Compare Debian versions';
+
+CREATE OR REPLACE FUNCTION debversion_eq (version1 debversion,
+       	  	  	   		  version2 debversion)
+  RETURNS boolean AS $$
+DECLARE
+  comp integer;
+  result boolean;
+BEGIN
+  comp := debversion_compare(version1, version2);
+  result := comp = 0;
+  RETURN result;
+END;
+$$
+  LANGUAGE plpgsql
+  IMMUTABLE STRICT;
+COMMENT ON FUNCTION debversion_eq (debversion, debversion)
+  IS 'debversion equal';
+
+CREATE OR REPLACE FUNCTION debversion_ne (version1 debversion,
+       	  	  	   		  version2 debversion)
+  RETURNS boolean AS $$
+DECLARE
+  comp integer;
+  result boolean;
+BEGIN
+  comp := debversion_compare(version1, version2);
+  result := comp <> 0;
+  RETURN result;
+END;
+$$
+  LANGUAGE plpgsql
+  IMMUTABLE STRICT;
+COMMENT ON FUNCTION debversion_ne (debversion, debversion)
+  IS 'debversion not equal';
+
+CREATE OR REPLACE FUNCTION debversion_lt (version1 debversion,
+       	  	  	   		  version2 debversion)
+  RETURNS boolean AS $$
+DECLARE
+  comp integer;
+  result boolean;
+BEGIN
+  comp := debversion_compare(version1, version2);
+  result := comp < 0;
+  RETURN result;
+END;
+$$
+  LANGUAGE plpgsql
+  IMMUTABLE STRICT;
+COMMENT ON FUNCTION debversion_lt (debversion, debversion)
+  IS 'debversion less-than';
+
+CREATE OR REPLACE FUNCTION debversion_gt (version1 debversion,
+       	  	  	   		  version2 debversion) RETURNS boolean AS $$
+DECLARE
+  comp integer;
+  result boolean;
+BEGIN
+  comp := debversion_compare(version1, version2);
+  result := comp > 0;
+  RETURN result;
+END;
+$$
+  LANGUAGE plpgsql
+  IMMUTABLE STRICT;
+COMMENT ON FUNCTION debversion_gt (debversion, debversion)
+  IS 'debversion greater-than';
+
+CREATE OR REPLACE FUNCTION debversion_le (version1 debversion,
+       	  	  	   		  version2 debversion)
+  RETURNS boolean AS $$
+DECLARE
+  comp integer;
+  result boolean;
+BEGIN
+  comp := debversion_compare(version1, version2);
+  result := comp <= 0;
+  RETURN result;
+END;
+$$
+  LANGUAGE plpgsql
+  IMMUTABLE STRICT;
+COMMENT ON FUNCTION debversion_le (debversion, debversion)
+  IS 'debversion less-than-or-equal';
+
+CREATE OR REPLACE FUNCTION debversion_ge (version1 debversion,
+       	  	  	   		  version2 debversion)
+  RETURNS boolean AS $$
+DECLARE
+  comp integer;
+  result boolean;
+BEGIN
+  comp := debversion_compare(version1, version2);
+  result := comp >= 0;
+  RETURN result;
+END;
+$$
+  LANGUAGE plpgsql
+  IMMUTABLE STRICT;
+COMMENT ON FUNCTION debversion_ge (debversion, debversion)
+  IS 'debversion greater-than-or-equal';
+
+CREATE OPERATOR = (
+  PROCEDURE = debversion_eq,
+  LEFTARG = debversion,
+  RIGHTARG = debversion,
+  COMMUTATOR = =,
+  NEGATOR = !=
+);
+COMMENT ON OPERATOR = (debversion, debversion)
+  IS 'debversion equal';
+
+CREATE OPERATOR != (
+  PROCEDURE = debversion_eq,
+  LEFTARG = debversion,
+  RIGHTARG = debversion,
+  COMMUTATOR = !=,
+  NEGATOR = =
+);
+COMMENT ON OPERATOR != (debversion, debversion)
+  IS 'debversion not equal';
+
+CREATE OPERATOR < (
+  PROCEDURE = debversion_lt,
+  LEFTARG = debversion,
+  RIGHTARG = debversion,
+  COMMUTATOR = >,
+  NEGATOR = >=
+);
+COMMENT ON OPERATOR < (debversion, debversion)
+  IS 'debversion less-than';
+
+CREATE OPERATOR > (
+  PROCEDURE = debversion_gt,
+  LEFTARG = debversion,
+  RIGHTARG = debversion,
+  COMMUTATOR = <,
+  NEGATOR = >=
+);
+COMMENT ON OPERATOR > (debversion, debversion)
+  IS 'debversion greater-than';
+
+CREATE OPERATOR <= (
+  PROCEDURE = debversion_le,
+  LEFTARG = debversion,
+  RIGHTARG = debversion,
+  COMMUTATOR = >=,
+  NEGATOR = >
+);
+COMMENT ON OPERATOR <= (debversion, debversion)
+  IS 'debversion less-than-or-equal';
+
+CREATE OPERATOR >= (
+  PROCEDURE = debversion_ge,
+  LEFTARG = debversion,
+  RIGHTARG = debversion,
+  COMMUTATOR = <=,
+  NEGATOR = <
+);
+COMMENT ON OPERATOR >= (debversion, debversion)
+  IS 'debversion greater-than-or-equal';
-- 
1.5.6.5


Reply to: