■ Ansibleとは
- 
Ansibleの特徴 - エージェントレスでサーバ構築を自動化してくれるツールです。
- sshdが動作しているマシンにSSHで接続し環境を構築してくれます。
- 設定ファイルはYAMLで書きます。プログラムの知識(Ruby、Python、etc.)は不要です。
- 数台~数十台規模のサーバ構築だったらchefよりも簡単です。
 
- 
環境の準備 
 以前の記事を参照してください。
 Ansible 最初の一歩
■ はじめに
Ansible 1.9から、他のユーザの権限(例えばroot権限)で実行するための'become'が追加されました。
rootでの直接アクセスが禁止されているサーバを管理する場合、一旦、一般ユーザとしてログインし、その後、sudo su -でrootになって操作されるかと思います。
Ansibleでbecome: tureしてコマンドを実行する場合と、sudo su -した後にコマンドラインからコマンドを直接実行する場合で、環境変数のPATHが異なるため、コマンドの実行結果が異なる場合があり注意が必要です。
今まで、この辺をあまり意識せずに使っていたため、コマンドを直接実行して確認済みのコマンドが、PlayBookでは失敗する事態が発生し、少し苦労したので、becomeと環境変数のPATHについて、改めて調べてみました。
つまづいたポイント
- 「become: true≠sudo su -」だったこと。
- 
sudo su -の状態で各種コマンドを実行する方法が不明なこと。
Ansible本家の見解
Ansible公式HP、GitHubのissuesを調査してみました。
Ansible公式HP(リンク)
Only one method may be enabled per host
Methods cannot be chained. You cannot use
sudo /bin/su -to become a user, you need to have privileges to run the command as that user in sudo or be able to su directly to it.
意訳:
sudoとsuは、同時に使えないので、sudo su -したい人は、直接コマンドを実行してください。
Ansible GitHub issues
本家でのディスカッションも結論がでていない模様(ステータスがOpenのままです)。
  GitHub issues#12686
■ 動作検証
まずは、PATHがどのようになるのか調査してみました。
調査環境:Amazon Linux AMI 2016.03
remote_user: ec2-user
1. become: false(ec2-user)
  /usr/local/bin:/bin:/usr/bin:/opt/aws/bin
2. become: true(root)
  /sbin:/bin:/usr/sbin:/usr/bin
3. lookup('pipe', 'echo $PATH')
  /usr/local/bin:/bin:/usr/bin:/usr/local/sbin:/usr/sbin:
  /sbin:/opt/aws/bin:/home/ec2-user/.local/bin:/home/ec2-user/bin
4. lookup('env', 'PATH')
  /usr/local/bin:/bin:/usr/bin:/usr/local/sbin:/usr/sbin:
  /sbin:/opt/aws/bin:/home/ec2-user/.local/bin:/home/ec2-user/bin
5. ec2-user
  /usr/local/bin:/bin:/usr/bin:/usr/local/sbin:/usr/sbin:
  /sbin:/opt/aws/bin:/home/ec2-user/.local/bin:/home/ec2-user/bin
6. sudo su -
  /usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:
  /opt/aws/bin:/root/bin
7. sudo su
  /sbin:/bin:/usr/sbin:/usr/bin:/opt/aws/bin
見事にバラバラですね ^^;
■ 「sudo su -」相当のPATHを設定する
Googleで検索すると、commandやshellで.bashrcや.bash_profileを読み込む方法が紹介されていますが、これをやってしまうと冪等性の保証が難しくなってしまいます。そこで、environmentでPATHを設定する方法について紹介します。
設定前の状態
まず、何もしていないとどのようになるのか実行してみます。
- hosts: all
  remote_user: ec2-user
  tasks:
    - command: echo $PATH
      register: become_path
      become: true
      changed_when: False
    - name: become
      debug: var=become_path.stdout
    - command: echo $PATH
      register: not_become_path
      changed_when: False
    - name: not become
      debug: var=not_become_path.stdout
実行してみます。
$ ansible-playbook env_path.yml
PLAY [all] *********************************************************************
TASK [setup] *******************************************************************
ok: [target03]
TASK [command] *****************************************************************
ok: [target03]
TASK [become] ******************************************************************
ok: [target03] => {
    "become_path.stdout": "/sbin:/bin:/usr/sbin:/usr/bin"
}
TASK [command] *****************************************************************
ok: [target03]
TASK [not become] **************************************************************
ok: [target03] => {
    "not_become_path.stdout": "/usr/local/bin:/bin:/usr/bin:/opt/aws/bin"
}
PLAY RECAP *********************************************************************
target03                   : ok=5    changed=0    unreachable=0    failed=0
一般ユーザのPATH:/usr/local/bin:/bin:/usr/bin:/opt/aws/bin
rootのPATH:/sbin:/bin:/usr/sbin:/usr/bin
になっています。いわゆるログインしていない状態のようです。
lookupを使った場合は以下のようになります。
---
- hosts: all
  remote_user: ec2-user
  vars:
    pipe_path: "{{ lookup('pipe', 'echo $PATH') }}"
    env_path: "{{ lookup('env', 'PATH') }}"
  tasks:
    - name: pipe path
      debug: msg={{ pipe_path }}
    - name: env path
      debug: msg={{ env_path }}
