1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120
|
#!/usr/bin/python3 -cimport os, sys; os.execv(os.path.dirname(sys.argv[1]) + "/../common/pywrap", sys.argv)
# This file is part of Cockpit.
#
# Copyright (C) 2013 Red Hat, Inc.
#
# Cockpit is free software; you can redistribute it and/or modify it
# under the terms of the GNU Lesser General Public License as published by
# the Free Software Foundation; either version 2.1 of the License, or
# (at your option) any later version.
#
# Cockpit is distributed in the hope that it will be useful, but
# WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with Cockpit; If not, see <https://www.gnu.org/licenses/>.
import testlib
@testlib.nondestructive
class TestSession(testlib.MachineCase):
def testBasic(self):
m = self.machine
b = self.browser
def wait_session(should_exist):
if should_exist:
m.execute("until loginctl list-sessions | grep admin; do sleep 1; done")
else:
# restart logind to mop up empty "closing" sessions, like in testlib terminate_sessions()
m.execute("""while true; do
OUT=$(loginctl list-sessions)
echo "$OUT" | grep admin || break
if echo "$OUT | grep closing"; then
systemctl stop systemd-logind
fi
sleep 3
done""")
wait_session(should_exist=False)
# Login
self.login_and_go("/system")
wait_session(should_exist=True)
# Check session type
if not m.ws_container:
# Systemd 256 also shows a class=manager session for admin
session_id = m.execute("loginctl list-sessions | grep -v manager | awk '/admin/ {print $1}'").strip()
self.assertEqual(m.execute(f"loginctl show-session -p Type {session_id}").strip(), "Type=web")
if not m.ws_container and m.image != "arch":
# starts ssh-agent in session
bridge = m.execute("pgrep -u admin cockpit-bridge").strip()
out = m.execute(f"grep --null-data SSH_AGENT_PID /proc/{bridge}/environ | xargs -0")
agent = out.split("=")[1].strip()
self.assertIn("ssh-agent", m.execute(f"readlink /proc/{agent}/exe").strip())
# Logout
b.logout()
b.wait_visible("#login")
wait_session(should_exist=False)
# cleans up ssh-agent
m.execute("while pgrep -u admin ssh-agent; do sleep 1; done", timeout=10)
# Login again
b.set_val("#login-user-input", "admin")
b.set_val("#login-password-input", "foobar")
b.click('#login-button')
b.enter_page("/system")
wait_session(should_exist=True)
# Kill session from the outside
m.execute("loginctl terminate-user admin")
wait_session(should_exist=False)
b.relogin("/system", "admin")
wait_session(should_exist=True)
# Kill session from the inside
b.logout()
wait_session(should_exist=False)
# try to pwn $SSH_AGENT_PID via pam_env's user_readenv=1 (CVE-2024-6126)
if m.ws_container:
# not using cockpit's PAM config
return
# this is enabled by default in tools/cockpit.debian.pam, as well as
# Debian/Ubuntu's /etc/pam.d/sshd; but not in Fedora/RHEL
if "debian" not in m.image and "ubuntu" not in m.image:
self.write_file("/etc/pam.d/cockpit", "session required pam_env.so user_readenv=1\n", append=True)
victim_pid = m.spawn("sleep infinity", "sleep.log")
self.addCleanup(m.execute, f"kill {victim_pid} || true")
self.write_file("/home/admin/.pam_environment", f"SSH_AGENT_PID DEFAULT={victim_pid}\n", owner="admin")
b.login_and_go()
wait_session(should_exist=True)
# starts ssh-agent in session
m.execute("pgrep -u admin ssh-agent")
# but the session has the modified SSH_AGENT_PID
bridge = m.execute("pgrep -u admin cockpit-bridge").strip()
agent = m.execute(f"grep --null-data SSH_AGENT_PID /proc/{bridge}/environ | xargs -0 | sed 's/.*=//'").strip()
self.assertEqual(agent, str(victim_pid))
# logging out still kills the actual ssh-agent, not the victim pid
b.logout()
wait_session(should_exist=False)
m.execute("while pgrep -u admin ssh-agent; do sleep 1; done", timeout=10)
m.execute(f"test -e /proc/{victim_pid}")
if __name__ == '__main__':
testlib.test_main()
|