#!/bin/bash #--------------------------------------------------------------------- # Script to determine if specified config file is valid or not. # By default, two checks are performed: # # - Ensure Upstart can parse overall file successfully # - Ensure all script sections are parseable by shell # #--------------------------------------------------------------------- # # Copyright (C) 2011 Canonical Ltd. # # Author: James Hunt # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, version 3 of the License. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . # #--------------------------------------------------------------------- script_name=${0##*/} confdir=$(mktemp -d /tmp/${script_name}.XXXXXXXXXX) upstart_path=/sbin/init initctl_path=/sbin/initctl debug_enabled=n file_valid=n running=n check_scripts=y cleanup() { if [ ! -z "$upstart_pid" ] then debug "stopping secondary Upstart (running with PID $upstart_pid)" kill -0 "$upstart_pid" >/dev/null 2>&1 && \ kill -9 "$upstart_pid" >/dev/null 2>&1 fi [ -d "$confdir" ] && rm -rf "$confdir" [ $file_valid = y ] && exit 0 exit 1 } usage() { cat < $script_name [options] Options: -d, --debug : Show some debug output. -f , : Job configuration file to check. --file= (no default). -i , : Specify path to initctl binary --initctl-path= (default=$initctl_path). -s, --noscript : Do not check script sections. -x : Specify path to init daemon binary --upstart-path= (default=$upstart_path). -h, --help : Show this help. EOT } debug() { msg="$*" [ $debug_enabled = y ] && echo "DEBUG: $msg" } error() { msg="$*" printf "ERROR: %s\n" "$msg" >&2 } die() { error "$*" exit 1 } # Return 0 if Upstart is running on the D-Bus session bus, else 1. upstart_running() { dbus-send --session --print-reply \ --dest='com.ubuntu.Upstart' /com/ubuntu/Upstart \ org.freedesktop.DBus.Properties.GetAll \ string:'com.ubuntu.Upstart0_6' >/dev/null 2>&1 } trap cleanup EXIT INT TERM args=$(getopt \ -n "$script_name" \ -a \ --options="df:hi:sx:" \ --longoptions="debug file: help initctl-path: noscript upstart-path:" \ -- "$@") eval set -- "$args" [ $? -ne 0 ] && { usage; exit 1; } [ $# -eq 0 ] && { usage; exit 0; } while [ $# -gt 0 ] do case "$1" in -d|--debug) debug_enabled=y ;; -f|--file) file="$2" shift ;; -h|--help) usage exit 0 ;; -i|--initctl-path) initctl_path="$2" shift ;; -s|--noscript) check_scripts=n ;; -x|--upstart-path) upstart_path="$2" shift ;; --) shift break ;; esac shift done [ -z "$file" ] && file="$1" # safety first [ "$(id -u)" -eq 0 ] && die "cannot run as root" [ -z "$file" ] && die "must specify configuration file" [ ! -f "$file" ] && die "file $file does not exist" debug "upstart_path=$upstart_path" debug "initctl_path=$initctl_path" for cmd in "$upstart_path" "$initctl_path" do [ -f "$cmd" ] || die "Path $cmd does not exist" [ -x "$cmd" ] || die "File $cmd not executable" "$cmd" --help | grep -q -- --session || die "version of $cmd too old" done # this is the only safe way to run another instance of Upstart "$upstart_path" --help|grep -q -- --no-startup-event || die "$upstart_path too old" debug "confdir=$confdir" debug "file=$file" filename=$(basename $file) echo "$filename" | egrep -q '\.conf$' || die "file must end in .conf" job="${filename%.conf}" cp "$file" "$confdir" debug "job=$job" upstart_running [ $? -eq 0 ] && die "Another instance of this program is already running" debug "ok - no other running instances detected" upstart_out="$(mktemp --tmpdir "${script_name}-upstart-output.XXXXXXXXXX")" debug "upstart_out=$upstart_out" upstart_cmd=$(printf \ "%s --session --no-sessions --no-startup-event --verbose --confdir %s" \ "$upstart_path" \ "$confdir") debug "upstart_cmd=$upstart_cmd" nohup $upstart_cmd >"$upstart_out" 2>&1 & upstart_pid=$! # Stop the shell outputting a message when Upstart is killed. # We handle this ourselves in cleanup(). disown # wait for Upstart to initialize for i in $(seq 1 5) do debug "Waiting for Upstart to reply over D-Bus (attempt $i)" upstart_running if [ $? -eq 0 ] then running=y break fi sleep 1 done [ $running = n ] && die "failed to ask Upstart to check conf file" debug "Secondary Upstart ($upstart_cmd) running with PID $upstart_pid" if [ "$check_scripts" = y ] then for section in pre-start post-start script pre-stop post-stop do if egrep -q "\<${section}\>" "$file" then cmd='sed -n "/^ *${section}/,/^ *end script/p" $file | /bin/sh -n 2>&1' errors=$(eval "$cmd") [ $? -ne 0 ] && \ die "$(printf "File $file: shell syntax invalid in $section section:\n${errors}")" fi done fi "$initctl_path" --session list|grep -q "^${job}" if [ $? -eq 0 ] then file_valid=y echo "File $file: syntax ok" exit 0 fi errors=$(grep "$job" "$upstart_out"|sed "s,${confdir}/,,g") die "$(printf "File $file: syntax invalid:\n${errors}")"