postgresql/src/interfaces/libpq/t/001_uri.pl

245 lines
6.4 KiB
Perl

# Copyright (c) 2021-2022, PostgreSQL Global Development Group
use strict;
use warnings;
use PostgreSQL::Test::Utils;
use Test::More;
use IPC::Run;
# List of URIs tests. For each test the first element is the input string, the
# second the expected stdout and the third the expected stderr.
my @tests = (
[
q{postgresql://uri-user:secret@host:12345/db},
q{user='uri-user' password='secret' dbname='db' host='host' port='12345' (inet)},
q{},
],
[
q{postgresql://uri-user@host:12345/db},
q{user='uri-user' dbname='db' host='host' port='12345' (inet)}, q{},
],
[
q{postgresql://uri-user@host/db},
q{user='uri-user' dbname='db' host='host' (inet)}, q{},
],
[
q{postgresql://host:12345/db},
q{dbname='db' host='host' port='12345' (inet)}, q{},
],
[ q{postgresql://host/db}, q{dbname='db' host='host' (inet)}, q{}, ],
[
q{postgresql://uri-user@host:12345/},
q{user='uri-user' host='host' port='12345' (inet)},
q{},
],
[
q{postgresql://uri-user@host/},
q{user='uri-user' host='host' (inet)},
q{},
],
[ q{postgresql://uri-user@}, q{user='uri-user' (local)}, q{}, ],
[ q{postgresql://host:12345/}, q{host='host' port='12345' (inet)}, q{}, ],
[ q{postgresql://host:12345}, q{host='host' port='12345' (inet)}, q{}, ],
[ q{postgresql://host/db}, q{dbname='db' host='host' (inet)}, q{}, ],
[ q{postgresql://host/}, q{host='host' (inet)}, q{}, ],
[ q{postgresql://host}, q{host='host' (inet)}, q{}, ],
[ q{postgresql://}, q{(local)}, q{}, ],
[
q{postgresql://?hostaddr=127.0.0.1}, q{hostaddr='127.0.0.1' (inet)},
q{},
],
[
q{postgresql://example.com?hostaddr=63.1.2.4},
q{host='example.com' hostaddr='63.1.2.4' (inet)},
q{},
],
[ q{postgresql://%68ost/}, q{host='host' (inet)}, q{}, ],
[
q{postgresql://host/db?user=uri-user},
q{user='uri-user' dbname='db' host='host' (inet)},
q{},
],
[
q{postgresql://host/db?user=uri-user&port=12345},
q{user='uri-user' dbname='db' host='host' port='12345' (inet)},
q{},
],
[
q{postgresql://host/db?u%73er=someotheruser&port=12345},
q{user='someotheruser' dbname='db' host='host' port='12345' (inet)},
q{},
],
[
q{postgresql://host/db?u%7aer=someotheruser&port=12345}, q{},
q{uri-regress: invalid URI query parameter: "uzer"},
],
[
q{postgresql://host:12345?user=uri-user},
q{user='uri-user' host='host' port='12345' (inet)},
q{},
],
[
q{postgresql://host?user=uri-user},
q{user='uri-user' host='host' (inet)},
q{},
],
[ q{postgresql://host?}, q{host='host' (inet)}, q{}, ],
[
q{postgresql://[::1]:12345/db},
q{dbname='db' host='::1' port='12345' (inet)},
q{},
],
[ q{postgresql://[::1]/db}, q{dbname='db' host='::1' (inet)}, q{}, ],
[
q{postgresql://[2001:db8::1234]/}, q{host='2001:db8::1234' (inet)},
q{},
],
[
q{postgresql://[200z:db8::1234]/}, q{host='200z:db8::1234' (inet)},
q{},
],
[ q{postgresql://[::1]}, q{host='::1' (inet)}, q{}, ],
[ q{postgres://}, q{(local)}, q{}, ],
[ q{postgres:///}, q{(local)}, q{}, ],
[ q{postgres:///db}, q{dbname='db' (local)}, q{}, ],
[
q{postgres://uri-user@/db}, q{user='uri-user' dbname='db' (local)},
q{},
],
[
q{postgres://?host=/path/to/socket/dir},
q{host='/path/to/socket/dir' (local)},
q{},
],
[
q{postgresql://host?uzer=}, q{},
q{uri-regress: invalid URI query parameter: "uzer"},
],
[
q{postgre://},
q{},
q{uri-regress: missing "=" after "postgre://" in connection info string},
],
[
q{postgres://[::1},
q{},
q{uri-regress: end of string reached when looking for matching "]" in IPv6 host address in URI: "postgres://[::1"},
],
[
q{postgres://[]},
q{},
q{uri-regress: IPv6 host address may not be empty in URI: "postgres://[]"},
],
[
q{postgres://[::1]z},
q{},
q{uri-regress: unexpected character "z" at position 17 in URI (expected ":" or "/"): "postgres://[::1]z"},
],
[
q{postgresql://host?zzz},
q{},
q{uri-regress: missing key/value separator "=" in URI query parameter: "zzz"},
],
[
q{postgresql://host?value1&value2},
q{},
q{uri-regress: missing key/value separator "=" in URI query parameter: "value1"},
],
[
q{postgresql://host?key=key=value},
q{},
q{uri-regress: extra key/value separator "=" in URI query parameter: "key"},
],
[
q{postgres://host?dbname=%XXfoo}, q{},
q{uri-regress: invalid percent-encoded token: "%XXfoo"},
],
[
q{postgresql://a%00b},
q{},
q{uri-regress: forbidden value %00 in percent-encoded value: "a%00b"},
],
[
q{postgresql://%zz}, q{},
q{uri-regress: invalid percent-encoded token: "%zz"},
],
[
q{postgresql://%1}, q{},
q{uri-regress: invalid percent-encoded token: "%1"},
],
[
q{postgresql://%}, q{},
q{uri-regress: invalid percent-encoded token: "%"},
],
[ q{postgres://@host}, q{host='host' (inet)}, q{}, ],
[ q{postgres://host:/}, q{host='host' (inet)}, q{}, ],
[ q{postgres://:12345/}, q{port='12345' (local)}, q{}, ],
[
q{postgres://otheruser@?host=/no/such/directory},
q{user='otheruser' host='/no/such/directory' (local)},
q{},
],
[
q{postgres://otheruser@/?host=/no/such/directory},
q{user='otheruser' host='/no/such/directory' (local)},
q{},
],
[
q{postgres://otheruser@:12345?host=/no/such/socket/path},
q{user='otheruser' host='/no/such/socket/path' port='12345' (local)},
q{},
],
[
q{postgres://otheruser@:12345/db?host=/path/to/socket},
q{user='otheruser' dbname='db' host='/path/to/socket' port='12345' (local)},
q{},
],
[
q{postgres://:12345/db?host=/path/to/socket},
q{dbname='db' host='/path/to/socket' port='12345' (local)},
q{},
],
[
q{postgres://:12345?host=/path/to/socket},
q{host='/path/to/socket' port='12345' (local)},
q{},
],
[
q{postgres://%2Fvar%2Flib%2Fpostgresql/dbname},
q{dbname='dbname' host='/var/lib/postgresql' (local)},
q{},
]);
# test to run for each of the above test definitions
sub test_uri
{
local $Test::Builder::Level = $Test::Builder::Level + 1;
my $uri;
my %expect;
my %result;
($uri, $expect{stdout}, $expect{stderr}) = @$_;
$expect{'exit'} = $expect{stderr} eq '';
my $cmd = [ 'uri-regress', $uri ];
$result{exit} = IPC::Run::run $cmd, '>', \$result{stdout}, '2>',
\$result{stderr};
chomp($result{stdout});
chomp($result{stderr});
# use is_deeply so there's one test result for each test above, without
# loosing the information whether stdout/stderr mismatched.
is_deeply(\%result, \%expect, $uri);
}
foreach (@tests)
{
test_uri($_);
}
done_testing();