mirror of https://github.com/huashengdun/webssh
Auto detect system default encoding
parent
cb5424a166
commit
469d86ac77
|
@ -22,6 +22,7 @@ from binascii import hexlify
|
||||||
import socket
|
import socket
|
||||||
# import sys
|
# import sys
|
||||||
import threading
|
import threading
|
||||||
|
import random
|
||||||
# import traceback
|
# import traceback
|
||||||
|
|
||||||
import paramiko
|
import paramiko
|
||||||
|
@ -36,8 +37,10 @@ host_key = paramiko.RSAKey(filename='tests/test_rsa.key')
|
||||||
|
|
||||||
print('Read key: ' + u(hexlify(host_key.get_fingerprint())))
|
print('Read key: ' + u(hexlify(host_key.get_fingerprint())))
|
||||||
|
|
||||||
|
banner = u'\r\n\u6b22\u8fce\r\n'
|
||||||
|
|
||||||
class Server (paramiko.ServerInterface):
|
|
||||||
|
class Server(paramiko.ServerInterface):
|
||||||
# 'data' is the output of base64.b64encode(key)
|
# 'data' is the output of base64.b64encode(key)
|
||||||
# (using the "user_rsa_key" files)
|
# (using the "user_rsa_key" files)
|
||||||
data = (b'AAAAB3NzaC1yc2EAAAABIwAAAIEAyO4it3fHlmGZWJaGrfeHOVY7RWO3P9M7hp'
|
data = (b'AAAAB3NzaC1yc2EAAAABIwAAAIEAyO4it3fHlmGZWJaGrfeHOVY7RWO3P9M7hp'
|
||||||
|
@ -46,8 +49,13 @@ class Server (paramiko.ServerInterface):
|
||||||
b'UWT10hcuO4Ks8=')
|
b'UWT10hcuO4Ks8=')
|
||||||
good_pub_key = paramiko.RSAKey(data=decodebytes(data))
|
good_pub_key = paramiko.RSAKey(data=decodebytes(data))
|
||||||
|
|
||||||
|
langs = ['en_US.UTF-8', 'zh_CN.GBK']
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.event = threading.Event()
|
self.shell_event = threading.Event()
|
||||||
|
self.exec_event = threading.Event()
|
||||||
|
self.lang = random.choice(self.langs)
|
||||||
|
self.encoding = self.lang.split('.')[-1]
|
||||||
|
|
||||||
def check_channel_request(self, kind, chanid):
|
def check_channel_request(self, kind, chanid):
|
||||||
if kind == 'session':
|
if kind == 'session':
|
||||||
|
@ -68,8 +76,19 @@ class Server (paramiko.ServerInterface):
|
||||||
def get_allowed_auths(self, username):
|
def get_allowed_auths(self, username):
|
||||||
return 'password,publickey'
|
return 'password,publickey'
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
def check_channel_shell_request(self, channel):
|
def check_channel_shell_request(self, channel):
|
||||||
self.event.set()
|
self.shell_event.set()
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def check_channel_pty_request(self, channel, term, width, height,
|
def check_channel_pty_request(self, channel, term, width, height,
|
||||||
|
@ -112,12 +131,19 @@ def run_ssh_server(port=2200, running=True):
|
||||||
username = t.get_username()
|
username = t.get_username()
|
||||||
print('{} Authenticated!'.format(username))
|
print('{} Authenticated!'.format(username))
|
||||||
|
|
||||||
server.event.wait(10)
|
server.shell_event.wait(2)
|
||||||
if not server.event.is_set():
|
if not server.shell_event.is_set():
|
||||||
print('*** Client never asked for a shell.')
|
print('*** Client never asked for a shell.')
|
||||||
continue
|
continue
|
||||||
|
|
||||||
chan.send('\r\n\r\nWelcome!\r\n\r\n')
|
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))
|
||||||
if username == 'bar':
|
if username == 'bar':
|
||||||
msg = chan.recv(1024)
|
msg = chan.recv(1024)
|
||||||
chan.send(msg)
|
chan.send(msg)
|
||||||
|
|
|
@ -9,7 +9,7 @@ from tornado.testing import AsyncHTTPTestCase
|
||||||
from tornado.options import options
|
from tornado.options import options
|
||||||
from webssh.main import make_app, make_handlers
|
from webssh.main import make_app, make_handlers
|
||||||
from webssh.settings import get_app_settings
|
from webssh.settings import get_app_settings
|
||||||
from tests.sshserver import run_ssh_server
|
from tests.sshserver import run_ssh_server, banner
|
||||||
|
|
||||||
|
|
||||||
handler.DELAY = 0.1
|
handler.DELAY = 0.1
|
||||||
|
@ -79,8 +79,10 @@ class TestApp(AsyncHTTPTestCase):
|
||||||
response = self.fetch('/')
|
response = self.fetch('/')
|
||||||
self.assertEqual(response.code, 200)
|
self.assertEqual(response.code, 200)
|
||||||
response = self.fetch('/', method="POST", body=self.body)
|
response = self.fetch('/', method="POST", body=self.body)
|
||||||
worker_id = json.loads(response.body.decode('utf-8'))['id']
|
data = json.loads(response.body.decode('utf-8'))
|
||||||
self.assertIsNotNone(worker_id)
|
self.assertIsNone(data['status'])
|
||||||
|
self.assertIsNotNone(data['id'])
|
||||||
|
self.assertIsNotNone(data['encoding'])
|
||||||
|
|
||||||
@tornado.testing.gen_test
|
@tornado.testing.gen_test
|
||||||
def test_app_with_correct_credentials_timeout(self):
|
def test_app_with_correct_credentials_timeout(self):
|
||||||
|
@ -90,11 +92,13 @@ class TestApp(AsyncHTTPTestCase):
|
||||||
self.assertEqual(response.code, 200)
|
self.assertEqual(response.code, 200)
|
||||||
|
|
||||||
response = yield client.fetch(url, method="POST", body=self.body)
|
response = yield client.fetch(url, method="POST", body=self.body)
|
||||||
worker_id = json.loads(response.body.decode('utf-8'))['id']
|
data = json.loads(response.body.decode('utf-8'))
|
||||||
self.assertIsNotNone(worker_id)
|
self.assertIsNone(data['status'])
|
||||||
|
self.assertIsNotNone(data['id'])
|
||||||
|
self.assertIsNotNone(data['encoding'])
|
||||||
|
|
||||||
url = url.replace('http', 'ws')
|
url = url.replace('http', 'ws')
|
||||||
ws_url = url + 'ws?id=' + worker_id
|
ws_url = url + 'ws?id=' + data['id']
|
||||||
yield tornado.gen.sleep(handler.DELAY + 0.1)
|
yield tornado.gen.sleep(handler.DELAY + 0.1)
|
||||||
ws = yield tornado.websocket.websocket_connect(ws_url)
|
ws = yield tornado.websocket.websocket_connect(ws_url)
|
||||||
msg = yield ws.read_message()
|
msg = yield ws.read_message()
|
||||||
|
@ -109,14 +113,16 @@ class TestApp(AsyncHTTPTestCase):
|
||||||
self.assertEqual(response.code, 200)
|
self.assertEqual(response.code, 200)
|
||||||
|
|
||||||
response = yield client.fetch(url, method="POST", body=self.body)
|
response = yield client.fetch(url, method="POST", body=self.body)
|
||||||
worker_id = json.loads(response.body.decode('utf-8'))['id']
|
data = json.loads(response.body.decode('utf-8'))
|
||||||
self.assertIsNotNone(worker_id)
|
self.assertIsNone(data['status'])
|
||||||
|
self.assertIsNotNone(data['id'])
|
||||||
|
self.assertIsNotNone(data['encoding'])
|
||||||
|
|
||||||
url = url.replace('http', 'ws')
|
url = url.replace('http', 'ws')
|
||||||
ws_url = url + 'ws?id=' + worker_id
|
ws_url = url + 'ws?id=' + data['id']
|
||||||
ws = yield tornado.websocket.websocket_connect(ws_url)
|
ws = yield tornado.websocket.websocket_connect(ws_url)
|
||||||
msg = yield ws.read_message()
|
msg = yield ws.read_message()
|
||||||
self.assertIn(b'Welcome!', msg)
|
self.assertEqual(msg.decode(data['encoding']), banner)
|
||||||
ws.close()
|
ws.close()
|
||||||
|
|
||||||
@tornado.testing.gen_test
|
@tornado.testing.gen_test
|
||||||
|
@ -128,14 +134,16 @@ class TestApp(AsyncHTTPTestCase):
|
||||||
|
|
||||||
body = self.body.replace('robey', 'bar')
|
body = self.body.replace('robey', 'bar')
|
||||||
response = yield client.fetch(url, method="POST", body=body)
|
response = yield client.fetch(url, method="POST", body=body)
|
||||||
worker_id = json.loads(response.body.decode('utf-8'))['id']
|
data = json.loads(response.body.decode('utf-8'))
|
||||||
self.assertIsNotNone(worker_id)
|
self.assertIsNone(data['status'])
|
||||||
|
self.assertIsNotNone(data['id'])
|
||||||
|
self.assertIsNotNone(data['encoding'])
|
||||||
|
|
||||||
url = url.replace('http', 'ws')
|
url = url.replace('http', 'ws')
|
||||||
ws_url = url + 'ws?id=' + worker_id
|
ws_url = url + 'ws?id=' + data['id']
|
||||||
ws = yield tornado.websocket.websocket_connect(ws_url)
|
ws = yield tornado.websocket.websocket_connect(ws_url)
|
||||||
msg = yield ws.read_message()
|
msg = yield ws.read_message()
|
||||||
self.assertIn(b'Welcome!', msg)
|
self.assertEqual(msg.decode(data['encoding']), banner)
|
||||||
|
|
||||||
# messages below will be ignored silently
|
# messages below will be ignored silently
|
||||||
yield ws.write_message('hello')
|
yield ws.write_message('hello')
|
||||||
|
|
|
@ -3,7 +3,22 @@ import os.path
|
||||||
import paramiko
|
import paramiko
|
||||||
|
|
||||||
from tornado.httputil import HTTPServerRequest
|
from tornado.httputil import HTTPServerRequest
|
||||||
from webssh.handler import MixinHandler, IndexHandler
|
from webssh.handler import MixinHandler, IndexHandler, parse_encoding
|
||||||
|
|
||||||
|
|
||||||
|
class TestHandler(unittest.TestCase):
|
||||||
|
|
||||||
|
def test_parse_encoding(self):
|
||||||
|
data = ''
|
||||||
|
self.assertIsNone(parse_encoding(data))
|
||||||
|
data = 'UTF-8'
|
||||||
|
self.assertEqual(parse_encoding(data), 'UTF-8')
|
||||||
|
data = 'en_US.UTF-8'
|
||||||
|
self.assertEqual(parse_encoding(data), 'UTF-8')
|
||||||
|
data = 'LANG=en_US.UTF-8\nLANGUAGE=\nLC_CTYPE="en_US.UTF-8"\n'
|
||||||
|
self.assertEqual(parse_encoding(data), 'UTF-8')
|
||||||
|
data = 'LANGUAGE=\nLC_CTYPE="en_US.UTF-8"\n'
|
||||||
|
self.assertEqual(parse_encoding(data), 'UTF-8')
|
||||||
|
|
||||||
|
|
||||||
class TestMixinHandler(unittest.TestCase):
|
class TestMixinHandler(unittest.TestCase):
|
||||||
|
|
|
@ -27,6 +27,13 @@ except ImportError:
|
||||||
DELAY = 3
|
DELAY = 3
|
||||||
|
|
||||||
|
|
||||||
|
def parse_encoding(data):
|
||||||
|
for line in data.split('\n'):
|
||||||
|
s = line.split('=')[-1]
|
||||||
|
if s:
|
||||||
|
return s.strip('"').split('.')[-1]
|
||||||
|
|
||||||
|
|
||||||
class MixinHandler(object):
|
class MixinHandler(object):
|
||||||
|
|
||||||
def get_real_client_addr(self):
|
def get_real_client_addr(self):
|
||||||
|
@ -122,6 +129,17 @@ class IndexHandler(MixinHandler, tornado.web.RequestHandler):
|
||||||
return self.get_real_client_addr() or self.request.connection.stream.\
|
return self.get_real_client_addr() or self.request.connection.stream.\
|
||||||
socket.getpeername()
|
socket.getpeername()
|
||||||
|
|
||||||
|
def get_default_encoding(self, ssh):
|
||||||
|
try:
|
||||||
|
_, stdout, _ = ssh.exec_command('locale')
|
||||||
|
except paramiko.SSHException:
|
||||||
|
result = None
|
||||||
|
else:
|
||||||
|
data = stdout.read().decode()
|
||||||
|
result = parse_encoding(data)
|
||||||
|
|
||||||
|
return result if result else 'utf-8'
|
||||||
|
|
||||||
def ssh_connect(self):
|
def ssh_connect(self):
|
||||||
ssh = paramiko.SSHClient()
|
ssh = paramiko.SSHClient()
|
||||||
ssh._system_host_keys = self.host_keys_settings['system_host_keys']
|
ssh._system_host_keys = self.host_keys_settings['system_host_keys']
|
||||||
|
@ -146,6 +164,7 @@ class IndexHandler(MixinHandler, tornado.web.RequestHandler):
|
||||||
chan.setblocking(0)
|
chan.setblocking(0)
|
||||||
worker = Worker(self.loop, ssh, chan, dst_addr)
|
worker = Worker(self.loop, ssh, chan, dst_addr)
|
||||||
worker.src_addr = self.get_client_addr()
|
worker.src_addr = self.get_client_addr()
|
||||||
|
worker.encoding = self.get_default_encoding(ssh)
|
||||||
return worker
|
return worker
|
||||||
|
|
||||||
def ssh_connect_wrapped(self, future):
|
def ssh_connect_wrapped(self, future):
|
||||||
|
@ -164,6 +183,7 @@ class IndexHandler(MixinHandler, tornado.web.RequestHandler):
|
||||||
def post(self):
|
def post(self):
|
||||||
worker_id = None
|
worker_id = None
|
||||||
status = None
|
status = None
|
||||||
|
encoding = None
|
||||||
|
|
||||||
future = Future()
|
future = Future()
|
||||||
t = threading.Thread(target=self.ssh_connect_wrapped, args=(future,))
|
t = threading.Thread(target=self.ssh_connect_wrapped, args=(future,))
|
||||||
|
@ -178,8 +198,9 @@ class IndexHandler(MixinHandler, tornado.web.RequestHandler):
|
||||||
worker_id = worker.id
|
worker_id = worker.id
|
||||||
workers[worker_id] = worker
|
workers[worker_id] = worker
|
||||||
self.loop.call_later(DELAY, recycle_worker, worker)
|
self.loop.call_later(DELAY, recycle_worker, worker)
|
||||||
|
encoding = worker.encoding
|
||||||
|
|
||||||
self.write(dict(id=worker_id, status=status))
|
self.write(dict(id=worker_id, status=status, encoding=encoding))
|
||||||
|
|
||||||
|
|
||||||
class WsockHandler(MixinHandler, tornado.websocket.WebSocketHandler):
|
class WsockHandler(MixinHandler, tornado.websocket.WebSocketHandler):
|
||||||
|
|
|
@ -59,12 +59,14 @@ jQuery(function($){
|
||||||
join = (ws_url[ws_url.length-1] === '/' ? '' : '/'),
|
join = (ws_url[ws_url.length-1] === '/' ? '' : '/'),
|
||||||
url = ws_url + join + 'ws?id=' + msg.id,
|
url = ws_url + join + 'ws?id=' + msg.id,
|
||||||
sock = new window.WebSocket(url),
|
sock = new window.WebSocket(url),
|
||||||
|
encoding = msg.encoding,
|
||||||
terminal = document.getElementById('#terminal'),
|
terminal = document.getElementById('#terminal'),
|
||||||
term = new window.Terminal({
|
term = new window.Terminal({
|
||||||
cursorBlink: true,
|
cursorBlink: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
console.log(url);
|
console.log(url);
|
||||||
|
console.log(encoding);
|
||||||
wssh.sock = sock;
|
wssh.sock = sock;
|
||||||
wssh.term = term;
|
wssh.term = term;
|
||||||
|
|
||||||
|
@ -83,7 +85,7 @@ jQuery(function($){
|
||||||
var reader = new window.FileReader();
|
var reader = new window.FileReader();
|
||||||
|
|
||||||
reader.onloadend = function(){
|
reader.onloadend = function(){
|
||||||
var decoder = new window.TextDecoder();
|
var decoder = new window.TextDecoder(encoding);
|
||||||
var text = decoder.decode(reader.result);
|
var text = decoder.decode(reader.result);
|
||||||
// console.log(text);
|
// console.log(text);
|
||||||
term.write(text);
|
term.write(text);
|
||||||
|
|
Loading…
Reference in New Issue