Skip to content

Commit

Permalink
Allow for private CA on the Agent side (#3494)
Browse files Browse the repository at this point in the history
* Allow for private CA on the Agent side

PBENCH-1209

The staging server and local development servers now use HTTPS with certs
signed by a private pbench CA. An HTTPS connection can't be validated without
a reference to this CA.

The primary change here is in the `contrib/containerized-pbench/pbench` script
which now looks for a private CA definition, maps the CA bundle file into the
container, and defines `REQUESTS_CA_BUNDLE` within the container.

In an attempt to handle RPM installs, there's also logic to support a new
`[results]` section `pbench_ca` configuration variable to define a CA path
that will be used to verify the `PUT` to a server.

Note, this isn't being used for the `--relay` path, as that's currently not
using `https` and we'll need to figure out how we want to configure this in
the future.
  • Loading branch information
dbutenhof authored Jul 26, 2023
1 parent 971ed19 commit 2d0b189
Show file tree
Hide file tree
Showing 4 changed files with 110 additions and 40 deletions.
1 change: 1 addition & 0 deletions agent/config/pbench-agent-default.cfg
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ ssh_opts = -o BatchMode=yes -o StrictHostKeyChecking=no
api_version = 1
rest_endpoint = api/v%(api_version)s
server_rest_url = https://%(pbench_web_server)s/%(rest_endpoint)s
#server_ca =

[pbench/tools]
light-tool-set = vmstat
Expand Down
67 changes: 67 additions & 0 deletions agent/containers/images/container_build.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
#!/bin/bash

# This is an essentially "trivial" script to interactively build a full set of
# Pbench Agent containers for a platform specified by the PB_AGENT_DISTRO
# environment variable, defaulting to fedora-38.
#
# These are the same commands used in the CI; however, the CI builds the
# RPMs in the common build.sh script while the Jenkins Pipeline.gy builds the
# CI container on top of that. Here we encapsulate the steps used to get a
# functional result to make this more convenient interactively.
#
# WORKSPACE_TMP is assumed to be the root of the work area, and by default will
# be set to ${HOME}.
#
# If you want to clean the make targets for RPM and container builds, use:
#
# agent/containers/images/container_build.sh --clean

export PB_AGENT_DISTRO=${PB_AGENT_DISTRO:-fedora-38}
export WORKSPACE_TMP=${WORKSPACE_TMP:-${HOME}}

function usage {
printf "Build a Pbench Agent container for the distribution named by the\n"
printf "PB_AGENT_DISTRO environment variable, which defaults to '${PB_AGENT_DISTRO}'.\n"
printf "\nThe following options are available:\n"
printf "\n"
printf -- "\t-c|--clean\n"
printf "\t\tRemove old RPM and container image targets before building.\n"
printf -- "\t-h|--help\n"
printf "\t\tPrint this usage message and terminate.\n"
}

opts=$(getopt -q -o ch --longoptions "clean,help" -n "${0}" -- "${@}")
if [[ ${?} -ne 0 ]]; then
printf -- "%s %s\n\n\tunrecognized option specified\n\n" "${0}" "${*}" >&2
usage >&2
exit 1
fi
eval set -- "${opts}"
rpm_clean=
image_clean=
while true; do
arg=${1}
shift
case "${arg}" in
-c|--clean)
rpm_clean=distclean
image_clean=clean
;;
-h|--help)
usage
exit 0
;;
--)
break
;;
*)
printf -- "${0}: unrecognized command line argument, '${arg}'\n" >&2
usage >&2
exit 1
;;
esac
done

make -C agent/rpm ${rpm_clean} ${PB_AGENT_DISTRO}-rpm
make -C agent/containers/images CI=1 CI_RPM_ROOT=${WORKSPACE_TMP} \
${image_clean} ${PB_AGENT_DISTRO}-everything
71 changes: 32 additions & 39 deletions contrib/containerized-pbench/pbench
Original file line number Diff line number Diff line change
@@ -1,69 +1,62 @@
#! /bin/bash
#
# This script is a wrapper to facilitate the invocation of a Pbench Agent
# command using a containerized deployment of the Pbench Agent. Simply prefix
# command using a containerized deployment of the Pbench Agent. Simply prefix
# a Pbench Agent command line with the path to this script to run it inside a
# container, without needing to install the Agent on the host system.
#
# Invocation options are provided as environment variables:
# PB_AGENT_IMAGE_NAME: the full image name for the containerized Pbench Agent
# _PBENCH_AGENT_CONFIG: the location of the Pbench Agent configuration file
# PB_AGENT_RUN_DIR: the directory for use as the Pbench Agent "run directory"
# PB_AGENT_SERVER_LOC: the host and port for the Pbench Server
# PB_AGENT_PODMAN_OPTIONS: Additional options to be supplied to Podman run
# PB_AGENT_IMAGE_NAME: the full image name for the containerized Pbench Agent
# PB_AGENT_RUN_DIR: the directory for use as the Pbench Agent "run directory"
# PB_AGENT_CA: a CA bundle to verify Pbench Server PUTs
# PB_AGENT_PODMAN_OPTIONS: Additional options to be supplied to Podman run
#
# In all cases, reasonable defaults are supplied if the environment variables
# are not defined.
#
# This script checks for the presence of a `~/.ssh` directory, an existing
# Pbench Agent configuration file, and a Pbench Agent "run directory" and maps
# them into the container if they exist. If the configuration file is missing
# but the location of the Pbench Server is available, then this script will
# generate the configuration file, and the script creates the run directory if
# it does not exist. The script then invokes the Pbench Agent container with
# these options and any others which the user has specified and passes in the
# command to be executed.
# This script manages a persistent host Pbench Agent "run directory", which
# defaults to /var/tmp/{USER}/pbench-agent/run, and maps that directory into
# the container so that multiple runs can be generated and uploaded at once.
#
# To upload results to a Pbench Server, use this script to execute the
# pbench-results-move command within the container, specifying either --relay
# with the address of a Pbench Relay Server, or --server with the address of a
# Pbench Server and --token to specify a Pbench Server API key for user
# authentication.
#
# To use a server with a certificate signed by the Pbench development CA bundle
# define the environment variable PB_AGENT_CA to cause the CA to be mapped into
# the container and defined using REQUESTS_CA_BUNDLE:
#
# PB_AGENT_CA=server/pbenchinacan/etc/pki/tls/certs/pbench_CA.crt \
# contrib/containerized-pbench/pbench pbench-results-move \
# --server https://<server>:8443 --token <api-token>

image_name=${PB_AGENT_IMAGE_NAME:-quay.io/pbench/pbench-agent-all-centos-8:main}
config_file=${_PBENCH_AGENT_CONFIG:-${HOME}/.config/pbench/pbench-agent.cfg}
pbench_run_dir=${PB_AGENT_RUN_DIR:-/var/tmp/${USER}/pbench-agent/run}
pbench_server=${PB_AGENT_SERVER_LOC}
ca=${PB_AGENT_CA:-${REQUESTS_CA_BUNDLE}}
if [[ ${ca} ]]; then
pbench_ca=$(realpath ${ca}) # expand path outside container
fi
container_ca=/etc/pki/tls/certs/pbench_CA.crt # path inside container
other_options=${PB_AGENT_PODMAN_OPTIONS}

if [[ $# == 0 || $1 == "help" || $1 == "-h" || $1 == "--help" ]]; then
echo "Usage: ${0} <Pbench Agent Command> [<arg>...]" >&2
exit 2
fi

if [[ -d "${HOME}/.ssh" && -r "${HOME}/.ssh" ]]; then
other_options="--security-opt=label=disable -v ${HOME}/.ssh:/root/.ssh ${other_options}"
fi

if [[ -f "${config_file}" && -r "${config_file}" ]]; then
other_options="-v ${config_file}:/opt/pbench-agent/config/pbench-agent.cfg:z ${other_options}"
elif [[ -n "${pbench_server}" ]]; then
echo "Warning: the Pbench Agent config file is missing; attempting to generate one in ${config_file}" >&2
# TODO: this should be handled by a separate Pbench Agent "configuration wizard".
mkdir -p $(dirname ${config_file})
cat > ${config_file} <<- EOF
[DEFAULT]
pbench_install_dir = /opt/pbench-agent
pbench_web_server = ${pbench_server}
[config]
path = %(pbench_install_dir)s/config
files = pbench-agent-default.cfg
EOF
else
echo "Warning: the Pbench Agent config file (e.g., ${config_file}) is missing or inaccessible -- using default configuration." >&2
fi

mkdir -p ${pbench_run_dir}
other_options="-v ${pbench_run_dir}:/var/lib/pbench-agent:z ${other_options}"
if [[ -f "${pbench_ca}" ]]; then
other_options="-v ${pbench_ca}:${container_ca}:Z ${other_options}"
other_options="-e REQUESTS_CA_BUNDLE=${container_ca} ${other_options}"
fi

podman run \
-it \
--rm \
--network host \
--name pbench-agent \
-v ${pbench_run_dir}:/var/lib/pbench-agent:Z \
${other_options} \
${image_name} "${@}"
11 changes: 10 additions & 1 deletion lib/pbench/agent/results.py
Original file line number Diff line number Diff line change
Expand Up @@ -406,8 +406,13 @@ def __init__(
if server:
path = config.get("results", "rest_endpoint")
uri = f"{server}/{path}"
self.ca = None
else:
uri = config.get("results", "server_rest_url")

# If the "server_ca" config variable isn't defined, we expect to verify
# using a registered CA.
self.ca = config.get("results", "server_ca", fallback=None)
self.uri = f"{uri}/upload/{{name}}"
self.headers.update({"Authorization": f"Bearer {token}"})

Expand All @@ -430,7 +435,11 @@ def push(self, tarball: Path, tarball_md5: str) -> requests.Response:
tar_uri = self.uri.format(name=tarball.name)
with tarball.open("rb") as f:
return requests.put(
tar_uri, data=f, headers=self.headers, params=self.params
tar_uri,
data=f,
headers=self.headers,
params=self.params,
verify=self.ca,
)


Expand Down

0 comments on commit 2d0b189

Please sign in to comment.