Starting or restarting Unicorn with Capistrano 3.x

前端 未结 5 1019
爱一瞬间的悲伤
爱一瞬间的悲伤 2021-02-01 09:31

I\'m trying to start or restart Unicorn when I do cap production deploy with Capistrano 3.0.1. I have some examples that I got working with Capistrano 2.x using so

相关标签:
5条回答
  • 2021-02-01 09:40

    I'm using following code:

    namespace :unicorn do
      desc 'Stop Unicorn'
      task :stop do
        on roles(:app) do
          if test("[ -f #{fetch(:unicorn_pid)} ]")
            execute :kill, capture(:cat, fetch(:unicorn_pid))
          end
        end
      end
    
      desc 'Start Unicorn'
      task :start do
        on roles(:app) do
          within current_path do
            with rails_env: fetch(:rails_env) do
              execute :bundle, "exec unicorn -c #{fetch(:unicorn_config)} -D"
            end
          end
        end
      end
    
      desc 'Reload Unicorn without killing master process'
      task :reload do
        on roles(:app) do
          if test("[ -f #{fetch(:unicorn_pid)} ]")
            execute :kill, '-s USR2', capture(:cat, fetch(:unicorn_pid))
          else
            error 'Unicorn process not running'
          end
        end
      end
    
      desc 'Restart Unicorn'
      task :restart
      before :restart, :stop
      before :restart, :start
    end
    
    0 讨论(0)
  • 2021-02-01 09:41

    You can try to use native capistrano way as written here:

    If preload_app:true and you need capistrano to cleanup your oldbin pid use:

    after 'deploy:publishing', 'deploy:restart'
    namespace :deploy do
      task :restart do
        invoke 'unicorn:legacy_restart'
      end
    end
    
    0 讨论(0)
  • 2021-02-01 09:44

    I'm just going to throw this in the ring: capistrano 3 unicorn gem

    However, my issue with the gem (and any approach NOT using an init.d script), is that you may now have two methods of managing your unicorn process. One with this cap task and one with init.d scripts. Things like Monit / God will get confused and you may spend hours debugging why you have two unicorn processes trying to start, and then you may start to hate life.

    Currently I'm using the following with capistrano 3 and unicorn:

      namespace :unicorn do
      desc 'Restart application'
        task :restart do
          on roles(:app) do
            puts "restarting unicorn..."
            execute "sudo /etc/init.d/unicorn_#{fetch(:application)} restart"
            sleep 5
            puts "whats running now, eh unicorn?"
            execute "ps aux | grep unicorn"
          end
        end
    end
    

    The above is combined with the preload_app: true and the before_fork and after_fork statements mentioned by @dredozubov

    Note I've named my init.d/unicorn script unicorn_application_name.

    The new worker that is started should kill off the old one. You can see with ps aux | grep unicorn that the old master hangs around for a few seconds before it disappears.

    0 讨论(0)
  • 2021-02-01 09:44

    To view all caps:

    cap -T
    

    and it shows:

    ***
    cap unicorn:add_worker             # Add a worker (TTIN)
    cap unicorn:duplicate              # Duplicate Unicorn; alias of unicorn:re...
    cap unicorn:legacy_restart         # Legacy Restart (USR2 + QUIT); use this...
    cap unicorn:reload                 # Reload Unicorn (HUP); use this when pr...
    cap unicorn:remove_worker          # Remove a worker (TTOU)
    cap unicorn:restart                # Restart Unicorn (USR2); use this when ...
    cap unicorn:start                  # Start Unicorn
    cap unicorn:stop                   # Stop Unicorn (QUIT)
    ***
    

    So, to start unicorn in production:

    cap production unicorn:start
    

    and restart:

    cap production unicorn:restart 
    

    PS do not forget to correct use gem capistrano3-unicorn

    https://github.com/tablexi/capistrano3-unicorn

    0 讨论(0)
  • 2021-02-01 10:02

    Can't say anything specific about capistrano 3(i use 2), but i think this may help: How to run shell commands on server in Capistrano v3?. Also i can share some unicorn-related experience, hope this helps.

    I assume you want 24/7 graceful restart approach.

    Let's consult unicorn documentation for this matter. For graceful restart(without downtime) you can use two strategies:

    1. kill -HUP unicorn_master_pid It requires your app to have 'preload_app' directive disabled, increasing starting time of every one of unicorn workers. If you can live with that - go on, it's your call.

    2. kill -USR2 unicorn_master_pid kill -QUIT unicorn_master_pid

    More sophisticated approach, when you're already dealing with performance concerns. Basically it will reexecute unicorn master process, then you should kill it's predecessor. Theoretically you can deal with usr2-sleep-quit approach. Another(and the right one, i may say) way is to use unicorn before_fork hook, it will be executed, when new master process will be spawned and will try to for new children for itself. You can put something like this in config/unicorn.rb:

    # Where to drop a pidfile
    pid project_home + '/tmp/pids/unicorn.pid'
    
    before_fork do |server, worker|
      server.logger.info("worker=#{worker.nr} spawning in #{Dir.pwd}")
    
      # graceful shutdown.
      old_pid_file = project_home + '/tmp/pids/unicorn.pid.oldbin'
      if File.exists?(old_pid_file) && server.pid != old_pid_file
        begin
          old_pid = File.read(old_pid_file).to_i
          server.logger.info("sending QUIT to #{old_pid}")
          # we're killing old unicorn master right there
          Process.kill("QUIT", old_pid)
        rescue Errno::ENOENT, Errno::ESRCH
          # someone else did our job for us
        end
      end
    end
    

    It's more or less safe to kill old unicorn when the new one is ready to fork workers. You won't get any downtime that way and old unicorn will wait for it's workers to finish.

    And one more thing - you may want to put it under runit or init supervision. That way your capistrano tasks will be as simple as sv reload unicorn, restart unicorn or /etc/init.d/unicorn restart. This is good thing.

    0 讨论(0)
提交回复
热议问题