SSHKit 사용 예제

###다른 계정으로 명령 실행하기

on hosts do |host|
  as 'www-data' do
    puts capture(:whoami)
  end
end

###디폴트 환경 변수로 실행하기

SSHKit.config.default_env = { path: '/usr/local/libexec/bin:$PATH' }
on hosts do |host|
  puts capture(:env)
end

###다른 디렉토리에서 명령 실행하기

on hosts do |host|
  within '/var/log' do
    puts capture(:head, '-n5', 'messages')
  end
end

###특정 환경 변수로 명령 실행하기

on hosts do |host|
  with rack_env: :test do
    puts capture("env | grep RACK_ENV")
  end
end

###로깅 메소드를 이용하여 출력하기

on hosts do |host|
  f = '/some/file'
  if test("[ -d #{f} ]")
    execute :touch, f
  else
    info "#{f} already exists on #{host}!"
  end
end

SSHKit.config.output_vervosity 의 현재 로그 레벨에 따라 debug(), info(), warn(), error(), fatal() 메소드를 사용할 수 있다.

###다른 계정으로 다른 디렉토리에서 명령 실행하기

on hosts do |host|
  as 'www-data' do
    within '/var/log' do
      puts capture(:whoami)
      puts capture(:pwd)
    end
  end
end

이것은 아래와 같은 결과를 보여 준다.

www-data
/var/log

주의사항 : 이 예제는 약간 오해의 소지가 있다. 즉, www-data 계정은 쉘이 정의되어 있지 않아서 이 계정으로 변경할 수 없다.

###디스크로부터 파일 업로그하기

on hosts do |host|
  upload! '/config/database.yml', '/opt/my_project/shared/database.yml'
end

주의사항 : upload!() 메소드는 within(), as() 등의 값을 받지 않는다. 이 문제는 라이브러리가 성숙해 짐에 따라 해결될 것이지만 현재로서는 아직 그대로다.

###스트림으로부터 파일 업로드하기

on hosts do |host|
  file = File.open('/config/database.yml')
  io   = StringIO.new(....)
  upload! file, '/opt/my_project/shared/database.yml'
  upload! io,   '/opt/my_project/shared/io.io.io'
end

IO 스트리밍하는 것은 “cat” 하기 보다는 뭔가를 업로드하는데 유용하다. 예를 들면,

on hosts do |host|
  contents = StringIO.new('ALL ALL = (ALL) NOPASSWD: ALL')
  upload! contents, '/etc/sudoers.d/yolo'
end

이것은 “echo(:cat, ‘…?…’, ‘> /etc/sudoers.d/yolo’)”와 같이 정확한 이스케이핑 순서를 기술해야 하는 수고를 덜어 준다.

주의사항 : upload!() 메소드는 within(), as() 등의 값을 받지 않는다. 이 문제는 라이브러리가 성숙해 짐에 따라 해결될 것이지만 현재로서는 아직 그대로다.

###디렉토리내의 파일을 업로드하기

on hosts do |host|
  upload! '.', '/tmp/mypwd', recursive: true
end

이 경우에 recursive: true 옵션은 Net::{SCP, SFTP}에서 사용가능한 옵션과 같은 것이다.

###전역(global) SSH 옵션 설정하기

이 옵션들은 호스트마다 옵션을 달리 지정할 수 있다.

Netssh.configure do |ssh|
  ssh.connection_timeout = 30
  ssh.ssh_options = {
    keys: %w(/home/user/.ssh/id_rsa),
    forward_agent: false,
    auth_methods: %w(publickey password)
  }
end

###다른 group ID로 명령 실행하기

on hosts do |host|
  as user: 'www-data', group: 'project-group' do
    within '/var/log' do
      execute :touch, 'somefile'
      execute :ls, '-l'
    end
  end
end

위의 설정에서, 생성된 파일은 www-data 계정과 project-group 그룹이 소유하게 된다.

