2018-04-26 17:11:16 +00:00
|
|
|
WebSSH
|
|
|
|
------
|
|
|
|
|
2018-04-27 00:44:22 +00:00
|
|
|
|Build Status| |codecov| |PyPI - Python Version| |PyPI|
|
2018-04-26 17:11:16 +00:00
|
|
|
|
2018-08-20 15:09:09 +00:00
|
|
|
Introduction
|
2019-05-19 12:19:51 +00:00
|
|
|
~~~~~~~~~~~~
|
2018-08-20 15:09:09 +00:00
|
|
|
|
2018-04-26 17:11:16 +00:00
|
|
|
A simple web application to be used as an ssh client to connect to your
|
2018-08-29 15:12:21 +00:00
|
|
|
ssh servers. It is written in Python, base on tornado, paramiko and
|
|
|
|
xterm.js.
|
2018-04-26 17:11:16 +00:00
|
|
|
|
|
|
|
Features
|
2019-05-19 12:19:51 +00:00
|
|
|
~~~~~~~~
|
2018-04-26 17:11:16 +00:00
|
|
|
|
|
|
|
- SSH password authentication supported, including empty password.
|
|
|
|
- SSH public-key authentication supported, including DSA RSA ECDSA
|
|
|
|
Ed25519 keys.
|
|
|
|
- Encrypted keys supported.
|
2019-12-10 09:10:01 +00:00
|
|
|
- Two-Factor Authentication (time-based one-time password) supported.
|
2018-04-26 17:11:16 +00:00
|
|
|
- Fullscreen terminal supported.
|
|
|
|
- Terminal window resizable.
|
2018-08-20 15:09:09 +00:00
|
|
|
- Auto detect the ssh server's default encoding.
|
2018-09-15 01:46:40 +00:00
|
|
|
- Modern browsers including Chrome, Firefox, Safari, Edge, Opera
|
|
|
|
supported.
|
2018-08-20 15:09:09 +00:00
|
|
|
|
|
|
|
Preview
|
2019-05-19 12:19:51 +00:00
|
|
|
~~~~~~~
|
2018-08-20 15:09:09 +00:00
|
|
|
|
|
|
|
|Login| |Terminal|
|
|
|
|
|
2018-09-01 03:20:31 +00:00
|
|
|
How it works
|
|
|
|
~~~~~~~~~~~~
|
|
|
|
|
|
|
|
::
|
|
|
|
|
|
|
|
+---------+ http +--------+ ssh +-----------+
|
|
|
|
| browser | <==========> | webssh | <=======> | ssh server|
|
|
|
|
+---------+ websocket +--------+ ssh +-----------+
|
|
|
|
|
2018-08-29 15:12:21 +00:00
|
|
|
Requirements
|
|
|
|
~~~~~~~~~~~~
|
2018-08-20 15:09:09 +00:00
|
|
|
|
|
|
|
- Python 2.7/3.4+
|
2018-04-26 17:11:16 +00:00
|
|
|
|
2018-08-29 15:12:21 +00:00
|
|
|
Quickstart
|
|
|
|
~~~~~~~~~~
|
|
|
|
|
2018-08-29 23:18:34 +00:00
|
|
|
1. Install this app, run command ``pip install webssh``
|
|
|
|
2. Start a webserver, run command ``wssh``
|
2018-08-29 15:12:21 +00:00
|
|
|
3. Open your browser, navigate to ``127.0.0.1:8888``
|
|
|
|
4. Input your data, submit the form.
|
|
|
|
|
|
|
|
Server options
|
|
|
|
~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
.. code:: bash
|
|
|
|
|
2018-10-19 10:55:20 +00:00
|
|
|
# start a http server with specified listen address and listen port
|
2018-10-22 13:51:51 +00:00
|
|
|
wssh --address='2.2.2.2' --port=8000
|
2018-04-26 17:11:16 +00:00
|
|
|
|
2018-10-22 13:51:51 +00:00
|
|
|
# start a https server, certfile and keyfile must be passed
|
|
|
|
wssh --certfile='/path/to/cert.crt' --keyfile='/path/to/cert.key'
|
2018-10-19 10:55:20 +00:00
|
|
|
|
2018-08-10 05:45:01 +00:00
|
|
|
# missing host key policy
|
2018-04-26 17:11:16 +00:00
|
|
|
wssh --policy=reject
|
|
|
|
|
2018-08-10 05:45:01 +00:00
|
|
|
# logging level
|
2018-04-26 17:11:16 +00:00
|
|
|
wssh --logging=debug
|
|
|
|
|
|
|
|
# log to file
|
|
|
|
wssh --log-file-prefix=main.log
|
|
|
|
|
|
|
|
# more options
|
|
|
|
wssh --help
|
|
|
|
|
2018-12-15 13:40:16 +00:00
|
|
|
Browser console
|
|
|
|
~~~~~~~~~~~~~~~
|
2018-08-29 15:12:21 +00:00
|
|
|
|
|
|
|
.. code:: javascript
|
|
|
|
|
|
|
|
// connect to your ssh server
|
2019-07-07 07:19:39 +00:00
|
|
|
wssh.connect(hostname, port, username, password, privatekey, passphrase, totp);
|
2018-08-29 15:12:21 +00:00
|
|
|
|
2018-09-09 07:41:28 +00:00
|
|
|
// pass an object to wssh.connect
|
2018-08-29 15:12:21 +00:00
|
|
|
var opts = {
|
|
|
|
hostname: 'hostname',
|
|
|
|
port: 'port',
|
|
|
|
username: 'username',
|
|
|
|
password: 'password',
|
2019-07-07 07:19:39 +00:00
|
|
|
privatekey: 'the private key text',
|
|
|
|
passphrase: 'passphrase',
|
|
|
|
totp: 'totp'
|
2018-08-29 15:12:21 +00:00
|
|
|
};
|
|
|
|
wssh.connect(opts);
|
|
|
|
|
2018-09-01 03:20:31 +00:00
|
|
|
// without an argument, wssh will use the form data to connect
|
|
|
|
wssh.connect();
|
|
|
|
|
|
|
|
// set a new encoding for client to use
|
|
|
|
wssh.set_encoding(encoding);
|
|
|
|
|
|
|
|
// reset encoding to use the default one
|
|
|
|
wssh.reset_encoding();
|
|
|
|
|
2018-08-29 15:12:21 +00:00
|
|
|
// send a command to the server
|
|
|
|
wssh.send('ls -l');
|
|
|
|
|
2019-06-22 06:09:04 +00:00
|
|
|
Custom Font
|
|
|
|
~~~~~~~~~~~
|
|
|
|
|
2020-01-23 11:35:04 +00:00
|
|
|
To use custom font, put your font file in the directory
|
|
|
|
``webssh/static/css/fonts/`` and restart the server.
|
2019-06-22 06:09:04 +00:00
|
|
|
|
2019-05-18 06:27:14 +00:00
|
|
|
URL Arguments
|
|
|
|
~~~~~~~~~~~~~
|
|
|
|
|
2019-05-22 13:37:05 +00:00
|
|
|
Support passing arguments by url (query or fragment) like following
|
2019-05-18 06:27:14 +00:00
|
|
|
examples:
|
|
|
|
|
2019-08-19 11:04:00 +00:00
|
|
|
Passing form data (password must be encoded in base64, privatekey not
|
|
|
|
supported)
|
2019-05-18 06:27:14 +00:00
|
|
|
|
|
|
|
.. code:: bash
|
|
|
|
|
2019-08-10 06:48:58 +00:00
|
|
|
http://localhost:8888/?hostname=xx&username=yy&password=str_base64_encoded
|
2019-05-18 06:27:14 +00:00
|
|
|
|
|
|
|
Passing a terminal background color
|
|
|
|
|
|
|
|
.. code:: bash
|
|
|
|
|
|
|
|
http://localhost:8888/#bgcolor=green
|
|
|
|
|
|
|
|
Passing a user defined title
|
|
|
|
|
|
|
|
.. code:: bash
|
|
|
|
|
2019-05-19 12:19:51 +00:00
|
|
|
http://localhost:8888/?title=my-ssh-server
|
|
|
|
|
|
|
|
Passing an encoding
|
|
|
|
|
|
|
|
.. code:: bash
|
|
|
|
|
|
|
|
http://localhost:8888/#encoding=gbk
|
2019-05-18 06:27:14 +00:00
|
|
|
|
2019-08-19 11:04:00 +00:00
|
|
|
Passing a command executed right after login
|
2019-07-12 12:16:27 +00:00
|
|
|
|
|
|
|
.. code:: bash
|
|
|
|
|
|
|
|
http://localhost:8888/?command=pwd
|
|
|
|
|
2019-09-15 00:39:47 +00:00
|
|
|
Passing a terminal type
|
|
|
|
|
|
|
|
.. code:: bash
|
|
|
|
|
|
|
|
http://localhost:8888/?term=xterm-256color
|
|
|
|
|
2019-04-18 11:54:20 +00:00
|
|
|
Use Docker
|
|
|
|
~~~~~~~~~~
|
|
|
|
|
|
|
|
Start up the app
|
|
|
|
|
|
|
|
::
|
|
|
|
|
|
|
|
docker-compose up
|
|
|
|
|
|
|
|
Tear down the app
|
|
|
|
|
|
|
|
::
|
|
|
|
|
|
|
|
docker-compose down
|
|
|
|
|
2018-08-20 15:09:09 +00:00
|
|
|
Tests
|
|
|
|
~~~~~
|
|
|
|
|
2019-08-19 11:04:00 +00:00
|
|
|
Requirements
|
|
|
|
|
|
|
|
::
|
|
|
|
|
|
|
|
pip install pytest pytest-cov codecov flake8 mock
|
|
|
|
|
2018-08-20 15:33:17 +00:00
|
|
|
Use unittest to run all tests
|
2018-08-20 15:09:09 +00:00
|
|
|
|
|
|
|
::
|
|
|
|
|
|
|
|
python -m unittest discover tests
|
|
|
|
|
2018-08-20 15:33:17 +00:00
|
|
|
Use pytest to run all tests
|
2018-08-20 15:09:09 +00:00
|
|
|
|
|
|
|
::
|
|
|
|
|
|
|
|
python -m pytest tests
|
|
|
|
|
2018-10-22 13:51:51 +00:00
|
|
|
Deployment
|
|
|
|
~~~~~~~~~~
|
|
|
|
|
|
|
|
Running behind an Nginx server
|
|
|
|
|
|
|
|
.. code:: bash
|
|
|
|
|
|
|
|
wssh --address='127.0.0.1' --port=8888 --policy=reject
|
2018-04-26 17:11:16 +00:00
|
|
|
|
2018-08-29 15:12:21 +00:00
|
|
|
.. code:: nginx
|
2018-04-26 17:11:16 +00:00
|
|
|
|
2018-10-22 13:51:51 +00:00
|
|
|
# Nginx config example
|
2018-04-26 17:11:16 +00:00
|
|
|
location / {
|
|
|
|
proxy_pass http://127.0.0.1:8888;
|
|
|
|
proxy_http_version 1.1;
|
|
|
|
proxy_read_timeout 300;
|
|
|
|
proxy_set_header Upgrade $http_upgrade;
|
|
|
|
proxy_set_header Connection "upgrade";
|
|
|
|
proxy_set_header Host $http_host;
|
|
|
|
proxy_set_header X-Real-IP $remote_addr;
|
|
|
|
proxy_set_header X-Real-PORT $remote_port;
|
|
|
|
}
|
|
|
|
|
2018-10-22 13:51:51 +00:00
|
|
|
Running as a standalone server
|
|
|
|
|
|
|
|
.. code:: bash
|
|
|
|
|
2018-10-23 06:10:10 +00:00
|
|
|
wssh --port=8080 --sslport=4433 --certfile='cert.crt' --keyfile='cert.key' --xheaders=False --policy=reject
|
2018-10-22 13:51:51 +00:00
|
|
|
|
2018-04-26 17:11:16 +00:00
|
|
|
Tips
|
2019-05-19 12:19:51 +00:00
|
|
|
~~~~
|
2018-04-26 17:11:16 +00:00
|
|
|
|
2018-10-22 13:51:51 +00:00
|
|
|
- For whatever deployment choice you choose, don't forget to enable
|
|
|
|
SSL.
|
2018-12-15 13:40:16 +00:00
|
|
|
- By default plain http requests from a public network will be either
|
|
|
|
redirected or blocked and being redirected takes precedence over
|
|
|
|
being blocked.
|
2018-04-26 17:11:16 +00:00
|
|
|
- Try to use reject policy as the missing host key policy along with
|
|
|
|
your verified known\_hosts, this will prevent man-in-the-middle
|
|
|
|
attacks. The idea is that it checks the system host keys
|
|
|
|
file("~/.ssh/known\_hosts") and the application host keys
|
|
|
|
file("./known\_hosts") in order, if the ssh server's hostname is not
|
2018-10-23 06:10:10 +00:00
|
|
|
found or the key is not matched, the connection will be aborted.
|
2018-04-26 17:11:16 +00:00
|
|
|
|
|
|
|
.. |Build Status| image:: https://travis-ci.org/huashengdun/webssh.svg?branch=master
|
|
|
|
:target: https://travis-ci.org/huashengdun/webssh
|
|
|
|
.. |codecov| image:: https://codecov.io/gh/huashengdun/webssh/branch/master/graph/badge.svg
|
|
|
|
:target: https://codecov.io/gh/huashengdun/webssh
|
2018-04-27 00:44:22 +00:00
|
|
|
.. |PyPI - Python Version| image:: https://img.shields.io/pypi/pyversions/webssh.svg
|
|
|
|
.. |PyPI| image:: https://img.shields.io/pypi/v/webssh.svg
|
2018-04-26 17:11:16 +00:00
|
|
|
.. |Login| image:: https://github.com/huashengdun/webssh/raw/master/preview/login.png
|
|
|
|
.. |Terminal| image:: https://github.com/huashengdun/webssh/raw/master/preview/terminal.png
|
|
|
|
|