环境:
ubuntu server 10.04 + rvm + ruby 1.9.2 + rails 3.0.3
本来自启动是很简单的操作,可是今天为了让服务器上一个应用自启动,应用服务器用的是 thin, thin怎么也不能自启动,估计是rvm的问题,最后使用rvm wrapper解决了.
我以普通用户登录的ubuntu
安装 thin init.d 脚本
>sudo thin install
如果提示sudo thin 找不到 ,可以使用:
>rvmsudo thin install
这句命令实质在 /etc/init.d 下新建了 thin脚本
#!/bin/sh ### BEGIN INIT INFO # Provides: thin # Required-Start: $local_fs $remote_fs # Required-Stop: $local_fs $remote_fs # Default-Start: 2 3 4 5 # Default-Stop: S 0 1 6 # Short-Description: thin initscript # Description: thin ### END INIT INFO # Original author: Forrest Robertson # Do NOT "set -e" # DAEMON=/usr/local/rvm/gems/ruby-1.9.2-p318/bin/thin DAEMON=/usr/local/rvm/bin/bootup_thin SCRIPT_NAME=/etc/init.d/thin CONFIG_PATH=/etc/thin # Exit if the package is not installed [ -x "$DAEMON" ] || exit 0 case "$1" in start) $DAEMON start --all $CONFIG_PATH ;; stop) $DAEMON stop --all $CONFIG_PATH ;; restart) $DAEMON restart --all $CONFIG_PATH ;; *) echo "Usage: $SCRIPT_NAME {start|stop|restart}" >&2 exit 3 ;; esac :
设置 thin 开机自启动
>sudo update-rc.d -f thin defaults
然后把你的thin.yml配置文件放到 /etc/thin/ 下,看下我的:
wxianfeng@SNDA-192-168-2-15:~$ cat /etc/thin/thin.yml --- chdir: /data/projects/project_manager environment: production address: 0.0.0.0 port: 3000 timeout: 30 log: log/thin.log pid: tmp/pids/thin.pid max_conns: 1024 max_persistent_conns: 512 require: [] wait: 30 servers: 6 daemonize: true onebyone: true
就这么简单,发现 reboot 后,thin启动不起来 , 百思不得其解,
使用
>/etc/init.d/thin start
可以顺利启动
>service start thin
启动不了,提示找不到 thin,所以问题可能在这
于是google,最后找到了 rvm wrapper
我为我的项目使用了 rvm gemset , gemset名字为 huaianxinao
使用gemset
rvm @huaianxinao
生成wrapper
rvm wrapper ruby-1.9.3-p318@huaianxinao bootup thin
可以看到是生成了一个这样的可执行文件
/usr/local/rvm/bin/bootup_thin
于是把你的thin init.d 脚本的DAEMON修改为该路径
测试
>service thin start
可以顺利启动,
>sudo reboot
thin 顺利自启动, OH YEAR O_O
最后看下 wrapper 原理,ls -l 看下 bootup_thin文件:
wxianfeng@SNDA-192-168-2-15:~$ ls -l /usr/local/rvm/bin/bootup_thin lrwxrwxrwx 1 wxianfeng rvm 56 2012-04-21 17:29 /usr/local/rvm/bin/bootup_thin -> /usr/local/rvm/wrappers/ruby-1.9.2-p318@huaianxinao/thin
发现从gemset那ln -s 过来的,继续cat看下
wxianfeng@SNDA-192-168-2-15:~$ cat /usr/local/rvm/wrappers/ruby-1.9.2-p318@huaianxinao/thin #!/usr/bin/env bash if [[ -s "/usr/local/rvm/environments/ruby-1.9.2-p318@huaianxinao" ]] then source "/usr/local/rvm/environments/ruby-1.9.2-p318@huaianxinao" exec thin "$@" else echo "ERROR: Missing RVM environment file: '/usr/local/rvm/environments/ruby-1.9.2-p318@huaianxinao'" >&2 exit 1 fi
发现source了这个文件 /usr/local/rvm/environments/ruby-1.9.2-p318@huaianxinao,cat 看下
wxianfeng@SNDA-192-168-2-15:~$ cat /usr/local/rvm/environments/ruby-1.9.2-p318@huaianxinao export PATH ; PATH="/usr/local/rvm/gems/ruby-1.9.2-p318@huaianxinao/bin:/usr/local/rvm/gems/ruby-1.9.2-p318@global/bin:/usr/local/rvm/rubies/ruby-1.9.2-p318/bin:/usr/local/rvm/bin:$PATH" export rvm_env_string ; rvm_env_string='ruby-1.9.2-p318@huaianxinao' export rvm_path ; rvm_path='/usr/local/rvm' export rvm_ruby_string ; rvm_ruby_string='ruby-1.9.2-p318' export rvm_gemset_name ; rvm_gemset_name='huaianxinao' export RUBY_VERSION ; RUBY_VERSION='ruby-1.9.2-p318' export GEM_HOME ; GEM_HOME='/usr/local/rvm/gems/ruby-1.9.2-p318@huaianxinao' export GEM_PATH ; GEM_PATH='/usr/local/rvm/gems/ruby-1.9.2-p318@huaianxinao:/usr/local/rvm/gems/ruby-1.9.2-p318@global' export MY_RUBY_HOME ; MY_RUBY_HOME='/usr/local/rvm/rubies/ruby-1.9.2-p318' export IRBRC ; IRBRC='/usr/local/rvm/rubies/ruby-1.9.2-p318/.irbrc' unset MAGLEV_HOME unset RBXOPT
发现设置了很多环境变量,所以找到问题了,你必须设置 rvm一系列的环境变量,才可以搞定!
SEE:
http://beginrescueend.com/integration/init-d/
http://stackoverflow.com/questions/3230404/rvm-and-thin-root-vs-local-user
以前部署app还是用很老套的ssh客户端,windows我使用 SSH Secure Shell Client , linux 上使用 Filezilla , 传代码非常之慢,今天体验下rails capistrano 自动部署 , 确实挺方便的 , 实现了ssh远程登录操作的一系列操作 , capistrano 的实现原理是 , 本地客户端 执行命令,实质进行了一系列远程操作,部署同步的根本是check 版本控制系统的最新版,版本控制系统可以是subversion,git等 , 我用的是 subversion,步骤:
我的环境;nginx + thin + capistrano 2.5.19 (客户端) + centos 5.5 + subversion
1,安装 capistrano
gem install capistrano
2,rails app加入 capistrano 配置文件
capify .
3,配置 config 下的deploy.rb ,最核心的东西都在这个文件里
set :application, "blog.wxianfeng.com" # 工程名, 随便写了 set :repository, "svn://173.230.155.150/blog.wxianfeng.com" # svn 地址 set :scm, :subversion # 使用 suversion版本控制 # Or: `accurev`, `bzr`, `cvs`, `darcs`, `git`, `mercurial`, `perforce`, `subversion` or `none` set :scm_username, 'wxianfeng' # svn 用户名 set :scm_password, '123456' #svn 密码 set :checkout, "export" # 使用export方式, svn版本信息不需要 track下来 role :web, "173.230.155.150" # 这三个填你服务器的ip地址 role :app, "173.230.155.150" # role :db, "173.230.155.150", :primary => true # # role :db, "your slave db-server here" set :deploy_to, "/usr/local/system/www/blog.wxianfeng.com" # 部署到remote 服务器路径 set :deploy_via, :remote_cache # 通过 remote_cache 方式部署 ,还有一种本地copy的方式 set :user, "root" # remote 服务器的用户名 set :runner, "root" # 同上 set :password, "123456" # 服务器的 root 密码 set :rake, "/usr/bin/rake" # 服务器上rake 命令path default_run_options[:shell] = false default_run_options[:pty] = true set :use_sudo, true # 允许使用 sudo namespace :deploy do task :start , :roles => :app do invoke_command "cd .." # invoke_command 是连到 服务器上执行的命令 invoke_command "thin start -C /etc/thin/thin.yml" end task :stop ,:roles => :app do invoke_command "killall thin" # or "thin stop -C /etc/thin/thin.yml" end task :restart, :roles => :app, :except => { :no_release => true } do #run "#{try_sudo} touch #{File.join(current_path,'tmp','restart.txt')}" invoke_command "cd .." invoke_command "thin restart -C /etc/thin/thin.yml -O" # -O 是onebyone(关一个,启一个) 启动thin ,注意是 大写 end desc "Symlink shared configs and folders on each release." task :symlink_shared do run "rm -rf #{release_path}/public/files" # 创建 软连接, 保证服务器上 图片的 同步 run "ln -nfs #{shared_path}/public/files #{release_path}/public/files" # or current_path run 等同于 invoke_command end end after 'deploy:update_code', 'deploy:symlink_shared' # 在 deploy:update_code 执行后 callback deploy:symlink_shared
以上当你执行 cap deploy 后 大体执行过程是 先 update_code 然后 deploy:symlink_shared 最后 deploy:restart 服务器 , 上面有一个thin的配置 /etc/thin/thin.yml,如果没有的话, 可以到服务器上生成:
thin config -C /etc/thin/thin.yml -c /usr/local/system/www/blog.wxianfeng.com/current -s 3 -e production -p 3000
4,部署
第一次部署需要运行下面的命令:
cap deploy:setup
报错:
sh: sudo: command not found
解决:
>vi /etc/sudoers # Disable "ssh hostname sudo <cmd>", because it will show the password in clear. # You have to run "ssh -t hostname sudo <cmd>". # #Defaults requiretty #将这行注释掉
这个命令会在服务器上配置好相关的目录结构.然后再运行:
cap deploy:check
检查通过
第一次部署用的命令是:
cap deploy:cold
以后部署都可以直接用:
cap deploy
修改 nginx 配置:
root /usr/local/system/www/blog.wxianfeng.com/current/public/;
修改 root 根目录为 cpistrano 的 current_path 的 public 目录
ok , 你以后更新代码 只需要 cap deploy 就ok了 , 前提是你先 commit 代码到 svn 服务器上 , 用capistrano好处还有一个可以 备份代码 , 因为每一次部署 其实就是相当于 track了svn版本里的最新代码
See:
http://gautamrege.wordpress.com/2009/11/10/capistrano-nginx-thin-deployment-on-linode/
http://weekface.info/2010/03/15/rails-git-capistrano
http://rustammamedov.wordpress.com/2009/12/27/capistrano-nginx-mongrel-deployment/