umask 구성 옵션과 함께 사용하면, 로그인을 공유하지 않은 상태에서 팀 멤버간에 배포 스크립트를 쉽게 공유할 수 있다.

###중첩된 디렉토리에서 작성하기

on hosts do
  within "/var" do
    puts capture(:pwd)
    within :log do
      puts capture(:pwd)
    end
  end
end

이것은 아래와 같이 출력된다.

/var/
/var/log

디렉토리 경로는 File.join()를 이용해서 조합된다. 이 메소드는 경로 앞뒤에 오는 슬래시에 신경쓰지 않고도 각 부분을 정확하게 연결해서 조합해 준다. 이것은 File.join()이 코드를 실행하는 머신에서 동작하기 때문에 오해의 여지가 있다. 만약 윈도우즈 운영환경에서 실행된다면 명령을 받는 머신에 따라 경로가 부정확하게 조합될 수 있다.

###백그라운드에서 task 수행하기

on hosts do
  within '/opt/sites/example.com' do
    background :rails, :server
  end
end

이것은 nohup /usr/bin/env rails server > /dev/null &와 같이 실행하는데, 레일스 프로세스를 백그라운드에서 실행하도록 하여 파일 시스템을 nohup 로그 파일로 더럽히지 않게 된다.

[역주] nohup - 리눅스, 유닉스에서 쉘스크립트파일(*.sh)을 데몬형태로 실행시키는 프로그램

주의사항 : background() 메소드에 sleep 5 문자열을 넘겨 주면 제대로 동작하지 않을 것이다. 이것은 명령을 처리하는 규칙에 따른 것이어서, background(:sleep, "5")와 같이(즉, sleep이 명령이고, 5는 인수) 호출해야 한다.

더 주의해야 할 사항 : background() task가 어떤 상황에서 특정 명령을 nohup ... &로 감싸면, SSH 세션이 종료되어도 프로그램이 멈출 것이다.

###host 블록에 대해서 신경쓰지 않기

on hosts do
  # |host| 블록 인수는 옵션이다.
  # 블록 인수로 넘기지 않으면 블록내에서는 nil 값을 가지게 된다.
end

###모든 출력을 /dev/null 로 리디렉트하기

SSHKit.config.output = File.open('/dev/null')

###매우 간단한 formatter 클래스 구현하기

class MyFormatter < SSHKit::Formatter::Abstract
  def write(obj)
    case obj.is_a? SSHKit::Command
      # Do something here, see the SSHKit::Command documentation
    end
  end
end

SSHKit.config.output = MyFormatter.new($stdout)
SSHKit.config.output = MyFormatter.new(SSHKit.config.output)
SSHKit.config.output = MyFormatter.new(File.open('log/deploy.log', 'wb'))

###특정 호스트에 대한 비밀번호 지정하기

host = SSHKit::Host.new('user@example.com')
host.password = "hackme"

on host do |host|
  puts capture(:echo, "I don't care about security!")
end

###뭔가 잘 못 됐을 때 실행 후 에러 발생하기

on hosts do |host|
  execute!(:echo, '"Example Message!" 1>&2; false')
end

이것은 #message “Example Message!” 와 함께 ‘SSHKit::Command:Failed` 예외를 발생할 것이고 명령은 취소될 것이다.

###에러를 발생하지 않은 채 실패(fail)할 수 있는 테스트하기 또는 명령 실행하기

on hosts do |host|
  if test "[ -d /opt/sites ]"
    within "/opt/sites" do
      execute :git, :pull
    end
  else
    execute :git, :clone, 'some-repository', '/opt/sites'
  end
end

test() 명령는 execute()와 동일하게 동작하지만 0가 아닌 값으로 명령이 종료할 때 false 값을 반환한다(man 1 test가 하듯이). 논리값을 반환하기 때문에 블록내에서 제어 흐름의 방향을 지정할 때 사용될 수 있다.

###호스트 property에 따라 각 호스트마다 다른 작업 하기