lookupを使った場合の実行結果は以下のようになります。
$ ansible-playbook playbook.yml
PLAY [all] *********************************************************************
TASK [setup] *******************************************************************
ok: [target01]
TASK [pipe path]] **************************************************************
ok: [target01] => {
    "msg": "/usr/local/bin:/bin:/usr/bin:/usr/local/sbin:/usr/sbin:/sbin:/opt/aws/bin:/home/ec2-user/.local/bin:/home/ec2-user/bin"
}
TASK [env path]] ***************************************************************
ok: [target01] => {
    "msg": "/usr/local/bin:/bin:/usr/bin:/usr/local/sbin:/usr/sbin:/sbin:/opt/aws/bin:/home/ec2-user/.local/bin:/home/ec2-user/bin"
}
PLAY RECAP *********************************************************************
target01                   : ok=3    changed=0    unreachable=0    failed=0
こちらは、ec2-userでログインした状態になっていました。
environmentでPATHを設定してみる
environmentでPATHを設定してみます。
あらかじめ、sudo su -して、echo $PATHを調べておきます。
$ sudo su -
# echo $PATH
/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:/opt/aws/bin:/root/bin
PlayBookは以下のようになります。
- hosts: all
  remote_user: ec2-user
  environment:
    PATH: "/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:/opt/aws/bin:/root/bin"
  tasks:
    - command: echo $PATH
      register: become_path
      become: true
      changed_when: False
    - name: become
      debug: var=become_path.stdout
    - command: echo $PATH
      register: not_become_path
      changed_when: False
    - name: not become
      debug: var=not_become_path.stdout
実行してみましょう。
$ ansible-playbook playbook.yml
PLAY [all] *********************************************************************
TASK [setup] *******************************************************************
ok: [target01]
TASK [command] *****************************************************************
ok: [target01]
TASK [become] ******************************************************************
ok: [target01] => {
    "become_path.stdout": "/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:/opt/aws/bin:/root/bin"
}
TASK [command] *****************************************************************
ok: [target01]
TASK [not become] **************************************************************
ok: [target01] => {
    "not_become_path.stdout": "/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:/opt/aws/bin:/root/bin"
}
PLAY RECAP *********************************************************************
target01                   : ok=5    changed=0    unreachable=0    failed=0
environment:をPlayBookに対して設定したため、become: trueの場合も、そうではない場合も、いずれも設定したPATHになっています。
ちなみに、environment:は、タスクのほうに書くことも可能です。タスクごとに設定したい場合は、タスクに対して指定していくことになります。PlayBookは以下のようになります。
- hosts: all
  remote_user: ec2-user
  tasks:
    - command: echo $PATH
      register: become_path
      become: true
      changed_when: False
      environment:
        PATH: "/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:/opt/aws/bin:/root/bin"
    - name: become
      debug: var=become_path.stdout
    - command: echo $PATH
      register: not_become_path
      changed_when: False
      environment:
        PATH: "/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:/opt/aws/bin:/root/bin"
    - name: not become
      debug: var=not_become_path.stdout
実行結果は、PlayBook全体に対してenvironment:を設定した場合と同じになります。
environmentのスコープ
PlayBookやTaskに対して設定したenvironment:のスコープは、設定した範囲になります。
例えば、PlayBookに対して設定した場合、別のPlayBookへは影響がでないようになっています。
自分でコマンドを羅列して/bin/shで実行する場合は、設定したPATHを元に戻すコマンドも書かなければならないのですが、AnsibleのPlayBookは、その辺を自動で行ってくれるので便利です。
■ おわりに
今回は、environmentを使ってPATHを設定する方法を紹介しました。
PATHの設定方法は、説明した通りになりますが、PATH依存のPlayBookは思わぬ落とし穴にはまってしまう可能性があります。PlayBookで、shellやcommandモジュールを使う際には、フルパスで書く習慣をつけておくと、安全でよいかと思います。
- 
落とし穴にはまる可能性あり(PATH依存) - command: cmd
- 
安全な書き方 - command: /usr/local/bin/cmd
Ansibleのような構成管理ツールを使うにあたり、単に動作しますという状態ではなく、保守性や堅牢性を考慮した運用ができるように心がけていけるとよいですね。