Skip to content

Commit 74aae15

Browse files
committed
Add remote extension test with library component
The current test was just SQL files only, but we also want to test a remote extension which includes a loadable library. With both extensions we should cover a larger portion of compute_ctl's remote extension code paths. Signed-off-by: Tristan Partin <[email protected]>
1 parent eb6efda commit 74aae15

12 files changed

+420
-62
lines changed

test_runner/conftest.py

+1
Original file line numberDiff line numberDiff line change
@@ -16,4 +16,5 @@
1616
"fixtures.slow",
1717
"fixtures.reruns",
1818
"fixtures.fast_import",
19+
"fixtures.pg_config",
1920
)

test_runner/fixtures/pg_config.py

+249
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,249 @@
1+
from __future__ import annotations
2+
3+
import shlex
4+
from enum import StrEnum
5+
from pathlib import Path
6+
from typing import TYPE_CHECKING, cast, final
7+
8+
import pytest
9+
10+
from fixtures.neon_fixtures import PgBin
11+
12+
if TYPE_CHECKING:
13+
from collections.abc import Iterator
14+
from typing import IO
15+
16+
17+
@final
18+
class PgConfigKey(StrEnum):
19+
BINDIR = "BINDIR"
20+
DOCDIR = "DOCDIR"
21+
HTMLDIR = "HTMLDIR"
22+
INCLUDEDIR = "INCLUDEDIR"
23+
PKGINCLUDEDIR = "PKGINCLUDEDIR"
24+
INCLUDEDIR_SERVER = "INCLUDEDIR-SERVER"
25+
LIBDIR = "LIBDIR"
26+
PKGLIBDIR = "PKGLIBDIR"
27+
LOCALEDIR = "LOCALEDIR"
28+
MANDIR = "MANDIR"
29+
SHAREDIR = "SHAREDIR"
30+
SYSCONFDIR = "SYSCONFDIR"
31+
PGXS = "PGXS"
32+
CONFIGURE = "CONFIGURE"
33+
CC = "CC"
34+
CPPFLAGS = "CPPFLAGS"
35+
CFLAGS = "CFLAGS"
36+
CFLAGS_SL = "CFLAGS_SL"
37+
LDFLAGS = "LDFLAGS"
38+
LDFLAGS_EX = "LDFLAGS_EX"
39+
LDFLAGS_SL = "LDFLAGS_SL"
40+
LIBS = "LIBS"
41+
VERSION = "VERSION"
42+
43+
44+
if TYPE_CHECKING:
45+
# TODO: This could become a TypedDict if Python ever allows StrEnums to be
46+
# keys.
47+
PgConfig = dict[PgConfigKey, str | Path | list[str]]
48+
49+
50+
def __get_pg_config(pg_bin: PgBin) -> PgConfig:
51+
"""Get pg_config values by invoking the command"""
52+
53+
cmd = pg_bin.run_nonblocking(["pg_config"])
54+
cmd.wait()
55+
if cmd.returncode != 0:
56+
pytest.exit("")
57+
assert cmd.stdout
58+
59+
stdout = cast("IO[str]", cmd.stdout)
60+
61+
# Parse the output into a dictionary
62+
values: PgConfig = {}
63+
for line in stdout.readlines():
64+
if "=" in line:
65+
key, value = line.split("=", 1)
66+
value = value.strip()
67+
match PgConfigKey(key.strip()):
68+
case (
69+
(
70+
PgConfigKey.CC
71+
| PgConfigKey.CPPFLAGS
72+
| PgConfigKey.CFLAGS
73+
| PgConfigKey.CFLAGS_SL
74+
| PgConfigKey.LDFLAGS
75+
| PgConfigKey.LDFLAGS_EX
76+
| PgConfigKey.LDFLAGS_SL
77+
| PgConfigKey.LIBS
78+
) as k
79+
):
80+
values[k] = shlex.split(value)
81+
case (
82+
(
83+
PgConfigKey.BINDIR
84+
| PgConfigKey.DOCDIR
85+
| PgConfigKey.HTMLDIR
86+
| PgConfigKey.INCLUDEDIR
87+
| PgConfigKey.PKGINCLUDEDIR
88+
| PgConfigKey.INCLUDEDIR_SERVER
89+
| PgConfigKey.LIBDIR
90+
| PgConfigKey.PKGLIBDIR
91+
| PgConfigKey.LOCALEDIR
92+
| PgConfigKey.MANDIR
93+
| PgConfigKey.SHAREDIR
94+
| PgConfigKey.SYSCONFDIR
95+
| PgConfigKey.PGXS
96+
) as k
97+
):
98+
values[k] = Path(value)
99+
case _ as k:
100+
values[k] = value
101+
102+
return values
103+
104+
105+
@pytest.fixture(scope="function")
106+
def pg_config(pg_bin: PgBin) -> Iterator[PgConfig]:
107+
"""Dictionary of all pg_config values from the system"""
108+
109+
yield __get_pg_config(pg_bin)
110+
111+
112+
@pytest.fixture(scope="function")
113+
def pg_config_bindir(pg_config: PgConfig) -> Iterator[Path]:
114+
"""BINDIR value from pg_config"""
115+
yield cast("Path", pg_config[PgConfigKey.BINDIR])
116+
117+
118+
@pytest.fixture(scope="function")
119+
def pg_config_docdir(pg_config: PgConfig) -> Iterator[Path]:
120+
"""DOCDIR value from pg_config"""
121+
yield cast("Path", pg_config[PgConfigKey.DOCDIR])
122+
123+
124+
@pytest.fixture(scope="function")
125+
def pg_config_htmldir(pg_config: PgConfig) -> Iterator[Path]:
126+
"""HTMLDIR value from pg_config"""
127+
yield cast("Path", pg_config[PgConfigKey.HTMLDIR])
128+
129+
130+
@pytest.fixture(scope="function")
131+
def pg_config_includedir(
132+
pg_config: dict[PgConfigKey, str | Path | list[str]],
133+
) -> Iterator[Path]:
134+
"""INCLUDEDIR value from pg_config"""
135+
yield cast("Path", pg_config[PgConfigKey.INCLUDEDIR])
136+
137+
138+
@pytest.fixture(scope="function")
139+
def pg_config_pkgincludedir(pg_config: PgConfig) -> Iterator[Path]:
140+
"""PKGINCLUDEDIR value from pg_config"""
141+
yield cast("Path", pg_config[PgConfigKey.PKGINCLUDEDIR])
142+
143+
144+
@pytest.fixture(scope="function")
145+
def pg_config_includedir_server(pg_config: PgConfig) -> Iterator[Path]:
146+
"""INCLUDEDIR-SERVER value from pg_config"""
147+
yield cast("Path", pg_config[PgConfigKey.INCLUDEDIR_SERVER])
148+
149+
150+
@pytest.fixture(scope="function")
151+
def pg_config_libdir(pg_config: PgConfig) -> Iterator[Path]:
152+
"""LIBDIR value from pg_config"""
153+
yield cast("Path", pg_config[PgConfigKey.LIBDIR])
154+
155+
156+
@pytest.fixture(scope="function")
157+
def pg_config_pkglibdir(pg_config: PgConfig) -> Iterator[Path]:
158+
"""PKGLIBDIR value from pg_config"""
159+
yield cast("Path", pg_config[PgConfigKey.PKGLIBDIR])
160+
161+
162+
@pytest.fixture(scope="function")
163+
def pg_config_localedir(pg_config: PgConfig) -> Iterator[Path]:
164+
"""LOCALEDIR value from pg_config"""
165+
yield cast("Path", pg_config[PgConfigKey.LOCALEDIR])
166+
167+
168+
@pytest.fixture(scope="function")
169+
def pg_config_mandir(pg_config: PgConfig) -> Iterator[Path]:
170+
"""MANDIR value from pg_config"""
171+
yield cast("Path", pg_config[PgConfigKey.MANDIR])
172+
173+
174+
@pytest.fixture(scope="function")
175+
def pg_config_sharedir(pg_config: PgConfig) -> Iterator[Path]:
176+
"""SHAREDIR value from pg_config"""
177+
yield cast("Path", pg_config[PgConfigKey.SHAREDIR])
178+
179+
180+
@pytest.fixture(scope="function")
181+
def pg_config_sysconfdir(pg_config: PgConfig) -> Iterator[Path]:
182+
"""SYSCONFDIR value from pg_config"""
183+
yield cast("Path", pg_config[PgConfigKey.SYSCONFDIR])
184+
185+
186+
@pytest.fixture(scope="function")
187+
def pg_config_pgxs(pg_config: PgConfig) -> Iterator[Path]:
188+
"""PGXS value from pg_config"""
189+
yield cast("Path", pg_config[PgConfigKey.PGXS])
190+
191+
192+
@pytest.fixture(scope="function")
193+
def pg_config_configure(pg_config: PgConfig) -> Iterator[str]:
194+
"""CONFIGURE value from pg_config"""
195+
yield cast("str", pg_config[PgConfigKey.CONFIGURE])
196+
197+
198+
@pytest.fixture(scope="function")
199+
def pg_config_cc(pg_config: PgConfig) -> Iterator[list[str]]:
200+
"""CC value from pg_config"""
201+
yield cast("list[str]", pg_config[PgConfigKey.CC])
202+
203+
204+
@pytest.fixture(scope="function")
205+
def pg_config_cppflags(pg_config: PgConfig) -> Iterator[list[str]]:
206+
"""CPPFLAGS value from pg_config"""
207+
yield cast("list[str]", pg_config[PgConfigKey.CPPFLAGS])
208+
209+
210+
@pytest.fixture(scope="function")
211+
def pg_config_cflags(pg_config: PgConfig) -> Iterator[list[str]]:
212+
"""CFLAGS value from pg_config"""
213+
yield cast("list[str]", pg_config[PgConfigKey.CFLAGS])
214+
215+
216+
@pytest.fixture(scope="function")
217+
def pg_config_cflags_sl(pg_config: PgConfig) -> Iterator[list[str]]:
218+
"""CFLAGS_SL value from pg_config"""
219+
yield cast("list[str]", pg_config[PgConfigKey.CFLAGS_SL])
220+
221+
222+
@pytest.fixture(scope="function")
223+
def pg_config_ldflags(pg_config: PgConfig) -> Iterator[list[str]]:
224+
"""LDFLAGS value from pg_config"""
225+
yield cast("list[str]", pg_config[PgConfigKey.LDFLAGS])
226+
227+
228+
@pytest.fixture(scope="function")
229+
def pg_config_ldflags_ex(pg_config: PgConfig) -> Iterator[list[str]]:
230+
"""LDFLAGS_EX value from pg_config"""
231+
yield cast("list[str]", pg_config[PgConfigKey.LDFLAGS_EX])
232+
233+
234+
@pytest.fixture(scope="function")
235+
def pg_config_ldflags_sl(pg_config: PgConfig) -> Iterator[list[str]]:
236+
"""LDFLAGS_SL value from pg_config"""
237+
yield cast("list[str]", pg_config[PgConfigKey.LDFLAGS_SL])
238+
239+
240+
@pytest.fixture(scope="function")
241+
def pg_config_libs(pg_config: PgConfig) -> Iterator[list[str]]:
242+
"""LIBS value from pg_config"""
243+
yield cast("list[str]", pg_config[PgConfigKey.LIBS])
244+
245+
246+
@pytest.fixture(scope="function")
247+
def pg_config_version(pg_config: PgConfig) -> Iterator[str]:
248+
"""VERSION value from pg_config"""
249+
yield cast("str", pg_config[PgConfigKey.VERSION])

