Как настроить приложение rails (redmine) для запуска в качестве сервиса в Windows?
Я использую redmine как диспетчер билетов, и я бы хотел настроить его для автоматического запуска при запуске Windows.
Как настроить его для запуска в качестве службы?
-
Просто задал вопрос, чтобы документировать его, я надеюсь, что кто-то может найти его полезным...
Ответы
Ответ 1
1. используя webrick:
ref: http://www.redmine.org/boards/1/topics/4123
-
Загрузите и установите комплект ресурсов Windows NT из
http://www.microsoft.com/downloads/details.aspx?familyid=9d467a69-57ff-4ae7-96ee-b18c4790cffd&displaylang=en
-
Создайте службу, выполнив следующую команду:
path\INSTSRV.EXE your_service_name path\SRVANY.EXE
в моем случае path
:
"C:\Program Files\Windows NT Resource Kit\INSTSRV.EXE" redmine_webrick "C:\Program Files\Windows NT Resource Kit\SRVANY.EXE"
может быть также C:\Program Files\Windows Resource Kits\Tools\
.
-
Запустить regedit (Пуск → Выполнить → regedit)
-
Добавьте следующий раздел реестра, если он еще не существует:
HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\your_service_name
-
Щелкните правой кнопкой мыши этот раздел реестра и выберите New → Key. Назовите его Parameters
.
-
Добавьте два значения в клавишу Parameters
. Щелкните правой кнопкой мыши по параметру, New → String Value. Назовите его Application
. Создайте еще один AppParameters
. Дайте им следующие значения:
- Приложение:
PathToRuby.exe
, например. C:\ruby\bin\Ruby.exe
- AppParameters:
C:\RUBYAPP\script\server -e production
, где RUBYAPP
- это каталог, содержащий веб-сайт redmine.
Пример: C:\redmine\script\server -p 2000 -e production
(-p указывает, что порт webrick будет прослушивать, и -e используется среда)
Теперь вы можете перейти в "Администрирование" → "Службы". Там вы можете начать свою службу (с именем your_service_name
) и проверить, правильно ли она работает. Следует отметить, что служба будет отмечена как запущенная до того, как WEBrick завершит процедуру загрузки. Вы должны дать ему 1 минуту или около того, прежде чем пытаться попасть в службу, чтобы убедиться, что она работает правильно.
2. используя mongrel:
ref: http://mongrel.rubyforge.org/wiki
ref: http://mongrel.rubyforge.org/wiki/Win32
сначала установите mongrel и mongrel_service gem
gem install mongrel
gem install mongrel_service
затем создайте службу
mongrel_rails service::install -N redmine_mongrel -c c:\redmine -p 3000 -e production
3. используя тонкий:
Ссылки:
Инструкция:
-
Сначала установите тонкий (вам нужно установить стойку, если она еще не установлена
установлен)
gem install rack
gem install thin
-
Следуйте тем же шагам, которые указаны для webrick, но добавьте другое значение с именем "AppDirectory". Это необходимо для того, чтобы избежать использования c:\ruby \ bin\thin.bat. Если бы я просто указал на файл bat, я не смог бы остановить службу.
В HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\redmine_thin\Parameters
добавьте следующие клавиши:
Приложение: c:\ruby \ bin\ruby.exe
AppDirectory: c:\redmine
AppParameters: c:\ruby \ bin\thin start -p 4000 -e production
----------------------------------------------- -------------------------------------------
Вы можете управлять любым сервисом со следующими командами:
net start redmine_xxx
net stop redmine_xxx
sc config redmine_xxx start = auto
sc config redmine_xxx start = автозависимость = MySql
sc удалить redmine_xxx
Ответ 2
Для приложений Rails 3.0.x(проверено на версии 3.0.10 и Windows 7)
demo_daemon_ctl.rb
############################################################################
# demo_daemon_ctl.rb
#
# This is a command line script for installing and/or running a small
# Ruby program as a service. The service will simply write a small bit
# of text to a file every 20 seconds. It will also write some text to the
# file during the initialization (service_init) step.
#
# It should take about 10 seconds to start, which is intentional - it a test
# of the service_init hook, so don't be surprised if you see "one moment,
# start pending" about 10 times on the command line.
#
# The file in question is C:\test.log. Feel free to delete it when finished.
#
# To run the service, you must install it first.
#
# Usage: ruby demo_daemon_ctl.rb <option>
#
# Note that you *must* pass this program an option
#
# Options:
# install - Installs the service. The service name is "DemoSvc"
# and the display name is "Demo".
# start - Starts the service. Make sure you stop it at some point or
# you will eventually fill up your filesystem!.
# stop - Stops the service.
# pause - Pauses the service.
# resume - Resumes the service.
# uninstall - Uninstalls the service.
# delete - Same as uninstall.
#
# You can also used the Windows Services GUI to start and stop the service.
#
# To get to the Windows Services GUI just follow:
# Start -> Control Panel -> Administrative Tools -> Services
############################################################################
require 'win32/service'
require 'rbconfig'
include Win32
include Config
# Make sure you're using the version you think you're using.
puts 'VERSION: ' + Service::VERSION
SERVICE_NAME = 'DemoSvc'
SERVICE_DISPLAYNAME = 'Demo'
# Quote the full path to deal with possible spaces in the path name.
ruby = File.join(CONFIG['bindir'], 'ruby').tr('/', '\\')
path = ' "' + File.dirname(File.expand_path($0)).tr('/', '\\')
path += '\demo_daemon.rb"'
cmd = ruby + path
# You must provide at least one argument.
raise ArgumentError, 'No argument provided' unless ARGV[0]
case ARGV[0].downcase
when 'install'
Service.new(
:service_name => SERVICE_NAME,
:display_name => SERVICE_DISPLAYNAME,
:description => 'Sample Ruby service',
:binary_path_name => cmd
)
puts 'Service ' + SERVICE_NAME + ' installed'
when 'start'
if Service.status(SERVICE_NAME).current_state != 'running'
Service.start(SERVICE_NAME, nil, 'hello', 'world')
while Service.status(SERVICE_NAME).current_state != 'running'
puts 'One moment...' + Service.status(SERVICE_NAME).current_state
sleep 1
end
puts 'Service ' + SERVICE_NAME + ' started'
else
puts 'Already running'
end
when 'stop'
if Service.status(SERVICE_NAME).current_state != 'stopped'
Service.stop(SERVICE_NAME)
while Service.status(SERVICE_NAME).current_state != 'stopped'
puts 'One moment...' + Service.status(SERVICE_NAME).current_state
sleep 1
end
puts 'Service ' + SERVICE_NAME + ' stopped'
else
puts 'Already stopped'
end
when 'uninstall', 'delete'
if Service.status(SERVICE_NAME).current_state != 'stopped'
Service.stop(SERVICE_NAME)
end
while Service.status(SERVICE_NAME).current_state != 'stopped'
puts 'One moment...' + Service.status(SERVICE_NAME).current_state
sleep 1
end
Service.delete(SERVICE_NAME)
puts 'Service ' + SERVICE_NAME + ' deleted'
when 'pause'
if Service.status(SERVICE_NAME).current_state != 'paused'
Service.pause(SERVICE_NAME)
while Service.status(SERVICE_NAME).current_state != 'paused'
puts 'One moment...' + Service.status(SERVICE_NAME).current_state
sleep 1
end
puts 'Service ' + SERVICE_NAME + ' paused'
else
puts 'Already paused'
end
when 'resume'
if Service.status(SERVICE_NAME).current_state != 'running'
Service.resume(SERVICE_NAME)
while Service.status(SERVICE_NAME).current_state != 'running'
puts 'One moment...' + Service.status(SERVICE_NAME).current_state
sleep 1
end
puts 'Service ' + SERVICE_NAME + ' resumed'
else
puts 'Already running'
end
else
raise ArgumentError, 'unknown option: ' + ARGV[0]
end
demo_daemon.rb
APP_ROOT_CUSTOM = 'your app root dir'
LOG_FILE = APP_ROOT_CUSTOM + 'log/win32_daemon_test.log'
APP_PATH = File.expand_path( APP_ROOT_CUSTOM + 'config/application', APP_ROOT_CUSTOM + 'script/rails')
begin
require 'rubygems'
require 'win32/daemon'
include Win32
require File.expand_path( APP_ROOT_CUSTOM + 'config/boot', APP_ROOT_CUSTOM + 'script/rails')
require 'rails/commands/server'
module ::Rails
class Server
def default_options
super.merge({
:Port => 3000,
:environment => (ENV['RAILS_ENV'] || "development").dup,
:daemonize => false,
:debugger => false,
:pid => File.expand_path( APP_ROOT_CUSTOM + "tmp/pids/server.pid" ),
:config => File.expand_path( APP_ROOT_CUSTOM + "config.ru" )
})
end
end
end
class DemoDaemon < Daemon
# This method fires off before the +service_main+ mainloop is entered.
# Any pre-setup code you need to run before your service mainloop
# starts should be put here. Otherwise the service might fail with a
# timeout error when you try to start it.
#
def service_init
end
# This is the daemon mainloop. In other words, whatever runs here
# is the code that runs while your service is running. Note that the
# loop is not implicit.
#
# You must setup a loop as I've done here with the 'while running?'
# code, or setup your own loop. Otherwise your service will exit and
# won't be especially useful.
#
# In this particular case, I've setup a loop to append a short message
# and timestamp to a file on your C: drive every 20 seconds. Be sure
# to stop the service when you're done!
#
def service_main(*args)
Rails::Server.new.tap { |server|
require APP_PATH
Dir.chdir( APP_ROOT_CUSTOM )
server.start
}
msg = 'application started at: ' + Time.now.to_s
File.open(LOG_FILE, 'a'){ |f|
f.puts msg
f.puts "Args: " + args.join(',')
}
# While we're in here the daemon is running.
while running?
if state == RUNNING
sleep 20
msg = 'Service is running as of: ' + Time.now.to_s
File.open(LOG_FILE, 'a'){ |f| f.puts msg }
else # PAUSED or IDLE
sleep 0.5
end
end
# We've left the loop, the daemon is about to exit.
File.open(LOG_FILE, 'a'){ |f| f.puts "STATE: #{state}" }
msg = 'service_main left at: ' + Time.now.to_s
File.open(LOG_FILE, 'a'){ |f| f.puts msg }
end
# This event triggers when the service receives a signal to stop. I've
# added an explicit "exit!" here to ensure that the Ruby interpreter exits
# properly. I use 'exit!' instead of 'exit' because otherwise Ruby will
# raise a SystemExitError, which I don't want.
#
def service_stop
msg = 'Received stop signal at: ' + Time.now.to_s
File.open(LOG_FILE, 'a'){ |f| f.puts msg }
exit!
end
# This event triggers when the service receives a signal to pause.
#
def service_pause
msg = 'Received pause signal at: ' + Time.now.to_s
File.open(LOG_FILE, 'a'){ |f| f.puts msg }
end
# This event triggers when the service receives a signal to resume
# from a paused state.
#
def service_resume
msg = 'Received resume signal at: ' + Time.now.to_s
File.open(LOG_FILE, 'a'){ |f| f.puts msg }
end
end
# Create an instance of the Daemon and put it into a loop. I borrowed the
# method name 'mainloop' from Tk, btw.
#
DemoDaemon.mainloop
rescue Exception => err
File.open(LOG_FILE, 'a'){ |fh| fh.puts 'Daemon failure: ' + err }
raise
end
поместите оба файла в один и тот же каталог и запустите
ruby demo_daemon_ctl.rb install
Ответ 3
использовать http://www.firedaemon.com/
Ответ 4
Некоторое время назад я также попытался установить Redmine на Windows. Но мне не удалось заставить его работать, возможно, из-за отсутствия знаний Rails.
Затем я обнаружил Bitnami Redmine Stack. У них есть установщик Windows, который устанавливает Redmine со всеми необходимыми зависимостями, , и он просто работает.
Ответ 5
Для приложений Rails 4.0.x, предложенных Богданом,
мы должны заменить
CONFIG ['bindir'] с RbConfig:: CONFIG ['bindir']
Remmber:
gem install win32-service
Ответ 6
-
gem install win32-service
- оставьте ниже кода Ruby в файле service.rb и обновите путь REDMINE_DIR, чтобы он соответствовал вашей установке Redmine.
- создайте службу, например, с помощью
sc create redmine binPath= "C:\Ruby23-x64\bin\rubyw -C E:\www\redmine-3.3.2\ service.rb"
, где E:\www\redmine-3.3.2\
- это путь к каталогу, в котором находится файл service.rb, и C:\Ruby23-x64\bin\rubyw
ваш путь установки Ruby
начать требуется "win32/daemon" включить Win32
class RedmineService < Daemon
def service_init
File.open(LOG_FILE, 'a'){ |f| f.puts "Initializing service #{Time.now}" }
#@server_pid = Process.spawn 'ruby script/rails s -e production', :chdir => REDMINE_DIR, :err => [LOG_FILE, 'a']
# use full path
@server_pid = Process.spawn 'C:\Ruby23-x64\bin\ruby E:\www\redmine-3.3.2\bin\rails s -e production -p 3000', :chdir => REDMINE_DIR, :err => [LOG_FILE, 'a']
end
def service_main
File.open(LOG_FILE, 'a'){ |f| f.puts "Service is running #{Time.now} with pid #{@server_pid}" }
while running?
sleep 10
end
end
def service_stop
File.open(LOG_FILE, 'a'){ |f| f.puts "Stopping server thread #{Time.now}" }
system "taskkill /PID #{@server_pid} /T /F"
Process.waitall
File.open(LOG_FILE, 'a'){ |f| f.puts "Service stopped #{Time.now}" }
exit!
end
end
RedmineService.mainloop
rescue Exception => e
File.open(LOG_FILE,'a+'){ |f| f.puts " ***Daemon failure #{Time.now} exception=#{e.inspect}\n#{e.backtrace.join($/)}" }
raise
end
- Обратите внимание, что Process.spawn в service.rb использует полный путь.