2 # You can have redis on a different box
3 RAILS_ENV: 'production'
6 # this gives us very good cache coverage, 96 -> 99
7 # in practice it is 1-2% perf improvement
8 RUBY_GLOBAL_METHOD_CACHE_SIZE: 131072
9 # stop heap doubling in size so aggressively, this conserves memory
10 RUBY_GC_HEAP_GROWTH_MAX_SLOTS: 40000
11 RUBY_GC_HEAP_INIT_SLOTS: 400000
12 RUBY_GC_HEAP_OLDOBJECT_LIMIT_FACTOR: 1.5
14 DISCOURSE_DB_SOCKET: /var/run/postgresql
20 # SSH key is required for remote access into the container
23 home: /var/www/discourse
27 - exec: thpoff echo "thpoff is installed!"
28 - exec: /usr/local/bin/ruby -e 'if ENV["DISCOURSE_SMTP_ADDRESS"] == "smtp.example.com"; puts "Aborting! Mail is not configured!"; exit 1; end'
29 - exec: /usr/local/bin/ruby -e 'if ENV["DISCOURSE_HOSTNAME"] == "discourse.example.com"; puts "Aborting! Domain is not configured!"; exit 1; end'
30 - exec: chown -R discourse /home/discourse
31 # TODO: move to base image (anacron can not be fired up using rc.d)
32 - exec: rm -f /etc/cron.d/anacron
34 path: /etc/cron.d/anacron
37 PATH=/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin
39 30 7 * * * root /usr/sbin/anacron -s >/dev/null
41 path: /etc/runit/1.d/copy-env
46 conf=/var/www/discourse/config/discourse.conf
48 # find DISCOURSE_ env vars, strip the leader, lowercase the key
49 /usr/local/bin/ruby -e 'ENV.each{|k,v| puts "#{$1.downcase} = '\''#{v}'\''" if k =~ /^DISCOURSE_(.*)/}' > $conf
52 path: /etc/runit/1.d/enable-brotli
56 [ ! -z "$COMPRESS_BROTLI" ] && sed -i "s/. brotli/ brotli/" /etc/nginx/conf.d/discourse.conf || sed -i "s/. brotli/# brotli/" /etc/nginx/conf.d/discourse.conf
59 path: /etc/service/unicorn/run
67 chown -R discourse:www-data /shared/log/rails
68 LD_PRELOAD=$RUBY_ALLOCATOR HOME=/home/discourse USER=discourse exec thpoff chpst -u discourse:www-data -U discourse:www-data bundle exec config/unicorn_launcher -E production -c config/unicorn.conf.rb
71 path: /etc/service/nginx/run
79 path: /etc/runit/3.d/01-nginx
86 path: /etc/runit/3.d/02-unicorn
98 - git remote set-branches --add origin master
100 - git fetch origin $version
101 - git checkout $version
103 - mkdir -p tmp/sockets
105 - mkdir -p /shared/log/rails
106 - bash -c "touch -a /shared/log/rails/{production,production_errors,unicorn.stdout,unicorn.stderr}.log"
107 - bash -c "ln -s /shared/log/rails/{production,production_errors,unicorn.stdout,unicorn.stderr}.log $home/log"
108 - bash -c "mkdir -p /shared/{uploads,backups}"
109 - bash -c "ln -s /shared/{uploads,backups} $home/public"
110 - chown -R discourse:www-data /shared/log/rails /shared/uploads /shared/backups
114 - "cp $home/config/nginx.sample.conf /etc/nginx/conf.d/discourse.conf"
115 - "rm /etc/nginx/sites-enabled/default"
116 - "mkdir -p /var/nginx/cache"
119 filename: /etc/nginx/nginx.conf
120 from: pid /run/nginx.pid;
124 filename: "/etc/nginx/conf.d/discourse.conf"
125 from: /upstream[^\}]+\}/m
126 to: "upstream discourse {
127 server 127.0.0.1:3000;
131 filename: "/etc/nginx/conf.d/discourse.conf"
132 from: /server_name.+$/
136 filename: "/etc/nginx/conf.d/discourse.conf"
137 from: /client_max_body_size.+$/
138 to: client_max_body_size $upload_size ;
141 cmd: echo "done configuring web"
148 # ensure we are on latest bundler
150 - find $home ! -user discourse -exec chown discourse {} \+
156 - su discourse -c 'bundle install --deployment --verbose --without test --without development --retry 3 --jobs 4'
157 - su discourse -c 'bundle exec rake db:migrate'
158 - su discourse -c 'bundle exec rake assets:precompile'
161 path: /usr/local/bin/discourse
165 (cd /var/www/discourse && RAILS_ENV=production sudo -H -E -u discourse bundle exec script/discourse "$@")
168 path: /usr/local/bin/rails
172 # If they requested a console, load pry instead
173 if [ "$*" == "c" -o "$*" == "console" ]
175 (cd /var/www/discourse && RAILS_ENV=production sudo -H -E -u discourse bundle exec pry -r ./config/environment)
177 (cd /var/www/discourse && RAILS_ENV=production sudo -H -E -u discourse bundle exec script/rails "$@")
181 path: /usr/local/bin/rake
185 (cd /var/www/discourse && RAILS_ENV=production sudo -H -E -u discourse bundle exec bin/rake "$@")
188 path: /usr/local/bin/rbtrace
192 (cd /var/www/discourse && RAILS_ENV=production sudo -H -E -u discourse bundle exec rbtrace "$@")
195 path: /usr/local/bin/stackprof
199 (cd /var/www/discourse && RAILS_ENV=production sudo -H -E -u discourse bundle exec stackprof "$@")
202 path: /etc/update-motd.d/10-web
207 echo Use: rails, rake or discourse to execute commands in production
211 path: /etc/logrotate.d/rails
213 /shared/log/rails/*.log
227 path: /etc/logrotate.d/nginx
229 /var/log/nginx/*.log {
235 create 0644 www-data www-data
242 # move state out of the container this fancy is done to support rapid rebuilds of containers,
243 # we store anacron and logrotate state outside the container to ensure its maintained across builds
244 # later move this snipped into an intialization script
245 # we also ensure all the symlinks we need to /shared are in place in the correct structure
246 # this allows us to bootstrap on one machine and then run on another
248 path: /etc/runit/1.d/00-ensure-links
252 if [[ ! -L /var/lib/logrotate ]]; then
253 rm -fr /var/lib/logrotate
254 mkdir -p /shared/state/logrotate
255 ln -s /shared/state/logrotate /var/lib/logrotate
257 if [[ ! -L /var/spool/anacron ]]; then
258 rm -fr /var/spool/anacron
259 mkdir -p /shared/state/anacron-spool
260 ln -s /shared/state/anacron-spool /var/spool/anacron
262 if [[ ! -d /shared/log/rails ]]; then
263 mkdir -p /shared/log/rails
264 chown -R discourse:www-data /shared/log/rails
266 if [[ ! -d /shared/uploads ]]; then
267 mkdir -p /shared/uploads
268 chown -R discourse:www-data /shared/uploads
270 if [[ ! -d /shared/backups ]]; then
271 mkdir -p /shared/backups
272 chown -R discourse:www-data /shared/backups
275 # change login directory to Discourse home
277 path: /root/.bash_profile