-
Notifications
You must be signed in to change notification settings - Fork 48
/
Copy pathDockerfile
587 lines (503 loc) · 26.4 KB
/
Dockerfile
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
## The purpose of this Dockerfile is to build an image that contains:
## - timescale from (internal) sources
## - many PostgreSQL extensions
## - patroni for High Availability
## - Barman Cloud for CloudNativePG compatibility
## - spilo to allow the github.com/zalando/postgres-operator to be compatible
## - pgBackRest to allow good backups
## We have many base images to choose from, (alpine, bitnami) but as we're adding a lot
## of tools to the image anyway, the end result is that we would only
## reduce the final Docker image by single digit MB's, which is insignificant
## in relation to the total image size.
## By choosing a very basic base image, we do keep full control over every part
## of the build steps. This Dockerfile contains every piece of magic we want.
## To allow us to use specific glibc 2.33+ features, we need to find a way
## to run glibc 2.33. Running multiple glibc versions inside the same
## container is something we'd like to avoid, we've seen multiple glibc
## related bugs in our lifetime, adding multiple glibc versions in the mix
## would make debugging harder.
## Debian (and rust:debian) has served us well in the past, however even Debian's
## latest release (bullseye, August 2021) cannot give us glibc 2.33.
## Ubuntu however does give us glibc 2.33 - as Ubuntu is based upon Debian
## the changes required are not that big for this Docker Image. Most of the
## tools we use will be the same across the board, as most of our tools our
## installed using external repositories.
ARG DOCKER_FROM=ubuntu:22.04
FROM ${DOCKER_FROM} AS builder
# By including multiple versions of PostgreSQL we can use the same Docker image,
# regardless of the major PostgreSQL Version. It also allow us to support (eventually)
# pg_upgrade from one major version to another,
# so we need all the postgres & timescale libraries for all versions
ARG PG_VERSIONS="17 16 15 14 13"
ARG PG_MAJOR=17
ENV DEBIAN_FRONTEND=noninteractive
# We need full control over the running user, including the UID, therefore we
# create the postgres user as the first thing on our list
RUN adduser --home /home/postgres --uid 1000 --disabled-password --gecos "" postgres
RUN echo 'APT::Install-Recommends "false";' >> /etc/apt/apt.conf.d/01norecommend
RUN echo 'APT::Install-Suggests "false";' >> /etc/apt/apt.conf.d/01norecommend
# Ubuntu will throttle downloads which can slow things down so much that we can't complete. Since we're
# building in AWS, use their mirrors. arm64 and amd64 use different sources though
COPY sources /tmp/sources
RUN set -eux; \
source="/tmp/sources/sources.list.$(dpkg --print-architecture)"; \
mv /etc/apt/sources.list /etc/apt/sources.list.dist; \
cp "$source" /etc/apt/sources.list; \
rm -fr /tmp/sources
# Make sure we're as up-to-date as possible, and install the highlest level dependencies
RUN set -eux; \
apt-get update; \
apt-get upgrade -y; \
apt-get install -y ca-certificates curl gnupg1 gpg gpg-agent locales lsb-release wget unzip
RUN mkdir -p /build/scripts
RUN chmod 777 /build
WORKDIR /build/
RUN curl -Ls https://www.postgresql.org/media/keys/ACCC4CF8.asc | gpg --dearmor --output /usr/share/keyrings/postgresql.keyring
RUN for t in deb deb-src; do \
echo "$t [arch=$(dpkg --print-architecture) signed-by=/usr/share/keyrings/postgresql.keyring] http://apt.postgresql.org/pub/repos/apt/ $(lsb_release -s -c)-pgdg main" >> /etc/apt/sources.list.d/pgdg.list; \
done
# timescaledb-tune, as well as timescaledb-parallel-copy
RUN curl -Ls https://packagecloud.io/timescale/timescaledb/gpgkey | gpg --dearmor --output /usr/share/keyrings/timescaledb.keyring
RUN echo "deb [arch=$(dpkg --print-architecture) signed-by=/usr/share/keyrings/timescaledb.keyring] https://packagecloud.io/timescale/timescaledb/ubuntu/ $(lsb_release -cs) main" > /etc/apt/sources.list.d/timescaledb.list
# The following tools are required for some of the processes we (TimescaleDB) regularly
# run inside the containers that use this Docker Image
# awscli is useful in many situations, for example, to list backup buckets etc
RUN set -eux; \
apt-get update; \
apt-get upgrade -y; \
apt-get install -y \
less jq strace procps awscli vim-tiny gdb gdbserver dumb-init daemontools \
postgresql-common pgbouncer pgbackrest lz4 libpq-dev libpq5 pgtop libnss-wrapper gosu \
pg-activity lsof htop; \
curl -Lso /usr/local/bin/yq https://github.com/mikefarah/yq/releases/latest/download/yq_linux_"$(dpkg --print-architecture)"; \
chmod 755 /usr/local/bin/yq
# pgbackrest-exporter
ARG PGBACKREST_EXPORTER_VERSION="0.18.0"
RUN set -eux; \
arch="$(arch)"; [ "$arch" = aarch64 ] && arch=arm64; pkg="pgbackrest_exporter_${PGBACKREST_EXPORTER_VERSION}_linux_${arch}"; \
curl --silent \
--location \
--output /tmp/pkg.deb \
"https://github.com/woblerr/pgbackrest_exporter/releases/download/v${PGBACKREST_EXPORTER_VERSION}/${pkg}.deb"; \
cd /tmp; \
dpkg -i ./pkg.deb; \
rm -rfv /tmp/pkg.deb
# pgbouncer-exporter
ARG PGBOUNCER_EXPORTER_VERSION="0.9.0"
RUN set -eux; \
pkg="pgbouncer_exporter-${PGBOUNCER_EXPORTER_VERSION}.linux-$(dpkg --print-architecture)"; \
curl --silent \
--location \
--output /tmp/pkg.tgz \
"https://github.com/prometheus-community/pgbouncer_exporter/releases/download/v${PGBOUNCER_EXPORTER_VERSION}/${pkg}.tar.gz"; \
cd /tmp; \
tar xvzf /tmp/pkg.tgz "$pkg"/pgbouncer_exporter; \
mv -v /tmp/"$pkg"/pgbouncer_exporter /usr/local/bin/pgbouncer_exporter; \
rm -rfv /tmp/pkg.tgz /tmp/"$pkg"
# forbid creation of a main cluster when package is installed
RUN sed -ri 's/#(create_main_cluster) .*$/\1 = false/' /etc/postgresql-common/createcluster.conf
# The next 2 instructions (ENV + RUN) are directly copied from https://github.com/rust-lang/docker-rust/blob/master/stable/bullseye/Dockerfile
ENV RUSTUP_HOME=/usr/local/rustup \
CARGO_HOME=/usr/local/cargo \
PATH=/usr/local/cargo/bin:$PATH \
RUST_VERSION=1.82.0
RUN set -eux; \
dpkgArch="$(dpkg --print-architecture)"; \
case "${dpkgArch##*-}" in \
amd64) rustArch='x86_64-unknown-linux-gnu'; rustupSha256='6aeece6993e902708983b209d04c0d1dbb14ebb405ddb87def578d41f920f56d' ;; \
armhf) rustArch='armv7-unknown-linux-gnueabihf'; rustupSha256='3c4114923305f1cd3b96ce3454e9e549ad4aa7c07c03aec73d1a785e98388bed' ;; \
arm64) rustArch='aarch64-unknown-linux-gnu'; rustupSha256='1cffbf51e63e634c746f741de50649bbbcbd9dbe1de363c9ecef64e278dba2b2' ;; \
i386) rustArch='i686-unknown-linux-gnu'; rustupSha256='0a6bed6e9f21192a51f83977716466895706059afb880500ff1d0e751ada5237' ;; \
*) echo >&2 "unsupported architecture: ${dpkgArch}"; exit 1 ;; \
esac; \
url="https://static.rust-lang.org/rustup/archive/1.27.1/${rustArch}/rustup-init"; \
wget "$url"; \
echo "${rustupSha256} *rustup-init" | sha256sum -c -; \
chmod +x rustup-init; \
./rustup-init -y --no-modify-path --profile minimal --default-toolchain $RUST_VERSION --default-host ${rustArch}; \
rm rustup-init; \
chmod -R a+w $RUSTUP_HOME $CARGO_HOME; \
rustup --version; \
cargo --version; \
rustc --version
# Setup locales, and make sure we have a en_US.UTF-8 locale available
RUN set -eux; \
find /usr/share/i18n/charmaps/ -type f ! -name UTF-8.gz -delete; \
find /usr/share/i18n/locales/ -type f ! -name en_US ! -name en_GB ! -name i18n* ! -name iso14651_t1 ! -name iso14651_t1_common ! -name 'translit_*' -delete; \
echo 'en_US.UTF-8 UTF-8' > /usr/share/i18n/SUPPORTED; \
localedef -i en_US -c -f UTF-8 -A /usr/share/locale/locale.alias en_US.UTF-8
# We install pip3, as we need it for some of the extensions. This will install a lot of dependencies, all marked as auto to help with cleanup later
RUN apt-get install -y python3 python3-pip
# pgai requires pip 23.0.1 or greater due to the use of --break-system-packages flag
RUN set -ex; \
if [ "$(pip --version | awk '{print $2; exit}')" \< "23.0.1" ]; then \
python3 -m pip install --upgrade pip==23.0.1; \
fi;
# We install some build dependencies and mark the installed packages as auto-installed,
# this will cause the cleanup to get rid of all of these packages
ENV BUILD_PACKAGES="binutils cmake devscripts equivs gcc git gpg gpg-agent libc-dev libc6-dev libkrb5-dev libperl-dev libssl-dev lsb-release make patchutils python2-dev python3-dev wget libsodium-dev"
RUN apt-get install -y ${BUILD_PACKAGES}
RUN apt-mark auto ${BUILD_PACKAGES}
# https://salsa.debian.org/postgresql/postgresql/-/commit/b995beb3cd1c2b8834605007227b3cedab6462e4
# This looks like a build-/test-only dependency, and they expect us to use the real tzdata when actually running.
# TODO: keep watching this to see if they remove the limitation. If the old tzdata becomes unavailable, we'll have to
# do something more drastic.
RUN apt-get install -y --allow-downgrades tzdata="2022a-*"
# We install the PostgreSQL build dependencies and mark the installed packages as auto-installed,
RUN set -eux; \
for pg in ${PG_VERSIONS}; do \
mk-build-deps postgresql-${pg} && apt-get install -y ./postgresql-${pg}-build-deps*.deb && apt-mark auto postgresql-${pg}-build-deps || exit 1; \
done
# TODO: There's currently a build-dependency problem related to tzdata, remove this when it's resolved
RUN apt-get install -y tzdata
RUN set -eux; \
packages=""; \
for pg in ${PG_VERSIONS}; do \
packages="$packages postgresql-${pg} postgresql-server-dev-${pg} postgresql-${pg}-dbgsym \
postgresql-plpython3-${pg} postgresql-plperl-${pg} postgresql-${pg}-pgextwlist postgresql-${pg}-hll \
postgresql-${pg}-pgrouting postgresql-${pg}-repack postgresql-${pg}-hypopg postgresql-${pg}-unit \
postgresql-${pg}-pg-stat-kcache postgresql-${pg}-cron postgresql-${pg}-pldebugger postgresql-${pg}-pgpcre \
postgresql-${pg}-pglogical postgresql-${pg}-wal2json postgresql-${pg}-pgq3 postgresql-${pg}-pg-qualstats \
postgresql-${pg}-pgaudit postgresql-${pg}-ip4r postgresql-${pg}-pgtap postgresql-${pg}-orafce \
postgresql-${pg}-pgvector postgresql-${pg}-h3"; \
done; \
apt-get install -y $packages
ARG POSTGIS_VERSIONS="3"
RUN set -ex; \
if [ -n "${POSTGIS_VERSIONS}" ]; then \
for postgisv in ${POSTGIS_VERSIONS}; do \
for pg in ${PG_VERSIONS}; do \
apt-get install -y postgresql-${pg}-postgis-${postgisv}; \
done; \
done; \
fi
# pgai is an extension for artificial intelligence workloads
ARG PGAI_VERSION
RUN set -ex; \
if [ "${PG_MAJOR}" -gt 15 ] && [ -n "${PGAI_VERSION}" ]; then \
git clone --branch "${PGAI_VERSION}" https://github.com/timescale/pgai.git /build/pgai; \
cd /build/pgai; \
for pg in ${PG_VERSIONS}; do \
if [ "$pg" -gt 15 ]; then \
PG_BIN=$(/usr/lib/postgresql/${pg}/bin/pg_config --bindir) PG_MAJOR=${pg} ./projects/extension/build.py install; \
fi; \
done; \
fi
# Add a couple 3rd party extension managers to make extension additions easier
RUN set -eux; \
apt-get install -y pgxnclient
## Add pgsodium extension depedencies
RUN set -eux; \
apt-get install -y libsodium23
RUN set -eux; \
for pg in ${PG_VERSIONS}; do \
for pkg in pg_uuidv7 pgsodium; do \
PATH="/usr/lib/postgresql/${pg}/bin:$PATH" pgxnclient install --pg_config "/usr/lib/postgresql/${pg}/bin/pg_config" "$pkg"; \
done; \
done
# the strip command is due to the vectors.so size: 450mb before stripping, 12mb after
ARG PGVECTO_RS
RUN set -ex; \
if [ -n "${PGVECTO_RS}" ]; then \
for pg in ${PG_VERSIONS}; do \
# Vecto.rs only support PostgreSQL 14+
if [ $pg -ge 14 ]; then \
curl --silent \
--location \
--output /tmp/vectors.deb \
"https://github.com/tensorchord/pgvecto.rs/releases/download/v${PGVECTO_RS}/vectors-pg${pg}_${PGVECTO_RS}_$(dpkg --print-architecture).deb" && \
dpkg -i /tmp/vectors.deb && \
rm -rfv /tmp/vectors.deb && \
strip --strip-unneeded "/usr/lib/postgresql/${pg}/lib/vectors.so"; \
fi \
done; \
fi
COPY --chown=postgres:postgres build_scripts /build/scripts/
# Some Patroni prerequisites
# This need to be done after the PostgreSQL packages have been installed,
# to ensure we have the preferred libpq installations etc.
RUN apt-get install -y python3-etcd python3-requests python3-pystache python3-kubernetes python3-pysyncobj patroni
# Barman cloud
# Required for CloudNativePG compatibility
RUN pip3 install --no-cache-dir 'barman[cloud,azure,snappy,google]'
RUN apt-get install -y timescaledb-tools
## Entrypoints as they are from the Timescale image and its default upstream repositories.
## This ensures the default interface (entrypoint) equals the one of the github.com/timescale/timescaledb-docker one,
## which allows this Docker Image to be a drop-in replacement for those Docker Images.
ARG GITHUB_TIMESCALEDB_DOCKER_REF=main
ARG GITHUB_DOCKERLIB_POSTGRES_REF=master
RUN set -ex; \
cd /build; \
git clone https://github.com/timescale/timescaledb-docker; \
cd timescaledb-docker; \
git checkout ${GITHUB_TIMESCALEDB_DOCKER_REF}; \
cp -a docker-entrypoint-initdb.d /docker-entrypoint-initdb.d/; \
ln -s /usr/bin/timescaledb-tune /usr/local/bin/timescaledb-tune
# Add custom entrypoint to install timescaledb_toolkit
COPY scripts/010_install_timescaledb_toolkit.sh /docker-entrypoint-initdb.d/
COPY docker-entrypoint.sh /usr/local/bin/docker-entrypoint.sh
RUN chmod +x /usr/local/bin/docker-entrypoint.sh; \
ln -s /usr/local/bin/docker-entrypoint.sh /docker-entrypoint.sh
# The following allows *new* files to be created, so that extensions can be added to a running container.
# Existing files are still owned by root and have their sticky bit (the 1 in the 1775 permission mode) set,
# and therefore cannot be overwritten or removed by the unprivileged (postgres) user.
# This ensures the following:
# - libraries and supporting files that have been installed *before this step* are immutable
# - libraries and supporting files that have been installed *after this step* are mutable
# - files owned by postgres can be overwritten in a running container
# - new files can be added to the directories mentioned here
RUN set -ex; \
for pg in ${PG_VERSIONS}; do \
for dir in /usr/share/doc \
"$(/usr/lib/postgresql/${pg}/bin/pg_config --sharedir)/extension" \
"$(/usr/lib/postgresql/${pg}/bin/pg_config --pkglibdir)" \
"$(/usr/lib/postgresql/${pg}/bin/pg_config --bindir)" \
"$(/usr/lib/postgresql/${pg}/bin/pg_config --includedir-server)/extension"; do \
install --directory "${dir}" --group postgres --mode 1775; \
find "${dir}" -type d -exec install --directory {} --group postgres --mode 1775 \;; \
done; \
done
RUN for file in $(find /usr/share/postgresql -name 'postgresql.conf.sample'); do \
# We want timescaledb to be loaded in this image by every created cluster
sed -r -i "s/[#]*\s*(shared_preload_libraries)\s*=\s*'(.*)'/\1 = 'timescaledb,\2'/;s/,'/'/" $file \
# We need to listen on all interfaces, otherwise PostgreSQL is not accessible
&& echo "listen_addresses = '*'" >> $file; \
done
RUN chown -R postgres:postgres /usr/local/cargo
# required to install dbgsym packages
RUN set -ex; \
chgrp -R postgres /usr/lib/debug; \
chmod -R g+w /usr/lib/debug
USER postgres
ENV MAKEFLAGS=-j4
# pg_stat_monitor is a Query Performance Monitoring tool for PostgreSQL
# https://github.com/percona/pg_stat_monitor
ARG PG_STAT_MONITOR
RUN set -ex; \
if [ -n "${PG_STAT_MONITOR}" ]; then \
git clone https://github.com/percona/pg_stat_monitor /build/pg_stat_monitor; \
cd /build/pg_stat_monitor; \
git checkout "${PG_STAT_MONITOR}"; \
git reset HEAD --hard; \
for pg in ${PG_VERSIONS}; do \
PATH="/usr/lib/postgresql/${pg}/bin:${PATH}" make USE_PGXS=1 clean; \
PATH="/usr/lib/postgresql/${pg}/bin:${PATH}" make USE_PGXS=1 all; \
PATH="/usr/lib/postgresql/${pg}/bin:${PATH}" make USE_PGXS=1 install; \
done; \
fi
# pg_auth_mon is an extension to monitor authentication attempts
# It is also useful to determine whether the DB is actively used
# https://github.com/RafiaSabih/pg_auth_mon
ARG PG_AUTH_MON
RUN set -ex; \
if [ -n "${PG_AUTH_MON}" ]; then \
git clone https://github.com/RafiaSabih/pg_auth_mon /build/pg_auth_mon; \
cd /build/pg_auth_mon; \
git checkout "${PG_AUTH_MON}"; \
for pg in ${PG_VERSIONS}; do \
git reset HEAD --hard; \
PATH="/usr/lib/postgresql/${pg}/bin:${PATH}" make clean; \
PATH="/usr/lib/postgresql/${pg}/bin:${PATH}" make install; \
done; \
fi
# logerrors is an extension to count the number of errors logged by postgrs, grouped by the error codes
# https://github.com/munakoiso/logerrors
ARG PG_LOGERRORS
RUN set -ex; \
if [ -n "${PG_LOGERRORS}" ]; then \
git clone https://github.com/munakoiso/logerrors /build/logerrors; \
cd /build/logerrors; \
git checkout "${PG_LOGERRORS}"; \
for pg in ${PG_VERSIONS}; do \
git reset HEAD --hard; \
PATH="/usr/lib/postgresql/${pg}/bin:${PATH}" make clean; \
PATH="/usr/lib/postgresql/${pg}/bin:${PATH}" make install; \
done; \
fi
# INSTALL_METHOD will show up in the telemetry, which makes it easier to identify these installations
ARG INSTALL_METHOD=docker-ha
ARG OSS_ONLY
# RUST_RELEASE for some packages passes this to --profile
ARG RUST_RELEASE=release
# split the extension builds into two steps to allow caching of successful steps
ARG GITHUB_REPO=timescale/timescaledb
ARG TIMESCALEDB_VERSIONS
RUN set -ex; \
OSS_ONLY="${OSS_ONLY}" \
GITHUB_REPO="${GITHUB_REPO}" \
TIMESCALEDB_VERSIONS="${TIMESCALEDB_VERSIONS}" \
/build/scripts/install_extensions timescaledb
# install all rust packages in the same step to allow it to optimize for cargo-pgx installs
ARG TOOLKIT_VERSIONS
RUN set -ex; \
OSS_ONLY="${OSS_ONLY}" \
RUST_RELEASE="${RUST_RELEASE}" \
TOOLKIT_VERSIONS="${TOOLKIT_VERSIONS}" \
/build/scripts/install_extensions rust
ARG PGVECTORSCALE_VERSIONS
RUN set -ex; \
OSS_ONLY="${OSS_ONLY}" \
RUST_RELEASE="${RUST_RELEASE}" \
PGVECTORSCALE_VERSIONS="${PGVECTORSCALE_VERSIONS}" \
/build/scripts/install_extensions pgvectorscale
USER root
# All the tools that were built in the previous steps have their ownership set to postgres
# to allow mutability. To allow one to build this image with the default privileges (owned by root)
# one can set the ALLOW_ADDING_EXTENSIONS argument to anything but "true".
ARG ALLOW_ADDING_EXTENSIONS=true
RUN set -eu; \
if [ "${ALLOW_ADDING_EXTENSIONS}" != "true" ]; then \
for pg in ${PG_VERSIONS}; do \
for dir in /usr/share/doc "$(/usr/lib/postgresql/${pg}/bin/pg_config --sharedir)/extension" "$(/usr/lib/postgresql/${pg}/bin/pg_config --pkglibdir)" "$(/usr/lib/postgresql/${pg}/bin/pg_config --bindir)"; do \
chown -R root:root "{dir}"; \
done; \
done; \
fi
RUN apt-get clean
ARG PG_MAJOR
ENTRYPOINT ["/docker-entrypoint.sh"]
CMD ["postgres"]
## TimescaleDB entrypoints and configuration scripts
## Within a k8s context, we expect the ENTRYPOINT/CMD to always be explicitly specified
COPY timescaledb_entrypoint.sh /
## Backwards compatibility, some older deployments use patroni_entrypoint.sh
RUN ln -s /timescaledb_entrypoint.sh /patroni_entrypoint.sh
COPY pgbackrest_entrypoint.sh /
## Some patroni callbacks are configured by default by the operator.
COPY scripts /scripts/
## The mount being used by the Zalando postgres-operator is /home/postgres/pgdata
## for Patroni to do it's work it will sometimes move an old/invalid data directory
## inside the parent directory; therefore we need a subdirectory inside the mount
ENV PGROOT=/home/postgres \
PGDATA=/home/postgres/pgdata/data \
PGLOG=/home/postgres/pg_log \
PGSOCKET=/home/postgres/pgdata \
BACKUPROOT=/home/postgres/pgdata/backup \
PGBACKREST_CONFIG=/home/postgres/pgdata/backup/pgbackrest.conf \
PGBACKREST_STANZA=poddb \
PATH=/usr/lib/postgresql/${PG_MAJOR}/bin:${PATH} \
LC_ALL=C.UTF-8 \
LANG=C.UTF-8 \
PAGER=""
## The Zalando postgres-operator has strong opinions about the HOME directory of postgres,
## whereas we do not. Make the operator happy then
RUN usermod postgres --home "${PGROOT}" --move-home
## The /etc/supervisor/conf.d directory is a very Spilo (Zalando postgres-operator) oriented directory.
## However, to make things work the user postgres currently needs to have write access to this directory
## The /var/lib/postgresql/data is used as PGDATA by alpine/bitnami, which makes it useful to have it be owned by Postgres
RUN install -o postgres -g postgres -m 0750 -d "${PGROOT}" "${PGLOG}" "${PGDATA}" "${BACKUPROOT}" /etc/supervisor/conf.d /scripts /var/lib/postgresql
## Making sure that pgbackrest is pointing to the right file
RUN rm /etc/pgbackrest.conf && ln -s "${PGBACKREST_CONFIG}" /etc/pgbackrest.conf
## Some configurations allow daily csv files, with foreign data wrappers pointing to the files.
## to make this work nicely, they need to exist though
RUN for i in $(seq 0 7); do touch "${PGLOG}/postgresql-$i.log" "${PGLOG}/postgresql-$i.csv"; done
## Fix permissions
RUN set -e; \
chown -R postgres:postgres "${PGLOG}" "${PGROOT}" "${PGDATA}" /var/run/postgresql/; \
chown -R postgres:postgres /var/log/pgbackrest/ /var/lib/pgbackrest /var/spool/pgbackrest; \
chmod -x /usr/lib/postgresql/*/lib/*.so; \
chmod 1777 /var/run/postgresql; \
chmod 755 "${PGROOT}"
# return /etc/apt/sources.list back to a non-AWS version for anybody that wants to use this image elsewhere
RUN set -eux; \
mv -f /etc/apt/sources.list /etc/apt/sources.list.aws; \
mv -f /etc/apt/sources.list.dist /etc/apt/sources.list
# DOCKER_FROM needs re-importing as any args from before FROM only apply to FROM
ARG DOCKER_FROM
ARG BUILDER_URL
ARG RELEASE_URL
RUN /build/scripts/install_extensions versions > /.image_config; \
echo "OSS_ONLY=\"$OSS_ONLY\"" >> /.image_config; \
echo "PG_LOGERRORS=\"${PG_LOGERRORS}\"" >> /.image_config; \
echo "PG_STAT_MONITOR=\"${PG_STAT_MONITOR}\"" >> /.image_config; \
echo "PGVECTO_RS=\"${PGVECTO_RS}\"" >> /.image_config; \
echo "POSTGIS_VERSIONS=\"${POSTGIS_VERSIONS}\"" >> /.image_config; \
echo "PG_AUTH_MON=\"${PG_AUTH_MON}\"" >> /.image_config; \
echo "PGBOUNCER_EXPORTER_VERSION=\"${PGBOUNCER_EXPORTER_VERSION}\"" >> /.image_config; \
echo "PGBACKREST_EXPORTER_VERSION=\"${PGBACKREST_EXPORTER_VERSION}\"" >> /.image_config; \
echo "PGAI_VERSION=\"${PGAI_VERSION}\"" >> /.image_config; \
echo "PGVECTORSCALE_VERSIONS=\"${PGVECTORSCALE_VERSIONS}\"" >> /.image_config; \
echo "PG_MAJOR=\"${PG_MAJOR}\"" >> /.image_config; \
echo "PG_VERSIONS=\"${PG_VERSIONS}\"" >> /.image_config; \
echo "FROM=\"${DOCKER_FROM}\"" >> /.image_config; \
echo "RELEASE_URL=\"${RELEASE_URL}\"" >> /.image_config; \
echo "BUILDER_URL=\"${BUILDER_URL}\"" >> /.image_config; \
echo "BUILD_DATE=\"$(date -Iseconds)\"" >> /.image_config
WORKDIR /home/postgres
EXPOSE 5432 8008 8081
USER postgres
# This is run during the image build process so that the build will fail and the results won't be pushed
# to the registry if there's a problem. It's run independently during CI so the output can be used in the GH summary
# so you don't have to trawl through the huge amount of logs to find the output.
COPY --chown=postgres:postgres cicd /cicd/
RUN /cicd/install_checks -v
FROM builder AS trimmed
USER root
ENV BUILD_PACKAGES="binutils cmake devscripts equivs gcc git gpg gpg-agent libc-dev libc6-dev libkrb5-dev libperl-dev libssl-dev lsb-release make patchutils python2-dev python3-dev wget libsodium-dev"
RUN set -ex; \
apt-get purge -y ${BUILD_PACKAGES}; \
apt-get autoremove -y; \
apt-get clean; \
rm -rf /var/lib/apt/lists/* \
/var/cache/debconf/* \
/usr/share/doc \
/usr/share/man \
/usr/share/locale/?? \
/usr/share/locale/??_?? \
/home/postgres/.pgx \
/build/ \
/usr/local/rustup \
/usr/local/cargo \
/cicd; \
find /var/log -type f -exec truncate --size 0 {} \;
USER postgres
## Create a smaller Docker image from the builder image
FROM scratch AS release
COPY --from=trimmed / /
ARG PG_MAJOR
ENV PGROOT=/home/postgres \
PGDATA=/home/postgres/pgdata/data \
PGLOG=/home/postgres/pg_log \
PGSOCKET=/home/postgres/pgdata \
BACKUPROOT=/home/postgres/pgdata/backup \
PGBACKREST_CONFIG=/home/postgres/pgdata/backup/pgbackrest.conf \
PGBACKREST_STANZA=poddb \
PATH=/usr/lib/postgresql/${PG_MAJOR}/bin:${PATH} \
LC_ALL=C.UTF-8 \
LANG=C.UTF-8 \
PAGER=""
# https://github.com/docker-library/postgres/commit/bfc5d81c8f5647c690f452dc558e64fddb1802f6
# We set the default STOPSIGNAL to SIGINT, which corresponds to what PostgreSQL
# calls "Fast Shutdown mode" wherein new connections are disallowed and any
# in-progress transactions are aborted, allowing PostgreSQL to stop cleanly and
# flush tables to disk, which is the best compromise available to avoid data
# corruption.
#
# Users who know their applications do not keep open long-lived idle connections
# may way to use a value of SIGTERM instead, which corresponds to "Smart
# Shutdown mode" in which any existing sessions are allowed to finish and the
# server stops when all sessions are terminated.
#
# See https://www.postgresql.org/docs/12/server-shutdown.html for more details
# about available PostgreSQL server shutdown signals.
#
# See also https://www.postgresql.org/docs/12/server-start.html for further
# justification of this as the default value, namely that the example (and
# shipped) systemd service files use the "Fast Shutdown mode" for service
# termination.
#
STOPSIGNAL SIGINT
#
# An additional setting that is recommended for all users regardless of this
# value is the runtime "--stop-timeout" (or your orchestrator/runtime's
# equivalent) for controlling how long to wait between sending the defined
# STOPSIGNAL and sending SIGKILL (which is likely to cause data corruption).
#
# The default in most runtimes (such as Docker) is 10 seconds, and the
# documentation at https://www.postgresql.org/docs/12/server-start.html notes
# that even 90 seconds may not be long enough in many instances.
WORKDIR /home/postgres
EXPOSE 5432 8008 8081
USER postgres
ENTRYPOINT ["/docker-entrypoint.sh"]
CMD ["postgres"]