test_runner/regress/data/test_remote_extensions/test_extension/sql/test_extension--1.0.sql

-12
This file was deleted.

test_runner/regress/data/test_remote_extensions/test_extension/test_extension.control

-1
This file was deleted.

test_runner/regress/data/test_remote_extensions/test_extension/sql/test_extension--1.0--1.1.sql test_runner/regress/data/test_remote_extensions/test_extension_sql_only/sql/test_extension_sql_only--1.0--1.1.sql

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
1-
\echo Use "ALTER EXTENSION test_extension UPDATE TO '1.1'" to load this file. \quit
1+
\echo Use "ALTER EXTENSION test_extension_sql_only UPDATE TO '1.1'" to load this file. \quit
22

3-
CREATE FUNCTION test_extension.fun_fact()
3+
CREATE FUNCTION test_extension_sql_only.fun_fact()
44
RETURNS void
55
IMMUTABLE LEAKPROOF PARALLEL SAFE
66
AS $$
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
\echo Use "CREATE EXTENSION test_extension_sql_only" to load this file. \quit
2+
3+
CREATE SCHEMA test_extension_sql_only;
4+
5+
CREATE FUNCTION test_extension_sql_only.motd()
6+
RETURNS void
7+
IMMUTABLE LEAKPROOF PARALLEL SAFE
8+
AS $$
9+
BEGIN
10+
RAISE NOTICE 'Have a great day';
11+
END;
12+
$$ LANGUAGE 'plpgsql';
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
comment = 'Test extension SQL only'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
\echo Use "ALTER EXTENSION test_extension_with_lib UPDATE TO '1.1'" to load this file. \quit
2+
3+
CREATE FUNCTION test_extension_with_lib.fun_fact()
4+
RETURNS void
5+
IMMUTABLE LEAKPROOF PARALLEL SAFE
6+
AS 'MODULE_PATHNAME', 'fun_fact' LANGUAGE C;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
\echo Use "CREATE EXTENSION test_extension_with_lib" to load this file. \quit
2+
3+
CREATE SCHEMA test_extension_with_lib;
4+
5+
CREATE FUNCTION test_extension_with_lib.motd()
6+
RETURNS void
7+
IMMUTABLE LEAKPROOF PARALLEL SAFE
8+
AS 'MODULE_PATHNAME', 'motd' LANGUAGE C;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
#include <postgres.h>
2+
3+
#include <fmgr.h>
4+
5+
PG_MODULE_MAGIC;
6+
7+
PG_FUNCTION_INFO_V1(motd);
8+
PG_FUNCTION_INFO_V1(fun_fact);
9+
10+
void
11+
_PG_init(void)
12+
{
13+
}
14+
15+
Datum
16+
motd(PG_FUNCTION_ARGS)
17+
{
18+
elog(NOTICE, "Have a great day");
19+
20+
PG_RETURN_VOID();
21+
}
22+
23+
Datum
24+
fun_fact(PG_FUNCTION_ARGS)
25+
{
26+
elog(NOTICE, "Neon has a melting point of -246.08 C");
27+
28+
PG_RETURN_VOID();
29+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
comment = 'Test extension with lib'
2+
module_pathname = '$libdir/test_extension_with_lib'

0 commit comments

Comments
 (0)