webssh/tests/sshserver.py

163 lines
5.0 KiB
Python
Raw Normal View History

2018-04-26 13:51:01 +00:00
#!/usr/bin/env python
# Copyright (C) 2003-2007 Robey Pointer <robeypointer@gmail.com>
#
# 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 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 Paramiko; if not, write to the Free Software Foundation, Inc.,
# 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
from binascii import hexlify
import socket
# import sys
import threading
2018-05-30 13:29:44 +00:00
import random
2018-04-26 13:51:01 +00:00
# import traceback
import paramiko
from paramiko.py3compat import u, decodebytes
# setup logging
paramiko.util.log_to_file('tests/sshserver.log')
2018-04-26 13:51:01 +00:00
host_key = paramiko.RSAKey(filename='tests/test_rsa.key')
# host_key = paramiko.DSSKey(filename='test_dss.key')
print('Read key: ' + u(hexlify(host_key.get_fingerprint())))
2018-05-30 13:29:44 +00:00
banner = u'\r\n\u6b22\u8fce\r\n'
2018-04-26 13:51:01 +00:00
2018-05-30 13:29:44 +00:00
class Server(paramiko.ServerInterface):
2018-04-26 13:51:01 +00:00
# 'data' is the output of base64.b64encode(key)
# (using the "user_rsa_key" files)
data = (b'AAAAB3NzaC1yc2EAAAABIwAAAIEAyO4it3fHlmGZWJaGrfeHOVY7RWO3P9M7hp'
b'fAu7jJ2d7eothvfeuoRFtJwhUmZDluRdFyhFY/hFAh76PJKGAusIqIQKlkJxMC'
b'KDqIexkgHAfID/6mqvmnSJf0b5W8v5h2pI/stOSwTQ+pxVhwJ9ctYDhRSlF0iT'
b'UWT10hcuO4Ks8=')
good_pub_key = paramiko.RSAKey(data=decodebytes(data))
2018-05-30 13:29:44 +00:00
langs = ['en_US.UTF-8', 'zh_CN.GBK']
2018-04-26 13:51:01 +00:00
def __init__(self):
2018-05-30 13:29:44 +00:00
self.shell_event = threading.Event()
self.exec_event = threading.Event()
self.lang = random.choice(self.langs)
self.encoding = self.lang.split('.')[-1]
2018-04-26 13:51:01 +00:00
def check_channel_request(self, kind, chanid):
if kind == 'session':
return paramiko.OPEN_SUCCEEDED
return paramiko.OPEN_FAILED_ADMINISTRATIVELY_PROHIBITED
def check_auth_password(self, username, password):
2018-04-27 06:34:06 +00:00
if (username in ['robey', 'bar']) and (password == 'foo'):
2018-04-26 13:51:01 +00:00
return paramiko.AUTH_SUCCESSFUL
return paramiko.AUTH_FAILED
def check_auth_publickey(self, username, key):
print('Auth attempt with key: ' + u(hexlify(key.get_fingerprint())))
if (username == 'robey') and (key == self.good_pub_key):
return paramiko.AUTH_SUCCESSFUL
return paramiko.AUTH_FAILED
def get_allowed_auths(self, username):
return 'password,publickey'
2018-05-30 13:29:44 +00:00
def check_channel_exec_request(self, channel, command):
if command != b'locale':
ret = False
else:
ret = True
result = 'LANG={lang}\nLANGUAGE=\nLC_CTYPE="{lang}"\n'.format(lang=self.lang) # noqa
channel.send(result)
channel.shutdown(1)
self.exec_event.set()
return ret
2018-04-26 13:51:01 +00:00
def check_channel_shell_request(self, channel):
2018-05-30 13:29:44 +00:00
self.shell_event.set()
2018-04-26 13:51:01 +00:00
return True
def check_channel_pty_request(self, channel, term, width, height,
pixelwidth, pixelheight, modes):
return True
def check_channel_window_change_request(self, channel, width, height,
pixelwidth, pixelheight):
channel.send('resized')
return True
2018-04-26 13:51:01 +00:00
2018-04-27 01:49:48 +00:00
def run_ssh_server(port=2200, running=True):
2018-04-26 13:51:01 +00:00
# now connect
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
2018-04-27 01:49:48 +00:00
sock.bind(('127.0.0.1', port))
2018-04-26 13:51:01 +00:00
sock.listen(100)
2018-04-27 01:31:52 +00:00
while running:
2018-04-26 13:51:01 +00:00
client, addr = sock.accept()
print('Got a connection!')
t = paramiko.Transport(client)
t.load_server_moduli()
t.add_server_key(host_key)
server = Server()
try:
t.start_server(server=server)
except Exception as e:
print(e)
continue
# wait for auth
chan = t.accept(20)
if chan is None:
print('*** No channel.')
continue
2018-04-27 06:34:06 +00:00
username = t.get_username()
print('{} Authenticated!'.format(username))
2018-04-26 13:51:01 +00:00
2018-05-30 13:29:44 +00:00
server.shell_event.wait(2)
if not server.shell_event.is_set():
2018-04-26 13:51:01 +00:00
print('*** Client never asked for a shell.')
continue
2018-05-30 13:29:44 +00:00
server.exec_event.wait(2)
if not server.exec_event.is_set():
print('*** Client never asked for a command.')
continue
# chan.send('\r\n\r\nWelcome!\r\n\r\n')
print(server.encoding)
chan.send(banner.encode(server.encoding))
2018-04-27 06:34:06 +00:00
if username == 'bar':
2018-05-26 07:51:53 +00:00
msg = chan.recv(1024)
chan.send(msg)
2018-04-27 06:41:08 +00:00
2018-04-27 06:34:06 +00:00
chan.close()
2018-04-27 06:41:08 +00:00
t.close()
client.close()
2018-04-26 13:51:01 +00:00
try:
sock.close()
except Exception:
pass
if __name__ == '__main__':
2018-04-27 01:31:52 +00:00
run_ssh_server()