From 4c266a9eef44fdbca35ac0a758cfaca6141e9230 Mon Sep 17 00:00:00 2001 From: Javier Marcos <1271349+javuto@users.noreply.github.com> Date: Sat, 16 Nov 2024 02:13:22 +0100 Subject: [PATCH] Fixes for pkg builder script --- tools/.gitignore | 11 ++ tools/README.md | 7 +- tools/build-osctrl-pkg.sh | 267 ++++++++++++++++++++++++-------------- 3 files changed, 188 insertions(+), 97 deletions(-) create mode 100644 tools/.gitignore diff --git a/tools/.gitignore b/tools/.gitignore new file mode 100644 index 00000000..6c0dd8a5 --- /dev/null +++ b/tools/.gitignore @@ -0,0 +1,11 @@ +# Files used by packages +osquery.flags +osquery.secret + +# Generated packages +*.pkg +*.deb + +# Ignore temporary directories matching the date format YYYYMMDD-HHMMSS and YYYYMMDD-HHMMSS-scripts +[0-9][0-9][0-9][0-9][0-1][0-9][0-3][0-9]-[0-2][0-9][0-5][0-9][0-5][0-9]/ +[0-9][0-9][0-9][0-9][0-1][0-9][0-3][0-9]-[0-2][0-9][0-5][0-9][0-5][0-9]-scripts/ diff --git a/tools/README.md b/tools/README.md index 2c94f3ae..85f97162 100644 --- a/tools/README.md +++ b/tools/README.md @@ -109,11 +109,14 @@ Usage: ./build-osctrl-pkg.sh [-h|--help] [PARAMETER [ARGUMENT]] [PARAMETER [ARGU Options: -h Show this help message and exit + -n NAME Name to use for the package (default: osctrl) + -V VERSION Version to use for the package (default: 1.0) -c CERT Path to the osquery certificate file -s SECRET Path to the osquery secret file (default: osquery.secret) -f FLAGS Path to the osquery flags file (default: osquery.flags) - -i PKG Path to the osquery DEB file. Required. - -o PKG Path to the osctrl DEB file. Required. + -i PKG Path to the osquery PKG file. Required. + -o PKG Path to the osctrl PKG file. Required. + -k Generate a PKG without osquery. Only osctrl files. -x Clear the temporary directory after the process -v Enable verbose mode with 'set -x' diff --git a/tools/build-osctrl-pkg.sh b/tools/build-osctrl-pkg.sh index bca81188..5382bdea 100755 --- a/tools/build-osctrl-pkg.sh +++ b/tools/build-osctrl-pkg.sh @@ -10,11 +10,14 @@ function usage() { echo echo "Options:" echo " -h Show this help message and exit" + echo " -n NAME Name to use for the package (default: osctrl)" + echo " -V VERSION Version to use for the package (default: 1.0)" echo " -c CERT Path to the osquery certificate file" echo " -s SECRET Path to the osquery secret file (default: osquery.secret)" echo " -f FLAGS Path to the osquery flags file (default: osquery.flags)" echo " -i PKG Path to the osquery PKG file. Required." echo " -o PKG Path to the osctrl PKG file. Required." + echo " -k Generate a PKG without osquery. Only osctrl files." echo " -x Clear the temporary directory after the process" echo " -v Enable verbose mode with 'set -x'" echo @@ -30,13 +33,16 @@ CERT="" SECRET="osquery.secret" FLAGS="osquery.flags" OSCTRL_FLAGS="osquery.flags" +NAME="osctrl" +VERSION="1.0" OSQUERY_PKG="" OSCTRL_PKG="" REMOVE_TMP_DIR="false" VERBOSE_MODE="false" +OSCTRL_ONLY="false" # Parse command line arguments -while getopts "c:s:f:i:o:xh" opt; do +while getopts "c:s:f:i:o:kxhv" opt; do case "$opt" in c) CERT=${OPTARG} @@ -56,6 +62,9 @@ while getopts "c:s:f:i:o:xh" opt; do x) REMOVE_TMP_DIR="true" ;; + k) + OSCTRL_ONLY="true" + ;; v) VERBOSE_MODE="true" ;; @@ -67,23 +76,39 @@ while getopts "c:s:f:i:o:xh" opt; do done # If no osquery PKG file or no osctrl PKG file, show usage -if [[ -z "$OSQUERY_PKG" ]] || [[ ! -f "$OSQUERY_PKG" ]]; then - echo "[!] Invalid input for osquery PKG file. Please provide a valid file." - exit 1 +if [[ "$OSCTRL_ONLY" == "false" ]]; then + if [[ -z "$OSQUERY_PKG" ]] || [[ ! -f "$OSQUERY_PKG" ]]; then + echo "[!] Invalid input for osquery PKG file. Please provide a valid file." + exit 1 + fi fi - if [[ -z "$OSCTRL_PKG" ]]; then echo "[!] Output file for osctrl PKG can not be empty." + exit 1 fi -echo "[+] Using osquery PKG file: ${OSQUERY_PKG}" echo "[+] Generating osctrl PKG file: ${OSCTRL_PKG}" if [[ ! -z "$CERT" ]]; then echo "[+] Using osquery certificate file: ${CERT}" fi + +# Check if secret and flags files exist +if [[ ! -f "$SECRET" ]]; then + echo "[!] Invalid input for osquery secret file. Please provide a valid file." + exit 1 +fi +if [[ ! -f "$FLAGS" ]]; then + echo "[!] Invalid input for osquery flags file. Please provide a valid file." + exit 1 +fi echo "[+] Using osquery secret file: ${SECRET}" echo "[+] Using osquery flags file: ${FLAGS}" +# If we want verbose, set -x to debug the script +if [[ "$VERBOSE_MODE" == "true" ]]; then + set -x +fi + # Check if pkgutil is installed if ! command -v pkgutil &> /dev/null then @@ -91,107 +116,159 @@ then exit 1 fi -# Check if gzip is installed -if ! command -v gzip &> /dev/null -then - echo "[!] gzip could not be found" - exit 1 -fi +# Create a temporary directory to unpack the DEB file +TMP_DIR=$(date +'%Y%m%d-%H%M%S') +echo "[+] Using temporary directory $TMP_DIR" -# Check if cpio is installed -if ! command -v cpio &> /dev/null -then - echo "[!] cpio could not be found" - exit 1 -fi +# Do we want to generate a PKG without osquery? +if [[ "$OSCTRL_ONLY" == "true" ]]; then + # Files for osquery + mkdir -p "$TMP_DIR/private/var/osquery" + cp "$SECRET" "$TMP_DIR/private/var/osquery" + cp "$FLAGS" "$TMP_DIR/private/var/osquery" -# Check if mkbom is installed -if ! command -v mkbom &> /dev/null -then - echo "[!] mkbom could not be found" - exit 1 -fi + # Package scripts + mkdir -p "$TMP_DIR-scripts" + cat <"$TMP_DIR-scripts/preinstall" +#!/usr/bin/env bash -# If we want verbose, set -x to debug the script -if [[ "$VERBOSE_MODE" == "true" ]]; then - set -x +if launchctl list | grep -qcm1 io.osquery.agent; then + sudo launchctl unload /Library/LaunchDaemons/io.osquery.agent.plist fi -# Create a temporary directory to unpack the DEB file -TMP_DIR=$(date +'%Y%m%d-%H%M%S') -echo "[+] Using temporary directory $TMP_DIR" +exit 0 +EOF + cat <"$TMP_DIR-scripts/postinstall" +#!/usr/bin/env bash + +sudo cp /private/var/osquery/io.osquery.agent.plist /Library/LaunchDaemons/io.osquery.agent.plist +sudo launchctl load /Library/LaunchDaemons/io.osquery.agent.plist + +exit 0 +EOF + + # Prepare identifier string + _IDENTIFIER="$NAME-$VERSION" + + # Build pkg package natively with pkgbuild + pkgbuild --root "$TMP_DIR" \ + --scripts "$TMP_DIR-scripts" \ + --identifier "$_IDENTIFIER" \ + --version "$VERSION" \ + "$OSCTRL_PKG" + + echo "✅ Completed creating $_IDENTIFIER PKG file: $OSCTRL_PKG" +else + # Get the current working directory + cwd=$(pwd) + echo "[+] Using osquery PKG file: ${OSQUERY_PKG}" + + # Check if gzip is installed + if ! command -v gzip &> /dev/null + then + echo "[!] gzip could not be found" + exit 1 + fi + # Check if cpio is installed + if ! command -v cpio &> /dev/null + then + echo "[!] cpio could not be found" + exit 1 + fi + # Check if mkbom is installed + if ! command -v mkbom &> /dev/null + then + echo "[!] mkbom could not be found" + exit 1 + fi + # Extract the osquery PKG file + echo "[+] Extracting osquery PKG file" + pkgutil --expand-full "$OSQUERY_PKG" "$TMP_DIR" + + cd "$TMP_DIR" + ############## From here, we are in the temporary directory ############## + + # Get paths from the flags file for certificate and secret + echo "[+] Getting paths from the flags file" + _FLAGS=$FLAGS + if [[ ! -f "$FLAGS" ]]; then + _FLAGS="$cwd/$FLAGS" + fi + CERTPATH=$(grep "tls_server_certs=" "$_FLAGS" | awk -F'=' '{print $2}') + SECRETPATH=$(grep "enroll_secret_path=" "$_FLAGS" | awk -F'=' '{print $2}') + + # Extract number of files and size in KB from PackageInfo + echo "[+] Extracting number of files and size from PackageInfo" + NOF=$(cat PackageInfo | grep payload | awk -F'"' '{print $2}') + SIZEKB=$(cat PackageInfo | grep payload | awk -F'"' '{print $4}') + IDENTIFIER=$(cat PackageInfo | grep identifier | awk -F'"' '{print $6}') + VERSION=$(cat PackageInfo | grep 'version=' | awk -F'"' '{print $10}') -# Extract the osquery PKG file -echo "[+] Extracting osquery PKG file" -pkgutil --expand-full "$OSQUERY_PKG" "$TMP_DIR" - -cwd=$(pwd) -cd "$TMP_DIR" -############## From here, we are in the temporary directory ############## - -# Get paths from the flags file for certificate and secret -echo "[+] Getting paths from the flags file" -CERTPATH=$(grep "tls_server_certs=" "$cwd/$FLAGS" | awk -F'=' '{print $2}') -SECRETPATH=$(grep "enroll_secret_path=" "$cwd/$FLAGS" | awk -F'=' '{print $2}') - -# Extract number of files and size in KB from PackageInfo -echo "[+] Extracting number of files and size from PackageInfo" -NOF=$(cat PackageInfo | grep payload | awk -F'"' '{print $2}') -SIZEKB=$(cat PackageInfo | grep payload | awk -F'"' '{print $4}') -IDENTIFIER=$(cat PackageInfo | grep identifier | awk -F'"' '{print $6}') -VERSION=$(cat PackageInfo | grep 'version=' | awk -F'"' '{print $10}') - -counter_file=0 -counter_size=0 -# Copy the osctrl files to the data directory -echo "[+] Copying osctrl files to Payload directory" -if [[ ! -z "$CERT" ]] && [[ ! -z "$CERTPATH" ]]; then - cp "$cwd/$CERT" "Payload$CERTPATH" + counter_file=0 + counter_size=0 + + # Copy the osctrl files to the data directory + echo "[+] Copying osctrl files to Payload directory" + if [[ ! -z "$CERT" ]] && [[ ! -z "$CERTPATH" ]]; then + _CERT=$CERT + if [[ ! -f "$CERT" ]]; then + _CERT="$cwd/$CERT" + fi + cp "$_CERT" "Payload$CERTPATH" + counter_file=$((counter_file+1)) + counter_size=$((counter_size+$(du -k "$_CERT" | cut -f1))) + fi + _SECRET=$SECRET + if [[ ! -f "$SECRET" ]]; then + _SECRET="$cwd/$SECRET" + fi + cp "$_SECRET" "Payload$SECRETPATH" + counter_file=$((counter_file+1)) + counter_size=$((counter_size+$(du -k "$_SECRET" | cut -f1))) + cp "$_FLAGS" "Payload/private/var/osquery/$OSCTRL_FLAGS" counter_file=$((counter_file+1)) - counter_size=$((counter_size+$(du -k "$cwd/$CERT" | cut -f1))) + counter_size=$((counter_size+$(du -k "$_FLAGS" | cut -f1))) + + # Update the PackageInfo file + echo "[+] Updating PackageInfo file" + sed -i '' "s/$NOF/$((NOF+counter_file))/g" PackageInfo + sed -i '' "s/$SIZEKB/$((SIZEKB+counter_size))/g" PackageInfo + + # Update the BOM file + echo "[+] Updating BOM file" + mkbom "Payload/" Bom + + # Recompress the Payload directory + echo "[+] Recompressing Payload directory" + find Payload/ | cpio -o --format odc | gzip -c > Payload.gz + + # Replace the Payload directory with the new one + echo "[+] Replacing the Payload directory" + rm -rf Payload + mv Payload.gz Payload + + # Create the new PKG file + echo "[+] Creating the new PKG file" + #pkgutil --flatten . "$OSCTRL_PKG" + pkgbuild --root . \ + --identifier "$IDENTIFIER" \ + --version "$VERSION" \ + --install-location "/" \ + "$OSCTRL_PKG" + + # Move the new PKG file to the original directory + cp "$OSCTRL_PKG" "$cwd" + + # Clean up + cd "$cwd" + + echo "✅ Completed repacking osquery PKG file: $OSCTRL_PKG" fi -cp "$cwd/$SECRET" "Payload$SECRETPATH" -counter_file=$((counter_file+1)) -counter_size=$((counter_size+$(du -k "$cwd/$SECRET" | cut -f1))) -cp "$cwd/$FLAGS" "Payload/private/var/osquery/$OSCTRL_FLAGS" -counter_file=$((counter_file+1)) -counter_size=$((counter_size+$(du -k "$cwd/$FLAGS" | cut -f1))) - -# Update the PackageInfo file -echo "[+] Updating PackageInfo file" -sed -i '' "s/$NOF/$((NOF+counter_file))/g" PackageInfo -sed -i '' "s/$SIZEKB/$((SIZEKB+counter_size))/g" PackageInfo - -# Update the BOM file -echo "[+] Updating BOM file" -mkbom "Payload/" Bom - -# Recompress the Payload directory -echo "[+] Recompressing Payload directory" -find Payload/ | cpio -o --format odc | gzip -c > Payload.gz - -# Replace the Payload directory with the new one -echo "[+] Replacing the Payload directory" -rm -rf Payload -mv Payload.gz Payload - -# Create the new PKG file -echo "[+] Creating the new PKG file" -#pkgutil --flatten . "$OSCTRL_PKG" -pkgbuild --root . \ - --identifier "$IDENTIFIER" \ - --version "$VERSION" \ - --install-location "/" \ - "$OSCTRL_PKG" - -# Move the new PKG file to the original directory -cp "$OSCTRL_PKG" "$cwd" # Clean up -cd "$cwd" if [[ "$REMOVE_TMP_DIR" == "true" ]]; then echo "[+] Removing temporary directory" rm -rf "$TMP_DIR" fi -echo "✅ Completed repacking osquery PKG file: $OSCTRL_PKG" +exit 0