2020-03-04 21:23:49 +00:00
|
|
|
import sys
|
2020-01-09 02:37:58 +00:00
|
|
|
import struct
|
|
|
|
import itertools
|
|
|
|
|
2020-03-04 21:23:49 +00:00
|
|
|
assert sys.version_info[:2] >= (3,0), "This is Python 3 code"
|
|
|
|
|
2020-01-09 02:37:58 +00:00
|
|
|
def nbits(n):
|
|
|
|
# Mimic mp_get_nbits for ordinary Python integers.
|
|
|
|
assert 0 <= n
|
|
|
|
smax = next(s for s in itertools.count() if (n >> (1 << s)) == 0)
|
|
|
|
toret = 0
|
|
|
|
for shift in reversed([1 << s for s in range(smax)]):
|
|
|
|
if n >> shift != 0:
|
|
|
|
n >>= shift
|
|
|
|
toret += shift
|
|
|
|
assert n <= 1
|
|
|
|
if n == 1:
|
|
|
|
toret += 1
|
|
|
|
return toret
|
|
|
|
|
|
|
|
def ssh_byte(n):
|
|
|
|
return struct.pack("B", n)
|
|
|
|
|
|
|
|
def ssh_uint32(n):
|
|
|
|
return struct.pack(">L", n)
|
|
|
|
|
|
|
|
def ssh_string(s):
|
|
|
|
return ssh_uint32(len(s)) + s
|
|
|
|
|
|
|
|
def ssh1_mpint(x):
|
|
|
|
bits = nbits(x)
|
|
|
|
bytevals = [0xFF & (x >> (8*n)) for n in range((bits-1)//8, -1, -1)]
|
|
|
|
return struct.pack(">H" + "B" * len(bytevals), bits, *bytevals)
|
|
|
|
|
|
|
|
def ssh2_mpint(x):
|
|
|
|
bytevals = [0xFF & (x >> (8*n)) for n in range(nbits(x)//8, -1, -1)]
|
|
|
|
return struct.pack(">L" + "B" * len(bytevals), len(bytevals), *bytevals)
|
|
|
|
|
|
|
|
def decoder(fn):
|
|
|
|
def decode(s, return_rest = False):
|
|
|
|
item, length_consumed = fn(s)
|
|
|
|
if return_rest:
|
|
|
|
return item, s[length_consumed:]
|
|
|
|
else:
|
|
|
|
return item
|
|
|
|
return decode
|
|
|
|
|
|
|
|
@decoder
|
|
|
|
def ssh_decode_byte(s):
|
|
|
|
return struct.unpack_from("B", s, 0)[0], 1
|
|
|
|
|
|
|
|
@decoder
|
|
|
|
def ssh_decode_uint32(s):
|
|
|
|
return struct.unpack_from(">L", s, 0)[0], 4
|
|
|
|
|
|
|
|
@decoder
|
|
|
|
def ssh_decode_string(s):
|
|
|
|
length = ssh_decode_uint32(s)
|
|
|
|
assert length + 4 <= len(s)
|
|
|
|
return s[4:length+4], length+4
|
|
|
|
|
|
|
|
@decoder
|
|
|
|
def ssh1_get_mpint(s): # returns it unconsumed, still in wire encoding
|
|
|
|
nbits = struct.unpack_from(">H", s, 0)[0]
|
|
|
|
nbytes = (nbits + 7) // 8
|
|
|
|
assert nbytes + 2 <= len(s)
|
|
|
|
return s[:nbytes+2], nbytes+2
|
|
|
|
|
|
|
|
@decoder
|
|
|
|
def ssh1_decode_mpint(s):
|
|
|
|
nbits = struct.unpack_from(">H", s, 0)[0]
|
|
|
|
nbytes = (nbits + 7) // 8
|
|
|
|
assert nbytes + 2 <= len(s)
|
|
|
|
data = s[2:nbytes+2]
|
|
|
|
v = 0
|
|
|
|
for b in struct.unpack("B" * len(data), data):
|
|
|
|
v = (v << 8) | b
|
|
|
|
return v, nbytes+2
|
|
|
|
|
|
|
|
AGENT_MAX_MSGLEN = 262144
|
|
|
|
|
|
|
|
SSH1_AGENTC_REQUEST_RSA_IDENTITIES = 1
|
|
|
|
SSH1_AGENT_RSA_IDENTITIES_ANSWER = 2
|
|
|
|
SSH1_AGENTC_RSA_CHALLENGE = 3
|
|
|
|
SSH1_AGENT_RSA_RESPONSE = 4
|
|
|
|
SSH1_AGENTC_ADD_RSA_IDENTITY = 7
|
|
|
|
SSH1_AGENTC_REMOVE_RSA_IDENTITY = 8
|
|
|
|
SSH1_AGENTC_REMOVE_ALL_RSA_IDENTITIES = 9
|
|
|
|
SSH_AGENT_FAILURE = 5
|
|
|
|
SSH_AGENT_SUCCESS = 6
|
2021-05-24 14:04:35 +00:00
|
|
|
SSH_AGENT_EXTENSION_FAILURE = 28
|
2020-01-09 02:37:58 +00:00
|
|
|
SSH2_AGENTC_REQUEST_IDENTITIES = 11
|
|
|
|
SSH2_AGENT_IDENTITIES_ANSWER = 12
|
|
|
|
SSH2_AGENTC_SIGN_REQUEST = 13
|
|
|
|
SSH2_AGENT_SIGN_RESPONSE = 14
|
|
|
|
SSH2_AGENTC_ADD_IDENTITY = 17
|
|
|
|
SSH2_AGENTC_REMOVE_IDENTITY = 18
|
|
|
|
SSH2_AGENTC_REMOVE_ALL_IDENTITIES = 19
|
|
|
|
SSH2_AGENTC_EXTENSION = 27
|
|
|
|
|
|
|
|
SSH_AGENT_RSA_SHA2_256 = 2
|
|
|
|
SSH_AGENT_RSA_SHA2_512 = 4
|