2003-11-04 03:34:24 -05:00
|
|
|
#!/usr/bin/python
|
|
|
|
|
2005-02-28 03:06:08 -05:00
|
|
|
# Copyright (C) 2003-2005 Robey Pointer <robey@lag.net>
|
2004-01-04 05:26:00 -05:00
|
|
|
#
|
|
|
|
# This file is part of paramiko.
|
|
|
|
#
|
|
|
|
# Paramiko 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.
|
|
|
|
#
|
|
|
|
# Paramiko 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
|
2004-06-10 14:12:00 -04:00
|
|
|
# along with Paramiko; if not, write to the Free Software Foundation, Inc.,
|
2004-01-04 05:26:00 -05:00
|
|
|
# 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
|
|
|
|
|
2004-04-06 04:16:02 -04:00
|
|
|
from __future__ import generators
|
|
|
|
|
2004-01-04 04:29:13 -05:00
|
|
|
"""
|
|
|
|
Useful functions used by the rest of paramiko.
|
|
|
|
"""
|
|
|
|
|
2005-02-28 02:09:02 -05:00
|
|
|
import sys, struct, traceback, threading
|
2004-04-06 04:16:02 -04:00
|
|
|
from common import *
|
|
|
|
|
|
|
|
# Change by RogerB - python < 2.3 doesn't have enumerate so we implement it
|
|
|
|
if sys.version_info < (2,3):
|
|
|
|
class enumerate:
|
|
|
|
def __init__ (self, sequence):
|
|
|
|
self.sequence = sequence
|
|
|
|
def __iter__ (self):
|
|
|
|
count = 0
|
|
|
|
for item in self.sequence:
|
|
|
|
yield (count, item)
|
|
|
|
count += 1
|
2003-11-04 03:34:24 -05:00
|
|
|
|
2004-04-05 06:12:59 -04:00
|
|
|
def inflate_long(s, always_positive=False):
|
2003-11-04 03:34:24 -05:00
|
|
|
"turns a normalized byte string into a long-int (adapted from Crypto.Util.number)"
|
|
|
|
out = 0L
|
|
|
|
negative = 0
|
|
|
|
if not always_positive and (len(s) > 0) and (ord(s[0]) >= 0x80):
|
|
|
|
negative = 1
|
|
|
|
if len(s) % 4:
|
|
|
|
filler = '\x00'
|
|
|
|
if negative:
|
|
|
|
filler = '\xff'
|
|
|
|
s = filler * (4 - len(s) % 4) + s
|
|
|
|
for i in range(0, len(s), 4):
|
|
|
|
out = (out << 32) + struct.unpack('>I', s[i:i+4])[0]
|
|
|
|
if negative:
|
|
|
|
out -= (1L << (8 * len(s)))
|
|
|
|
return out
|
|
|
|
|
2004-04-05 06:12:59 -04:00
|
|
|
def deflate_long(n, add_sign_padding=True):
|
2003-11-04 03:34:24 -05:00
|
|
|
"turns a long-int into a normalized byte string (adapted from Crypto.Util.number)"
|
|
|
|
# after much testing, this algorithm was deemed to be the fastest
|
|
|
|
s = ''
|
|
|
|
n = long(n)
|
|
|
|
while (n != 0) and (n != -1):
|
|
|
|
s = struct.pack('>I', n & 0xffffffffL) + s
|
|
|
|
n = n >> 32
|
|
|
|
# strip off leading zeros, FFs
|
|
|
|
for i in enumerate(s):
|
|
|
|
if (n == 0) and (i[1] != '\000'):
|
|
|
|
break
|
|
|
|
if (n == -1) and (i[1] != '\xff'):
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
# degenerate case, n was either 0 or -1
|
|
|
|
i = (0,)
|
|
|
|
if n == 0:
|
|
|
|
s = '\000'
|
|
|
|
else:
|
|
|
|
s = '\xff'
|
|
|
|
s = s[i[0]:]
|
|
|
|
if add_sign_padding:
|
|
|
|
if (n == 0) and (ord(s[0]) >= 0x80):
|
|
|
|
s = '\x00' + s
|
|
|
|
if (n == -1) and (ord(s[0]) < 0x80):
|
|
|
|
s = '\xff' + s
|
|
|
|
return s
|
|
|
|
|
|
|
|
def format_binary_weird(data):
|
|
|
|
out = ''
|
|
|
|
for i in enumerate(data):
|
|
|
|
out += '%02X' % ord(i[1])
|
|
|
|
if i[0] % 2:
|
|
|
|
out += ' '
|
|
|
|
if i[0] % 16 == 15:
|
|
|
|
out += '\n'
|
|
|
|
return out
|
|
|
|
|
|
|
|
def format_binary(data, prefix=''):
|
|
|
|
x = 0
|
|
|
|
out = []
|
|
|
|
while len(data) > x + 16:
|
|
|
|
out.append(format_binary_line(data[x:x+16]))
|
|
|
|
x += 16
|
|
|
|
if x < len(data):
|
|
|
|
out.append(format_binary_line(data[x:]))
|
|
|
|
return [prefix + x for x in out]
|
|
|
|
|
|
|
|
def format_binary_line(data):
|
|
|
|
left = ' '.join(['%02X' % ord(c) for c in data])
|
2004-03-04 03:21:45 -05:00
|
|
|
right = ''.join([('.%c..' % c)[(ord(c)+63)//95] for c in data])
|
2003-11-04 03:34:24 -05:00
|
|
|
return '%-50s %s' % (left, right)
|
|
|
|
|
|
|
|
def hexify(s):
|
|
|
|
"turn a string into a hex sequence"
|
|
|
|
return ''.join(['%02X' % ord(c) for c in s])
|
|
|
|
|
2004-04-05 06:12:59 -04:00
|
|
|
def unhexify(s):
|
|
|
|
"turn a hex sequence back into a string"
|
|
|
|
return ''.join([chr(int(s[i:i+2], 16)) for i in range(0, len(s), 2)])
|
|
|
|
|
2003-11-04 03:34:24 -05:00
|
|
|
def safe_string(s):
|
|
|
|
out = ''
|
|
|
|
for c in s:
|
|
|
|
if (ord(c) >= 32) and (ord(c) <= 127):
|
|
|
|
out += c
|
|
|
|
else:
|
|
|
|
out += '%%%02X' % ord(c)
|
|
|
|
return out
|
|
|
|
|
|
|
|
# ''.join([['%%%02X' % ord(c), c][(ord(c) >= 32) and (ord(c) <= 127)] for c in s])
|
2003-11-09 15:59:51 -05:00
|
|
|
|
|
|
|
def bit_length(n):
|
|
|
|
norm = deflate_long(n, 0)
|
|
|
|
hbyte = ord(norm[0])
|
|
|
|
bitlen = len(norm) * 8
|
|
|
|
while not (hbyte & 0x80):
|
|
|
|
hbyte <<= 1
|
|
|
|
bitlen -= 1
|
|
|
|
return bitlen
|
|
|
|
|
|
|
|
def tb_strings():
|
|
|
|
return ''.join(traceback.format_exception(*sys.exc_info())).split('\n')
|
2004-01-04 04:29:13 -05:00
|
|
|
|
|
|
|
def generate_key_bytes(hashclass, salt, key, nbytes):
|
|
|
|
"""
|
|
|
|
Given a password, passphrase, or other human-source key, scramble it
|
|
|
|
through a secure hash into some keyworthy bytes. This specific algorithm
|
|
|
|
is used for encrypting/decrypting private key files.
|
|
|
|
|
|
|
|
@param hashclass: class from L{Crypto.Hash} that can be used as a secure
|
|
|
|
hashing function (like C{MD5} or C{SHA}).
|
|
|
|
@type hashclass: L{Crypto.Hash}
|
|
|
|
@param salt: data to salt the hash with.
|
|
|
|
@type salt: string
|
|
|
|
@param key: human-entered password or passphrase.
|
|
|
|
@type key: string
|
|
|
|
@param nbytes: number of bytes to generate.
|
|
|
|
@type nbytes: int
|
|
|
|
@return: key data
|
|
|
|
@rtype: string
|
|
|
|
"""
|
|
|
|
keydata = ''
|
|
|
|
digest = ''
|
|
|
|
if len(salt) > 8:
|
|
|
|
salt = salt[:8]
|
|
|
|
while nbytes > 0:
|
|
|
|
hash = hashclass.new()
|
|
|
|
if len(digest) > 0:
|
|
|
|
hash.update(digest)
|
|
|
|
hash.update(key)
|
|
|
|
hash.update(salt)
|
|
|
|
digest = hash.digest()
|
|
|
|
size = min(nbytes, len(digest))
|
|
|
|
keydata += digest[:size]
|
|
|
|
nbytes -= size
|
|
|
|
return keydata
|
2004-04-05 06:12:59 -04:00
|
|
|
|
|
|
|
def mod_inverse(x, m):
|
|
|
|
# it's crazy how small python can make this function.
|
|
|
|
u1, u2, u3 = 1, 0, m
|
|
|
|
v1, v2, v3 = 0, 1, x
|
|
|
|
|
|
|
|
while v3 > 0:
|
|
|
|
q = u3 // v3
|
|
|
|
u1, v1 = v1, u1 - v1 * q
|
|
|
|
u2, v2 = v2, u2 - v2 * q
|
|
|
|
u3, v3 = v3, u3 - v3 * q
|
|
|
|
if u2 < 0:
|
|
|
|
u2 += m
|
|
|
|
return u2
|
2004-04-05 15:36:40 -04:00
|
|
|
|
2005-02-28 02:09:02 -05:00
|
|
|
g_thread_ids = {}
|
|
|
|
g_thread_counter = 0
|
|
|
|
g_thread_lock = threading.Lock()
|
|
|
|
def get_thread_id():
|
|
|
|
global g_thread_ids, g_thread_counter
|
|
|
|
tid = id(threading.currentThread())
|
|
|
|
try:
|
|
|
|
return g_thread_ids[tid]
|
|
|
|
except KeyError:
|
|
|
|
g_thread_lock.acquire()
|
|
|
|
try:
|
|
|
|
g_thread_counter += 1
|
|
|
|
ret = g_thread_ids[tid] = g_thread_counter
|
|
|
|
finally:
|
|
|
|
g_thread_lock.release()
|
|
|
|
return ret
|
|
|
|
|
2004-04-06 04:16:02 -04:00
|
|
|
def log_to_file(filename, level=DEBUG):
|
2004-04-05 15:36:40 -04:00
|
|
|
"send paramiko logs to a logfile, if they're not already going somewhere"
|
|
|
|
l = logging.getLogger("paramiko")
|
|
|
|
if len(l.handlers) > 0:
|
|
|
|
return
|
|
|
|
l.setLevel(level)
|
|
|
|
f = open(filename, 'w')
|
|
|
|
lh = logging.StreamHandler(f)
|
2005-02-28 02:09:02 -05:00
|
|
|
lh.setFormatter(logging.Formatter('%(levelname)-.3s [%(asctime)s] thr=%(_threadid)-3d %(name)s: %(message)s',
|
2004-04-05 15:36:40 -04:00
|
|
|
'%Y%m%d-%H:%M:%S'))
|
|
|
|
l.addHandler(lh)
|
2005-02-28 02:09:02 -05:00
|
|
|
|
|
|
|
def get_logger(name):
|
|
|
|
l = logging.getLogger(name)
|
|
|
|
class PFilter (object):
|
|
|
|
def filter(self, record):
|
|
|
|
record._threadid = get_thread_id()
|
|
|
|
return True
|
|
|
|
l.addFilter(PFilter())
|
|
|
|
return l
|