From 2dd4a89d5726e13b962c1e287d84a6c30f5dd46c Mon Sep 17 00:00:00 2001 From: Eric Wong Date: Wed, 27 Oct 2010 13:51:12 -0700 Subject: configurator: reloading with unset values restores default If a configuration directive is set at startup and later unset, it correctly restores the original default value as if it had never been set in the first place. This applies to the majority of the configuration values with a few exceptions: * This only applies to stderr_path and stdout_path when daemonized (the usual case, they'll be redirected to "/dev/null"). When NOT daemonized, we cannot easily redirect back to the original stdout/stderr destinations. * Unsetting working_directory does not restore the original working directory where Unicorn was started. As far as we can tell unsetting this after setting it is rarely desirable and greatly increases the probability of user error. (cherry picked from commit 51b2b90284000aee8d79b37a5406173c45ae212d) --- lib/unicorn/configurator.rb | 16 ++++++--- t/t0012-reload-empty-config.sh | 82 ++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 93 insertions(+), 5 deletions(-) create mode 100755 t/t0012-reload-empty-config.sh diff --git a/lib/unicorn/configurator.rb b/lib/unicorn/configurator.rb index a927ed8..378a130 100644 --- a/lib/unicorn/configurator.rb +++ b/lib/unicorn/configurator.rb @@ -40,21 +40,24 @@ module Unicorn def initialize(defaults = {}) #:nodoc: self.set = Hash.new(:unset) - use_defaults = defaults.delete(:use_defaults) + @use_defaults = defaults.delete(:use_defaults) self.config_file = defaults.delete(:config_file) # after_reload is only used by unicorn_rails, unsupported otherwise self.after_reload = defaults.delete(:after_reload) - set.merge!(DEFAULTS) if use_defaults + set.merge!(DEFAULTS) if @use_defaults defaults.each { |key, value| self.__send__(key, value) } Hash === set[:listener_opts] or set[:listener_opts] = Hash.new { |hash,key| hash[key] = {} } Array === set[:listeners] or set[:listeners] = [] - reload + reload(false) end - def reload #:nodoc: + def reload(merge_defaults = true) #:nodoc: + if merge_defaults && @use_defaults + set.merge!(DEFAULTS) if @use_defaults + end instance_eval(File.read(config_file), config_file) if config_file parse_rackup_file @@ -361,7 +364,10 @@ module Unicorn # sets the working directory for Unicorn. This ensures SIGUSR2 will # start a new instance of Unicorn in this directory. This may be - # a symlink, a common scenario for Capistrano users. + # a symlink, a common scenario for Capistrano users. Unlike + # all other Unicorn configuration directives, this binds immediately + # for error checking and cannot be undone by unsetting it in the + # configuration file and reloading. def working_directory(path) # just let chdir raise errors path = File.expand_path(path) diff --git a/t/t0012-reload-empty-config.sh b/t/t0012-reload-empty-config.sh new file mode 100755 index 0000000..c18c030 --- /dev/null +++ b/t/t0012-reload-empty-config.sh @@ -0,0 +1,82 @@ +#!/bin/sh +. ./test-lib.sh +t_plan 9 "reloading unset config resets defaults" + +t_begin "setup and start" && { + unicorn_setup + rtmpfiles unicorn_config_orig before_reload after_reload + cat $unicorn_config > $unicorn_config_orig + cat >> $unicorn_config < $tmp +} + +t_begin "replace config file with original(-ish)" && { + grep -v ^pid < $unicorn_config_orig > $unicorn_config + cat >> $unicorn_config </dev/null + do + sleep 1 + done + + grep 'done reloading' $r_err >/dev/null +} + +t_begin "ensure worker is started" && { + curl -sSf http://$listen/ > $tmp +} + +t_begin "pid file no longer exists" && { + if test -f $pid + then + die "pid=$pid should not exist" + fi +} + +t_begin "killing succeeds" && { + kill $unicorn_pid +} + +t_begin "check stderr" && { + check_stderr +} + +t_begin "ensure reloading restored settings" && { + awk < $after_reload -F'|' ' +$1 != "before_fork" && $2 != $3 { print $0; exit(1) } +' +} + +t_done -- cgit v1.2.3-24-ge0c7