159 lines
5.2 KiB
Python
159 lines
5.2 KiB
Python
# Copyright (C) 2005 John Arbash-Meinel <john@arbash-meinel.com>
|
|
# Modified up by: Todd Whiteman <ToddW@ActiveState.com>
|
|
#
|
|
# This file is part of ssh.
|
|
#
|
|
# 'ssh' 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.
|
|
#
|
|
# 'ssh' is distrubuted 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 'ssh'; if not, write to the Free Software Foundation, Inc.,
|
|
# 51 Franklin Street, Suite 500, Boston, MA 02110-1335 USA.
|
|
|
|
"""
|
|
Functions for communicating with Pageant, the basic windows ssh agent program.
|
|
"""
|
|
|
|
import os
|
|
import struct
|
|
import tempfile
|
|
import mmap
|
|
import array
|
|
import platform
|
|
import ctypes.wintypes
|
|
|
|
# if you're on windows, you should have one of these, i guess?
|
|
# ctypes is part of standard library since Python 2.5
|
|
_has_win32all = False
|
|
_has_ctypes = False
|
|
try:
|
|
# win32gui is preferred over win32ui to avoid MFC dependencies
|
|
import win32gui
|
|
_has_win32all = True
|
|
except ImportError:
|
|
try:
|
|
import ctypes
|
|
_has_ctypes = True
|
|
except ImportError:
|
|
pass
|
|
|
|
_AGENT_COPYDATA_ID = 0x804e50ba
|
|
_AGENT_MAX_MSGLEN = 8192
|
|
# Note: The WM_COPYDATA value is pulled from win32con, as a workaround
|
|
# so we do not have to import this huge library just for this one variable.
|
|
win32con_WM_COPYDATA = 74
|
|
|
|
|
|
def _get_pageant_window_object():
|
|
if _has_win32all:
|
|
try:
|
|
hwnd = win32gui.FindWindow('Pageant', 'Pageant')
|
|
return hwnd
|
|
except win32gui.error:
|
|
pass
|
|
elif _has_ctypes:
|
|
# Return 0 if there is no Pageant window.
|
|
return ctypes.windll.user32.FindWindowA('Pageant', 'Pageant')
|
|
return None
|
|
|
|
|
|
def can_talk_to_agent():
|
|
"""
|
|
Check to see if there is a "Pageant" agent we can talk to.
|
|
|
|
This checks both if we have the required libraries (win32all or ctypes)
|
|
and if there is a Pageant currently running.
|
|
"""
|
|
if (_has_win32all or _has_ctypes) and _get_pageant_window_object():
|
|
return True
|
|
return False
|
|
|
|
ULONG_PTR = ctypes.c_uint64 if platform.architecture()[0] == '64bit' else ctypes.c_uint32
|
|
class COPYDATASTRUCT(ctypes.Structure):
|
|
"""
|
|
ctypes implementation of
|
|
http://msdn.microsoft.com/en-us/library/windows/desktop/ms649010%28v=vs.85%29.aspx
|
|
"""
|
|
_fields_ = [
|
|
('num_data', ULONG_PTR),
|
|
('data_size', ctypes.wintypes.DWORD),
|
|
('data_loc', ctypes.c_void_p),
|
|
]
|
|
|
|
def _query_pageant(msg):
|
|
hwnd = _get_pageant_window_object()
|
|
if not hwnd:
|
|
# Raise a failure to connect exception, pageant isn't running anymore!
|
|
return None
|
|
|
|
# Write our pageant request string into the file (pageant will read this to determine what to do)
|
|
filename = tempfile.mktemp('.pag')
|
|
map_filename = os.path.basename(filename)
|
|
|
|
f = open(filename, 'w+b')
|
|
f.write(msg )
|
|
# Ensure the rest of the file is empty, otherwise pageant will read this
|
|
f.write('\0' * (_AGENT_MAX_MSGLEN - len(msg)))
|
|
# Create the shared file map that pageant will use to read from
|
|
pymap = mmap.mmap(f.fileno(), _AGENT_MAX_MSGLEN, tagname=map_filename, access=mmap.ACCESS_WRITE)
|
|
try:
|
|
# Create an array buffer containing the mapped filename
|
|
char_buffer = array.array("c", map_filename + '\0')
|
|
char_buffer_address, char_buffer_size = char_buffer.buffer_info()
|
|
# Create a string to use for the SendMessage function call
|
|
cds = COPYDATASTRUCT(_AGENT_COPYDATA_ID, char_buffer_size, char_buffer_address)
|
|
|
|
if _has_win32all:
|
|
# win32gui.SendMessage should also allow the same pattern as
|
|
# ctypes, but let's keep it like this for now...
|
|
response = win32gui.SendMessage(hwnd, win32con_WM_COPYDATA, ctypes.sizeof(cds), ctypes.addressof(cds))
|
|
elif _has_ctypes:
|
|
response = ctypes.windll.user32.SendMessageA(hwnd, win32con_WM_COPYDATA, ctypes.sizeof(cds), ctypes.byref(cds))
|
|
else:
|
|
response = 0
|
|
|
|
if response > 0:
|
|
datalen = pymap.read(4)
|
|
retlen = struct.unpack('>I', datalen)[0]
|
|
return datalen + pymap.read(retlen)
|
|
return None
|
|
finally:
|
|
pymap.close()
|
|
f.close()
|
|
# Remove the file, it was temporary only
|
|
os.unlink(filename)
|
|
|
|
|
|
class PageantConnection (object):
|
|
"""
|
|
Mock "connection" to an agent which roughly approximates the behavior of
|
|
a unix local-domain socket (as used by Agent). Requests are sent to the
|
|
pageant daemon via special Windows magick, and responses are buffered back
|
|
for subsequent reads.
|
|
"""
|
|
|
|
def __init__(self):
|
|
self._response = None
|
|
|
|
def send(self, data):
|
|
self._response = _query_pageant(data)
|
|
|
|
def recv(self, n):
|
|
if self._response is None:
|
|
return ''
|
|
ret = self._response[:n]
|
|
self._response = self._response[n:]
|
|
if self._response == '':
|
|
self._response = None
|
|
return ret
|
|
|
|
def close(self):
|
|
pass
|