I’m using Ansible 1.5.3 and Git with ssh agent forwarding (https://help.github.com/articles/using-ssh-agent-forwarding). I can log into the server that I am managing with Ansible and test that my connection to git is correctly configured:
ubuntu@test:~$ ssh -T git@github.com
Hi gituser! You've successfully authenticated, but GitHub does not provide shell access.
I can also clone and update one of my repos using this account so my git configuration looks good and uses ssh forwarding when I log into my server directly via ssh.
The problem: When I attempt the same test shown above using the Ansible command module. It fails with “Permission denied”. Part of the Ansible output (with verbose logging) looks like this:
failed: [xxx.xxxxx.com] => {"changed": true, "cmd": ["ssh", "-T", "git@github.com"], "delta": "0:00:00.585481", "end": "2014-06-09 14:11:37.410907", "rc": 255, "start": "2014-06-09 14:11:36.825426"}
stderr: Permission denied (publickey).
Here is the simple playbook that runs this command:
- hosts: webservers
sudo: yes
remote_user: ubuntu
tasks:
- name: Test that git ssh connection is working.
command: ssh -T git@github.com
The question: why does everything work correctly when I manually log in via ssh and run the command but fail when the same command is run as the same user via Ansible?
I will post the answer shortly if no one else beats me to it. Although I am using git to demonstrate the problem, it could occur with any module that depends on ssh agent forwarding. It is not specific to Ansible but I suspect many will first encounter the problem in this scenario.
Another answer to your question (with the exception that I am using Ansible 1.9) could be the following:
You may want to check your /etc/ansible/ansible.cfg (or the other three potential locations where config settings can be overridden) for
transport=smart
as recommended in the ansible docs. Mine had defaulted totransport=paramiko
at some point during a previous install attempt, preventing my control machine from utilizing OpenSSH, and thus agent forwarding. This is probably a massive edge case, but who knows? It could be you!Though I didn't find it necessary for my configuration, I should note that others have mentioned that you should add
-o ForwardAgent=yes
to your ssh_args setting in the same file like so:I only mention it here for the sake of completeness.
To expand on @j.freckle's answer, the ansible way to change sudoers file is:
There are some very helpful partial answers here, but after running into this issue a number of times, I think an overview would be helpful.
First, you need to make sure that SSH agent forwarding is enabled when connecting from your client running Ansible to the target machine. Even with
transport=smart
, SSH agent forwarding may not be automatically enabled, depending on your client's SSH configuration. To ensure that it is, you can update your~/.ansible.cfg
to include this section:Next, you'll likely have to deal with the fact that
become: yes
(andbecome_user: root
) will generally disable agent forwarding because theSSH_AUTH_SOCK
environment variable is reset. (I find it shocking that Ansible seems to assume that people will SSH as root, since that makes any useful auditing impossible.) There are a few ways to deal with this. As of Ansible 2.2, the easiest approach is to preserve the (whole) environment when usingsudo
by specifying the-E
flag:However, this can have unwanted side-effects by preserving variables like
PATH
. The cleanest approach is to only preserveSSH_AUTH_SOCK
by including it inenv_keep
in your/etc/sudoers
file:To do this with Ansible:
This playbook task is a little more conservative than some of the others suggested, since it adds this after any other default
env_keep
settings (or at the end of the file, if none are found), without changing any existingenv_keep
settings or assumingSSH_AUTH_SOCK
is already present.The problem is resolved by removing this line from the playbook:
When sudo is run on the remote host, the environment variables set by ssh during login are no longer available. In particular, SSH_AUTH_SOCK, which "identifies the path of a UNIX-domain socket used to communicate with the agent" is no longer visible so ssh agent forwarding does not work.
Avoiding sudo when you don't need it is one way to work around the problem. Another way is to ensure that SSH_AUTH_SOCK sticks around during your sudo session by creating a sudoers file: