Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Building DEB packages with podman & pbuilder #8

Merged
merged 45 commits into from
Mar 29, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
45 commits
Select commit Hold shift + click to select a range
06df6f3
Working scaffold for Pgpm::Deb::Builder where every method is a build…
Mar 1, 2025
0a1c5f2
Pgpm::Deb::Builder #prepare and #create_container methods
Mar 2, 2025
676ac4d
Generating debian files used when building a deb package with pbuilder
Mar 2, 2025
11cc4de
Pgpm::Deb::Buider#cleanup stops & removes podman container, removes t…
Mar 2, 2025
0b2dd51
Pgpm::Deb::Builder build with pbuilder, then move .deb file to the host
Mar 2, 2025
62748ff
Pgpb::Deb:Builder pull image unless it exists
Mar 2, 2025
aa0d915
Refactor: replace most hardcoded values for Pgpm::Deb::Spec with valu…
Mar 6, 2025
97d5506
Refactor: make build run all podman commands in one go, then exit
Mar 6, 2025
e26594f
Fix: hardocoded podman image name instead of using Pgpm::Deb::Builder…
Mar 7, 2025
29b258b
Choosing arch suffix for debian packages (it's not the same as in RPM!)
Mar 7, 2025
1f8b9b2
WIP: Timescaledb package with #build_info_for() for Debian and RedHat
Mar 7, 2025
3165c10
Add Package#licence which attempts to read LICENCE file in pkg root a…
Mar 7, 2025
70cb28a
Fix: email address domain name for debian packages
Mar 8, 2025
7dec4dc
Refactor: use Pgpm::Arch.in_scope to determine architecture
Mar 8, 2025
998546d
Rename Pgpm::Package::PGXN#license into #license_text, restore old #l…
Mar 8, 2025
4ac7c0f
Specify default & custom dependencies in packages
Mar 8, 2025
97581e7
Fix: dpkg-buildpackage --build=source step in Pgpm::Deb::Builder shou…
Mar 8, 2025
3b02c92
Proper changelog template for pbuilder packaging
Mar 8, 2025
0a84b1d
Assign .os and .postgres_major_version to @package in Pgpm::RPM::Spec…
Mar 8, 2025
d39f047
Refactor: simpler way to extract postgres major version
Mar 9, 2025
164cc59
Add Dockerfile to build Debian images
Mar 10, 2025
3122b37
Run build commands in podman through `podman exec` so later failed co…
Mar 11, 2025
a1ca046
Patch pbuilder for before running it: prevent cleanup
Mar 11, 2025
0b62193
Fix: use Pgpm::OS.in_scope to determine OS name at the package level
Mar 11, 2025
0e9ce56
Remove 'require "debug"'
Mar 11, 2025
f2339f8
Refactor: use @spec.sources to download and extract sources when buil…
Mar 12, 2025
71bcfdd
Rename & package versioned files into a DEB package, assign proper pa…
Mar 15, 2025
cb58e0a
Create second .deb package (default) which depends on the versioned p…
Mar 16, 2025
8de62ed
Refactor: prepare_artifacts in deb/
Mar 17, 2025
6b56207
Merge branch 'master' into deb
Mar 17, 2025
7b2aee8
Restore accidentally deleted pgpm.spec
Mar 20, 2025
775da7c
Use configure_steps, make_steps & install_steps instead of build_info…
Mar 20, 2025
4bd0cbf
Remove build-essential from the list of default build dependencies
Mar 20, 2025
a99f00b
Accept a larger diversity of filenames for License
Mar 20, 2025
2a4a708
Oops: spelling of "ubuntu"
Mar 21, 2025
de2e49d
Merge branch 'master' into deb
Mar 21, 2025
e14be75
Use Package#native? as condition to add "build-essential" to build_de…
Mar 21, 2025
8d0c175
Refactor: use completely customized debian rules file -- without call…
Mar 26, 2025
8eb2766
Fix: better way to handle custom configure, build and install steps i…
Mar 27, 2025
6fb0482
Fix: timescaledb #build_steps method with custom steps for debian
Mar 27, 2025
a9ac637
Fix: deb package names should not include _, replace them with -
Mar 28, 2025
7f4217c
Fix: os-specific dependency names for pgsodium package
Mar 28, 2025
d0a7907
Oops: remove `require "debug"`
Mar 28, 2025
79e1172
Fix: remove default dependencies for rocky/redheat packages + update …
Mar 28, 2025
3073f73
Fixing rubocop issues
Mar 29, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
66 changes: 42 additions & 24 deletions exe/pgpm
Original file line number Diff line number Diff line change
Expand Up @@ -101,35 +101,53 @@ module Pgpm
exit(1)
end

unless os.is_a?(Pgpm::OS::RedHat)
puts "#{os.name} is not a supported OS at this moment"
exit(1)
end
puts "Building #{pkgs.map { |p| "#{p.name}@#{p.version}" }.join(", ")} for Postgres #{matching_pgver}"
selected_pgdist = Postgres::RedhatBasedPgdg.new(matching_pgver.to_s)

os.with_scope do
arch.with_scope do
selected_pgdist.with_scope do
pkgs = pkgs.flat_map(&:topologically_ordered_with_dependencies).uniq.reject(&:contrib?)

b = pkgs.reduce(nil) do |c, p|
if p.broken?
puts "Can't build a broken package #{p.name}@#{p.version}"
exit(1)
if os.is_a? Pgpm::OS::Debian
puts "Building #{pkgs.map { |p| "#{p.name}@#{p.version}" }.join(", ")} for Postgres #{matching_pgver}"
selected_pgdist = Postgres::RedhatBasedPgdg.new(matching_pgver.to_s)

os.with_scope do
arch.with_scope do
selected_pgdist.with_scope do
spec = nil
pkgs.reduce(nil) do |_c, p|
p = Pgpm::ScopedObject.new(p, os, arch)
spec = p.to_deb_spec
end
p = Pgpm::ScopedObject.new(p, os, arch)
spec = p.to_rpm_spec
builder = Pgpm::RPM::Builder.new(spec)
src_builder = builder.source_builder
p = c.nil? ? src_builder : c.and_then(src_builder)
p.and_then(builder.versionless_builder)
builder = Pgpm::Deb::Builder.new(spec)
builder.build
end
end
end
elsif os.is_a? Pgpm::OS::RedHat
puts "Building #{pkgs.map { |p| "#{p.name}@#{p.version}" }.join(", ")} for Postgres #{matching_pgver}"
selected_pgdist = Postgres::RedhatBasedPgdg.new(matching_pgver.to_s)

os.with_scope do
arch.with_scope do
selected_pgdist.with_scope do
pkgs = pkgs.flat_map(&:topologically_ordered_with_dependencies).uniq.reject(&:contrib?)

b = pkgs.reduce(nil) do |c, p|
if p.broken?
puts "Can't build a broken package #{p.name}@#{p.version}"
exit(1)
end
p = Pgpm::ScopedObject.new(p, os, arch)
spec = p.to_rpm_spec
builder = Pgpm::RPM::Builder.new(spec)
src_builder = builder.source_builder
p = c.nil? ? src_builder : c.and_then(src_builder)
p.and_then(builder.versionless_builder)
end

srpms = b.call
Pgpm::RPM::Builder.builder(srpms).call
srpms = b.call
Pgpm::RPM::Builder.builder(srpms).call
end
end
end
else
puts "#{os.name} is not a supported OS at this moment"
exit(1)
end
end

Expand Down
31 changes: 31 additions & 0 deletions lib/pgpm/deb/Dockerfile
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
# syntax = docker/dockerfile:experimental

# IMPORTANT: build it this way to allow for privileged execution
#
# Docker daemon config should have the entitlement
# ```json
# { "builder": {"Entitlements": {"security-insecure": true }} }
# ```
# ```
# DOCKER_BUILDKIT=1 docker build --allow security.insecure -t IMAGE_NAME /path/to/pgpm
# ```

# This Dockerfile is used to build a Debian image, which includes pbuilder and
# pbuilder chroot image with basic dependendencies needed for building most
# packages already pre-installed.

FROM docker.io/library/debian

MAINTAINER PGPM Debian Maintainer [email protected]

VOLUME /proc
ARG DEBIAN_FRONTEND=noninteractive
RUN apt update
RUN apt install -y build-essential pbuilder fakeroot fakechroot
RUN echo 'MIRRORSITE=http://deb.debian.org/debian' > /etc/pbuilderrc
RUN echo 'AUTO_DEBSIGN=${AUTO_DEBSIGN:-no}' > /root/.pbuilderrc
RUN echo 'HOOKDIR=/var/cache/pbuilder/hooks' >> /root/.pbuilderrc
RUN --security=insecure pbuilder create

COPY pbuilder_install_script.sh /root/pbuilder_install_script.sh
RUN --security=insecure pbuilder execute --save-after-exec /root/pbuilder_install_script.sh
207 changes: 207 additions & 0 deletions lib/pgpm/deb/builder.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,207 @@
# frozen_string_literal: true

require "English"
require "debug"

module Pgpm
module Deb
class Builder
def initialize(spec)
@spec = spec
@container_name = "pgpm-debian_build-#{Time.now.to_i}_#{rand(10_000)}"
@pgpm_dir = Dir.mktmpdir
end

def build
pull_image
start_container
patch_pbuilder

prepare_versioned_source
generate_deb_src_files(:versioned)
run_build(:versioned)
copy_build_from_container(:versioned)

prepare_default_source
generate_deb_src_files(:default)
run_build(:default)
copy_build_from_container(:default)

cleanup
end

private

# Depends on postgres version and arch
def image_name
"quay.io/qount25/pgpm-debian-pg#{@spec.package.postgres_major_version}-#{@spec.arch}"
end

def prepare_versioned_source
puts "Preparing build..."
puts " Creating container dir structure..."
Dir.mkdir "#{@pgpm_dir}/source-versioned"
Dir.mkdir "#{@pgpm_dir}/out"

puts " Downloading and unpacking sources to #{@pgpm_dir}"

fn = nil
@spec.sources.map do |src|
srcfile = File.join(@pgpm_dir.to_s, src.name)
File.write(srcfile, src.read)
fn = src.name
end

system("tar -xf #{@pgpm_dir}/#{fn} -C #{@pgpm_dir}/source-versioned/")
FileUtils.remove("#{@pgpm_dir}/#{fn}")

untar_dir_entries = Dir.entries("#{@pgpm_dir}/source-versioned/").reject do |entry|
[".", ".."].include?(entry)
end

if untar_dir_entries.size == 1
entry = untar_dir_entries[0]
if File.directory?("#{@pgpm_dir}/source-versioned/#{entry}")
FileUtils.mv "#{@pgpm_dir}/source-versioned/#{entry}", "#{@pgpm_dir}/"
FileUtils.remove_dir "#{@pgpm_dir}/source-versioned/"
FileUtils.mv "#{@pgpm_dir}/#{entry}", "#{@pgpm_dir}/source-versioned"
end
end

["prepare_artifacts.sh"].each do |f|
script_fn = File.expand_path("#{__dir__}/scripts/#{f}")
FileUtils.cp script_fn, "#{@pgpm_dir}/source-versioned/"
end
end

def prepare_default_source
Dir.mkdir "#{@pgpm_dir}/source-default"

# 1. All pbuilder builds are in /var/cache/pbuilder/build. At this point
# there's only one build, but we don't know what the directory is named
# (the name is usually some numbers). So we just pick the first (and only)
# entry at this location and this is our build dir.
pbuilds_dir = "/var/cache/pbuilder/build"
cmd = "ls -U #{pbuilds_dir} | head -1"
build_dir = `podman exec #{@container_name} /bin/bash -c '#{cmd}'`.strip
puts "BUILD DIR IS: #{pbuilds_dir}/#{build_dir}"

# 2. Determine the name of the .control file inside the versioned build
deb_dir = "#{pbuilds_dir}/#{build_dir}/build/#{@spec.deb_pkg_name(:versioned)}-0/debian/#{@spec.deb_pkg_name(:versioned)}"
control_fn = "#{deb_dir}/usr/share/postgresql/#{@spec.package.postgres_major_version}/extension/#{@spec.package.extension_name}--#{@spec.package.version}.control"

# 3. Copy .control file to the source-default dir
puts "Copying #{control_fn} into /root/pgpm/source-default/"
target_control_fn = "/root/pgpm/source-default/#{@spec.package.extension_name}.control"
cmd = "cp #{control_fn} #{target_control_fn}"
system("podman exec #{@container_name} /bin/bash -c '#{cmd}'")

["install_default_control.sh"].each do |fn|
script_fn = File.expand_path("#{__dir__}/scripts/#{fn}")
FileUtils.cp script_fn, "#{@pgpm_dir}/source-default/"
end
end

def pull_image
puts "Checking if podman image exists..."
# Check if image exists
system("podman image exists #{image_name}")
if $CHILD_STATUS.to_i.positive? # image doesn't exist -- pull image from a remote repository
puts " No. Pulling image #{image_name}..."
system("podman pull #{image_name}")
else
puts " Yes, image #{image_name} already exists! OK"
end
end

def generate_deb_src_files(pkg_type = :versioned)
puts "Generating debian files..."
Dir.mkdir "#{@pgpm_dir}/source-#{pkg_type}/debian"
%i[changelog control copyright files rules].each do |f|
puts " -> #{@pgpm_dir}/source-#{pkg_type}/debian/#{f}"
File.write "#{@pgpm_dir}/source-#{pkg_type}/debian/#{f}", @spec.generate(f, pkg_type)
end
File.chmod 0o740, "#{@pgpm_dir}/source-#{pkg_type}/debian/rules" # rules file must be executable
end

def start_container
# podman create options
create_opts = " -v #{@pgpm_dir}:/root/pgpm"
create_opts += ":z" if selinux_enabled?
create_opts += " --privileged --tmpfs /tmp"
create_opts += " --name #{@container_name} #{image_name}"

puts " Creating and starting container #{@container_name} & running pbuilder"
system("podman create -it #{create_opts}")
exit(1) if $CHILD_STATUS.to_i.positive?
system("podman start #{@container_name}")
exit(1) if $CHILD_STATUS.to_i.positive?
end

# Prevents clean-up after pbuilder finishes. There's no option
# in pbuilder to do it, so we have to patch it manually. The issue is
# with pbuilder not being able to delete some directories (presumably,
# due to directory names starting with ".") and returning error.
#
# This little patch avoids the error by returning from the python cleanup
# function early -- because the package itself is built successfully and
# we don't actually care that pbuilder is unable to clean something up.
# The container is going to be removed anyway, so it's even less work as
# a result.
def patch_pbuilder
cmd = "sed -E -i \"s/(^function clean_subdirectories.*$)/\\1\\n return/g\" /usr/lib/pbuilder/pbuilder-modules"
system("podman exec #{@container_name} /bin/bash -c '#{cmd}'")
end

def run_build(pkg_type = :versioned)
dsc_fn = "#{@spec.deb_pkg_name(pkg_type)}_0-1.dsc"
deb_fn = "#{@spec.deb_pkg_name(pkg_type)}_0-1_#{@spec.arch}.deb"

cmds = []
cmds << "dpkg-buildpackage --build=source -d" # -d flag helps with dependencies error
cmds << "fakeroot pbuilder build ../#{dsc_fn}"
cmds << "mv /var/cache/pbuilder/result/#{deb_fn} /root/pgpm/out/"

puts " Building package with pbuilder..."
cmds.each do |cmd|
system("podman exec -w /root/pgpm/source-#{pkg_type} #{@container_name} /bin/bash -c '#{cmd}'")
exit(1) if $CHILD_STATUS.to_i.positive?
end
end

def copy_build_from_container(pkg_type = :versioned)
puts "Copying .deb file from podman container into current directory..."
deb_fn = "#{@spec.deb_pkg_name(pkg_type)}_0-1_#{@spec.arch}.deb"
deb_copy_fn = "#{@spec.deb_pkg_name(pkg_type)}_#{@spec.arch}.deb"
FileUtils.cp("#{@pgpm_dir}/out/#{deb_fn}", "#{Dir.pwd}/#{deb_copy_fn}")
end

def cleanup
puts "Cleaning up..."

puts " Stopping destroying podman container: #{@container_name}"
system("podman container stop #{@container_name}")
system("podman container rm #{@container_name}")

# Remove temporary files
#
# Make sure @pgpm_dir starts with "/tmp/" or we may accidentally
# delete something everything! You can never be sure!
if @pgpm_dir.start_with?("/tmp/")
puts " Removing temporary files in #{@pgpm_dir}"
FileUtils.rm_rf(@pgpm_dir)
else
puts "WARNING: will not remove temporary files, strange path: \"#{@pgpm_dir}\""
end
end

# Needed because SELinux requires :z suffix for mounted directories to
# be accessible -- otherwise we get "Permission denied" when cd into a
# mounted dir inside the container.
def selinux_enabled?
# This returns true or false by itself
system("sestatus | grep 'SELinux status' | grep -o 'enabled'")
end
end
end
end
21 changes: 21 additions & 0 deletions lib/pgpm/deb/pbuilder_install_script.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
#!/usr/bin/env bash
apt update
DEBIAN_FRONTEND=noninteractive apt -y install build-essential curl lsb-release ca-certificates

### PostgreSQL installation
#
install -d /usr/share/postgresql-common/pgdg
curl -o /usr/share/postgresql-common/pgdg/apt.postgresql.org.asc --fail https://www.postgresql.org/media/keys/ACCC4CF8.asc

# Create the repository configuration file:
sh -c 'echo "deb [signed-by=/usr/share/postgresql-common/pgdg/apt.postgresql.org.asc] https://apt.postgresql.org/pub/repos/apt $(lsb_release -cs)-pgdg main" > /etc/apt/sources.list.d/pgdg.list'

# Update the package lists:
apt update

# Install the latest version of PostgreSQL:
# If you want a specific version, use 'postgresql-16' or similar instead of 'postgresql'
apt -y install postgresql-17 postgresql-server-dev-17 postgresql-common
#
### END OF PostgreSQL installation

12 changes: 12 additions & 0 deletions lib/pgpm/deb/scripts/install_default_control.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
#!/usr/bin/env bash

ext_dir="$PGPM_INSTALL_ROOT/$(pg_config --sharedir)/extension"
control_fn="$ext_dir/$PGPM_EXTENSION_NAME.control"

echo "Creating extension dir: $ext_dir"
mkdir -p "$ext_dir"

echo "Creating control file: $control_fn"
cp "$PGPM_BUILDROOT/$PGPM_EXTENSION_NAME.control" "$ext_dir/"
echo >> "$control_fn"
echo "default_version = '$PGPM_EXTENSION_VERSION'" >> "$control_fn"
21 changes: 21 additions & 0 deletions lib/pgpm/deb/scripts/pg_config.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
#! /usr/bin/env bash

# Ensure PG_CONFIG is set
if [[ -z "$PG_CONFIG" ]]; then
echo "Error: PG_CONFIG is not set."
exit 1
fi

# Wrapper function for pg_config
pg_config_wrapper() {
"$PG_CONFIG" "$@" | while read -r line; do
if [[ -n "$PGPM_REDIRECT_TO_BUILDROOT" && -f "$line" || -d "$line" ]]; then
echo "$PGPM_INSTALL_ROOT$line"
else
echo "$line"
fi
done
}

# Call the wrapper function with the arguments passed to the script
pg_config_wrapper "$@"
Loading