forked from stuicey/SSHy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
index.html
470 lines (423 loc) · 16.9 KB
/
index.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
<!doctype html>
<html>
<head>
<meta charset="utf-8">
<title>SSHy.us</title>
<script type="text/javascript">
var wsproxyURL = "localhost"
var wsproxyPorts = {"ws":5999, "wss":6001}
var wsproxyProto = "ws"
var ws, transport, settings, term = null;
// Test IE 11
if (window.msCrypto){
// Redirect window.crypto.getRandomValues() -> window.msCrypto.getRandomValues()
window.crypto = {}
window.crypto.getRandomValues = function(a) { return window.msCrypto.getRandomValues(a); }
// PolyFill Uint8Array.slice() for IE 11 for sjcl AES
if (!Uint8Array.prototype.slice) {
Object.defineProperty(Uint8Array.prototype, 'slice', {
value: Array.prototype.slice
});
}
}
var resizeInterval
</script>
<link rel="stylesheet" href="css/xterm.css" async/>
<link rel="stylesheet" href="css/main.css" async/>
<link rel="stylesheet" href="css/fonts.css" async/>
<!-- Cryptography & Terminal Libraries -->
<script type="text/javascript" src="js/combinedLibs.comb.js" async></script>
<script type="text/javascript">
window.onload = function() {
document.getElementById('login_cred').style.display = "block";
// Sets the default colorScheme to material
settings = new SSHyClient.settings();
settings.setColorScheme(1);
// Apply fit addon
fit.apply(Terminal)
// Build the default wsProxy URL for display on sidenav
buildWSProxyURL()
// Connect upon hitting Enter from the password field
document.getElementById("password").addEventListener("keyup", function(event) {
if(event.key !== "Enter") return;
document.getElementById("connect").click();
event.preventDefault();
});
};
// Run every time the webpage is resized
window.onresize = function() {
clearTimeout(resizeInterval);
resizeInterval = setTimeout(resize, 400);
}
// Recalculates the terminal Columns / Rows and sends new size to SSH server + xtermjs
function resize() {
if (term) {
term.fit()
}
}
// Build the entire websocket url eg (wss://localhost:5999/) based on http protocol
function buildWSProxyURL(portPassed) {
// Decide if we're using secure ws or not
if(window.location.protocol == "https:"){
wsproxyProto = "wss"
}
var port
if (portPassed){
port = ""
} else {
port = ':' + wsproxyPorts[wsproxyProto]
}
// Build the wsproxyURL up
wsproxyURL = wsproxyProto + '://' + wsproxyURL + port + '/'
document.getElementById("websockURL").value = wsproxyURL
}
// Changes the websocket proxy URL ** BEFORE ** connection ONLY
function modProxyURL(newURL){
if (!ws) {
// Strip it down to barebones URL:PORT(optional)
matches = /^w?s{0,2}:?\/{0,2}(([a-z0-9]+\.)*[a-z0-9]+\.?[a-z]+)\:?([0-9]{1,5})?/g.exec(newURL)
var port = ""
if (newURL.match(":")) {
port = ":"+matches[matches.length-1]
}
wsproxyURL = matches[1] + port;
buildWSProxyURL(port)
}
}
// Toggles the settings navigator
function toggleNav(size) {
document.getElementById("settingsNav").style.width = size + 'px';
settings.sidenavElementState = size;
// We need to update the network traffic whenever the nav is re-opened
if(size && transport){
settings.setNetTraffic(transport.parceler.recieveData, true);
settings.setNetTraffic(transport.parceler.transmitData, false);
}
var element = document.getElementById("gear").style;
element.visibility = element.visibility === "hidden" ? "visible" : "hidden";
}
// Rudimentary checks that an IP address is external and is a valid hostname or IP address
function validate(id, text) {
if (!text) {
document.getElementById(id).style.borderBottom = 'solid 2px #ff4d4d'
} else {
if (id == "ipaddress") {
// incase we have a error for the port
if (text.includes(':')) {
if (!validate_port(text.split(':')[1])) {
document.getElementById(id).style.borderBottom = 'solid 2px #ff4d4d'
return
} else {
document.getElementById(id).style.borderBottom = 'solid 2px #c9c9c9'
document.getElementById('failure').style.display = "none"
}
} else {
// if we're not doing ports then hide the failure message
document.getElementById('failure').style.display = "none"
}
// test for valid domain name
if (!/^([a-z0-9]+\.)*[a-z0-9]+\.[a-z]+(\:[0-9]{1,5})?$/.test(text)) {
if (check_internal(text)) {
display_error("Be aware - IP addresses are resolved at the websocket proxy")
}
// test ip aswell.
if (!/^((25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)(\.|$|(\:[0-9]{1,5}))){4}$/.test(text)) {
document.getElementById(id).style.borderBottom = 'solid 2px #ff4d4d'
return
} else {
if (check_internal(text)) {
display_error("Could no resolve hostname: Please use an external address")
}
document.getElementById(id).style.borderBottom = 'solid 2px #c9c9c9'
return
}
document.getElementById(id).style.borderBottom = 'solid 2px #ff4d4d'
return
}
}
document.getElementById(id).style.borderBottom = 'solid 2px #c9c9c9'
}
}
// Validates the port is 0 > port < 65536
function validate_port(port_num) {
if (port_num > 0 && port_num < 65536) {
return port_num
} else {
display_error("Invalid port: port must be between 1 - 65535")
validate('ipaddress', '')
}
}
// checks for 10.*.*.*, 192.168.*.*, 172.16.*.*, 127.0.0.1 & loocalhost
function check_internal(ip_address) {
if (/10\.\d+\.\d+\.\d+/.test(ip_address) || /192\.168\.\d+\.\d+/.test(ip_address) || /172\.16\.\d+\.\d+/.test(ip_address)) {
return true
} else if (ip_address == '127.0.0.1' || ip_address == 'localhost') {
return true
}
return false
}
// Displays a given err on the page
function display_error(err) {
// remove the loading cog and set the 'connecting' to connect
document.getElementById('load-container').style.display = "none"
document.getElementById('connect').value = "connect"
document.getElementById('failure').innerText = err
document.getElementById('failure').style.display = "block"
}
// Starts the SSH client in scripts/transport.js
function startSSHy() {
var html_ipaddress = document.getElementById('ipaddress').value
var termUsername = document.getElementById('username').value
var termPassword = document.getElementById('password').value
// find the port number
if (html_ipaddress.includes(':')) {
var split = html_ipaddress.split(':')
html_ipaddress = split[0]
html_port = validate_port(split[1])
} else {
html_port = 22
}
if (termUsername.length == 0 || termPassword.length == 0) {
validate('username', termUsername)
validate('password', termPassword)
return
}
// Error checking is done so remove any currently displayed errors
document.getElementById('failure').style.display = "none"
document.getElementById('connect').value = 'Connecting...'
document.getElementById('load-container').style.display = "block"
// Disable websocket proxy modifications
document.getElementById('websockURL').disabled = true
// Initialise the window title
document.title = "SSHy Client";
// Opens the websocket!
wsproxyURL += html_ipaddress + ':' + html_port
ws = new WebSocket(wsproxyURL , 'base64');
// Sets up websocket listeners
ws.onopen = function(e) {
transport = new SSHyClient.Transport(ws, settings);
transport.auth.termUsername = termUsername;
transport.auth.termPassword = termPassword;
transport.auth.hostname = html_ipaddress;
};
// Send all recieved messages to SSHyClient.Transport.handle()
ws.onmessage = function(e) {
// Convert the recieved data from base64 to a string
transport.parceler.handle(atob(e.data));
};
// Whenever the websocket is closed make sure to display an error if appropriate
ws.onclose = function(e) {
// Set the sidenav websocket proxy color to yellow
document.getElementById('websockURL').classList.remove('brightgreen')
document.getElementById('websockURL').classList.add('brightyellow')
if (term) {
// Don't display an error if SSH transport has already detected a graceful exit
if (transport.closing) {
return;
}
term.write('\n\n\rWebsocket connection to ' + transport.auth.hostname + ' was unexpectedly closed.');
// If there is no keepAliveInterval then inform users they can use it
if (!settings.keepAliveInterval) {
term.write('\n\n\rThis was likely caused by he remote SSH server timing out the session due to inactivity.\r\n- Session Keep Alive interval can be set in the settings to prevent this behaviour.');
}
} else {
// Since no terminal exists we need to initialse one before being able to write the error
termInit();
term.write('WebSocket connection failed: Error in connection establishment: code ' + e.code);
}
};
// Just a little abstraction from ws.send
ws.sendB64 = function(e){
this.send(btoa(e));
transport.parceler.transmitData += e.length;
settings.setNetTraffic(transport.parceler.transmitData, false);
};
// Set the sidenav websocket proxy color to green
document.getElementById('websockURL').classList.add('brightgreen')
}
// Initialises xtermjs
function termInit() {
// Define the terminal rows/cols
term = new Terminal({
cols: 80,
rows: 24
});
// start xterm.js
term.open(document.getElementById('terminal'), true);
term.fit()
term.focus()
// set the color scheme to whatever the user's changed it to in the mean time
var colName = document.getElementById('currentColor').innerHTML
for(i = 0; i < settings.colorSchemes.length; i++) {
if ( settings.colorSchemes[i][0] == colName ) {
settings.setColorScheme(i)
break
}
}
// clear the modal elements on screen
document.getElementById('load-container').style.display = "none";
document.getElementById('login_cred').style.display = "none";
}
// Binds custom listener functions to xtermjs's Terminal object
function startxtermjs() {
termInit();
// sets up some listeners for the terminal (keydown, paste)
term.textarea.onkeydown = function(e) {
// Sanity Checks
if (!ws || !transport || transport.auth.failedAttempts >= 5 || transport.auth.awaitingAuthentication) {
return;
}
// So we don't spam single control characters
if (e.key.length > 1 && (e.altKey || e.ctrlKey || e.metaKey || e.shiftKey) && e.key != "Backspace") {
return;
}
// We've already authenticated so now any keypress is a command for the SSH server
var command, pressedKey;
/** IE isn't very good so it displays one character keys as full names in .key
EG - e.key = " " to e.key = "Spacebar"
so assuming .char is one character we'll use that instead **/
if (e.char && e.char.length == 1) {
pressedKey = e.char;
} else {
pressedKey = e.key
}
// Decides if the keypress is an alphanumeric character or needs escaping
if (pressedKey.length == 1 && (!(e.altKey || e.ctrlKey || e.metaKey) || (e.altKey && e.ctrlKey))) {
command = pressedKey;
} else if (pressedKey.length == 1 && (e.shiftKey && e.ctrlKey)) {
// allows ctrl + shift + v for pasting
if (pressedKey != 'V') {
e.preventDefault();
return;
}
} else {
//xtermjs is kind enough to evaluate our special characters instead of having to translate every char ourself
command = term._evaluateKeyEscapeSequence(e).key;
}
// Decide if we're going to locally' echo this key or not
if (settings.localEcho) {
settings.parseKey(e);
}
/* Regardless of local echo we still want a reply to confirm / update terminal
could be controversial? but putty does this too (each key press shows up twice)
Instead we're checking the our locally echoed key and replacing it if the
recieved key !== locally echoed key */
return command === null ? null : transport.expect_key(command);
};
term.textarea.onpaste = function(ev) {
var text
// Yay IE11 stuff!
if ( window.clipboardData && window.clipboardData.getData ) {
text = window.clipboardData.getData('Text')
} else if ( ev.clipboardData && ev.clipboardData.getData ) {
text = ev.clipboardData.getData('text/plain');
}
if (text) {
// Just don't allow more than 1 million characters to be pasted.
if(text.length < 1000000){
if (text.length > 5000) {
// If its a long string then chunk it down to reduce load on SSHyClient.parceler
text = splitSlice(text);
for (var i = 0; i < text.length; i++) {
transport.expect_key(text[i]);
}
return;
}
transport.expect_key(text);
} else {
alert('Error: Pasting large strings is not permitted.');
}
}
};
}
</script>
</head>
<body>
<div id="terminal"></div>
<div id="settingsNav" class="sidenav" style="width:0;">
<a href="javascript:;" class="closebtn" onclick="toggleNav(0)">×</a>
<span class="title large">Terminal Options</span>
<hr>
<span class="title" style="padding-top:20px">Font Size</span>
<a class="leftarrow" href="javascript:;" onclick="settings.modFontSize(-1)"><--</a>
<span class="middle" id="currentFontSize">16px</span>
<a class="rightarrow" href="javascript:;" onclick="settings.modFontSize(1)">--></a>
<span class="title" style="padding-top:40px">Terminal Size</span>
<span class="leftarrow">Cols:
<input type="number" id="termCols" oninput="settings.modTerm(0, this.value)">
</span>
<span class="rightarrow">Rows:
<input type="number" id="termRows" oninput="settings.modTerm(1, this.value)">
</span>
<span class="title" style="padding-top:60px;">Local Echo</span>
<a class="leftarrow" href="javascript:;" onclick="settings.setLocalEcho(-1)"><--</a>
<a class="rightarrow" href="javascript:;" onclick="settings.setLocalEcho(1)">--></a>
<div class="fileUpload btn btn-primary nomargin">
<span class="tooltiptext" style="visibility:hidden;" id="autoEchoState">State: Disabled</span>
<span class="middle" id="currentLEcho">Force Off</span>
</div>
<span class="title" style="padding-top:50px">Colours</span>
<a class="leftarrow" href="javascript:;" onclick="settings.cycleColorSchemes(0)"><--</a>
<span class="middle" id="currentColor">Monokai</span>
<a class="rightarrow" href="javascript:;" onclick="settings.cycleColorSchemes(1)">--></a>
<div class="fileUpload btn btn-primary">
<span class="tooltiptext">Format: Xresources</span>
<span class="middle" style="width:220px;">Upload</span>
<input type="file" title=" " id="Xresources" class="upload" onchange="settings.importXresources()" />
</div>
<span class="title" style="padding-top:20px;">Keep Alive</span>
<div class="fileUpload btn btn-primary">
<span class="tooltiptext">0 to disable</span>
<input type="number" class="large" id="keepAlive" onchange="settings.setKeepAlive(this.value);" placeholder="0">
<span style="font-size:16px;"> seconds</span>
</div>
<span class="title" style="padding-top:20px;">Network Traffic</span>
<div class="netTraffic">
<span class="leftarrow brightgreen">rx: <span id="rxTraffic">0Bytes</span></span>
<span class="rightarrow brightyellow">tx: <span id="txTraffic">0Bytes</span></span>
</div>
<div id="hostKey">
<span class="title" style="padding-top:20px;">Host Key</span>
<span id="hostKeyImg" class="hostKeyImg"></span>
</div>
<span class="footer">
<span class="title">Websocket Proxy</span>
<div class="fileUpload btn btn-primary">
<span class="tooltiptext">Websocket Proxy URL</span>
<input type="text" class="large" id="websockURL" onchange="modProxyURL(this.value)">
</div>
</span>
</div>
<span id="gear" class="gear" style="visibility:visible;" onclick="toggleNav(250)">⚙</span>
<div id="login_cred" class="modal">
<div class="modal-content">
<div class="modal-header">
<h2><span class="icon-lock"></span> SSHv2 Login</h2>
</div>
<div class="modal-body">
<table style="width: 100%">
<tr>
<td><span class="icon-address-book"></span> IP Address:</td>
<td><input type="text" id="ipaddress" placeholder="sshy.us(:22)" style="width: 90%" onblur="validate(this.id, this.value)" autofocus></td>
</tr>
<tr>
<td><span class="icon-user-circle-o"></span> Username:</td>
<td><input type="text" id="username" placeholder="username" style="width: 90%" onblur="validate(this.id, this.value)"></td>
</tr>
<tr>
<td><span class="icon-eye"></span> Password:</td>
<td><input type="password" id="password" style="width: 90%" onblur="validate(this.id, this.value)"></td>
</tr>
</table>
<i id="failure" class="failure"></i>
<input type="button" id="connect" class="button" value="Connect" onclick="startSSHy()">
</div>
</div>
<div id="load-container" class="load-container">
<svg id="load" x="0px" y="0px" viewBox="0 0 150 150">
<circle id="loading-inner" cx="75" cy="75" r="60"></circle>
</svg>
</div>
</div>
</body>
</html>