host1 = SSHKit::Host.new 'user@example.com'
host2 = SSHKit::Host.new 'user@example.org'

on hosts do |host|
  target = "/var/www/sites/"
  if host.hostname =~ /org/
    target += "dotorg"
  else
    target += "dotcom"
  end
  execute! :git, :clone, "git@git.#{host.hostname}", target
end

###가장 쉽게 호스트에 연결하기

on 'example.com' do |host|
  execute :uptime
end

이것은 example.com 호스트명을 SSHKit::Host 객체로 변환해서 해당 호스트에 대한 정확한 구성 상태를 가져온다.

###명령 맵핑 없이 명령 실행하기

호출하려는 명령이 스페이스 문자를 포함할 경우 맵핑되지 않을 것이다.

Command.new(:git, :push, :origin, :master).to_s
# => /usr/bin/env git push origin master
# (also: execute(:git, :push, :origin, :master)

Command.new("git push origin master").to_s
# => git push origin master
# (also: execute("git push origin master"))

이것은 ( iftest와 같은) 쉘 내장 명령을 접근할 때 사용할 수 있다.

###heredoc 와 함께 명령 실행하기

위의 기능을 확장한 것으로 아래와 같이 명령을 작성할 수 있다.

c = Command.new <<-EOCOMMAND
  if test -d /var/log
  then echo "Directory Exists"
  fi
EOCOMMAND
c.to_s
# => if test -d /var/log; then echo "Directory Exists; fi
# (also: execute <<- EOCOMMAND........))

주의사항 : 스크립트를 한줄로 표시하는 로직은 세련되어 보이지 않지만 모든 테스트에서 제대로 동작한다. 핵심 사항은 if/usr/bin/env if로 매핑되지 않는다는 것인데, 이것은 문법상의 오류를 유발하지 않는다.

###Rake 사용하기

Rakefile에 아래와 같은 것을 둘 수 있다.

require 'sshkit/dsl'

SSHKit.config.command_map[:rake] = "./bin/rake"

desc "Deploy the site, pulls from Git, migrate the db and precompile assets, then restart Passenger."
task :deploy do
  on "example.com" do |host|
    within "/opt/sites/example.com" do
      execute :git, :pull
      execute :bundle, :install, '--deployment'
      execute :rake, 'db:migrate'
      execute :rake, 'assets:precompile'
      execute :touch, 'tmp/restart.txt'
    end
  end
end

###DSL 없이 사용하기

Coordinator 가 모든 호스트를 Host 객체로 변환해 줄 것이다.

Coordinator.new("one.example.com", SSHKit::Host.new('two.example.com')).each in: :sequence do
  puts capture :uptime
end

./lib/sshkit/dsl.rb 파일을 찾고할 수 있는 데, 여기에서 위와 거의 똑같은 코드를 볼 수 있지만 on() 메소드에 대해서 구현한 것이다.

###Host properties 속성 이용하기

v0.0.6부터 구현됨

servers = %w{one.example.com two.example.com
             three.example.com four.example.com}.collect do |s|
  h = SSHKit::Host.new(s)
  if s.match /(one|two)/
    h.properties.roles = [:web]
  else
    h.properties.roles = [:app]
  end
end

on servers do |host|
  if host.properties.roles.include?(:web)
    # 웹서버에 관련된 작업을 함.
  elsif host.properties.roles.include?(:app)
    # 어플리케이션 서버에 관련된 작업을 함.
  end
end

SSHKit::Host#propertiesOpenStruct이며 어쨌던 유효성 검증이 되지 않기 때문에 이에 관해서는 개발자에게 달려 있거나 라이브러리가 이 메카니즘에 의미나 규칙을 추가해 주어야 한다.

###로컬 명령 실행하기

onrun_locally로 대체한다.

run_locally do
  within '/tmp' do
    execute :whoami
  end
end

References:

  1. Usage Examples of SSHKit