Added automatic file upload resume & skip.
This commit is contained in:
parent
fd33c7decd
commit
b260f74053
|
@ -1865,8 +1865,7 @@ function onTunnelData(data) {
|
||||||
if (this.httprequest.state == 0) {
|
if (this.httprequest.state == 0) {
|
||||||
// Check if this is a relay connection
|
// Check if this is a relay connection
|
||||||
if ((data == 'c') || (data == 'cr')) { this.httprequest.state = 1; /*sendConsoleText("Tunnel #" + this.httprequest.index + " now active", this.httprequest.sessionid);*/ }
|
if ((data == 'c') || (data == 'cr')) { this.httprequest.state = 1; /*sendConsoleText("Tunnel #" + this.httprequest.index + " now active", this.httprequest.sessionid);*/ }
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
// Handle tunnel data
|
// Handle tunnel data
|
||||||
if (this.httprequest.protocol == 0) { // 1 = Terminal (admin), 2 = Desktop, 5 = Files, 6 = PowerShell (admin), 7 = Plugin Data Exchange, 8 = Terminal (user), 9 = PowerShell (user), 10 = FileTransfer
|
if (this.httprequest.protocol == 0) { // 1 = Terminal (admin), 2 = Desktop, 5 = Files, 6 = PowerShell (admin), 7 = Plugin Data Exchange, 8 = Terminal (user), 9 = PowerShell (user), 10 = FileTransfer
|
||||||
// Take a look at the protocol
|
// Take a look at the protocol
|
||||||
|
@ -2648,7 +2647,7 @@ function onTunnelData(data) {
|
||||||
var filepath = cmd.name ? obj.path.join(cmd.path, cmd.name) : cmd.path;
|
var filepath = cmd.name ? obj.path.join(cmd.path, cmd.name) : cmd.path;
|
||||||
this.httprequest.uploadFilePath = filepath;
|
this.httprequest.uploadFilePath = filepath;
|
||||||
this.httprequest.uploadFileSize = 0;
|
this.httprequest.uploadFileSize = 0;
|
||||||
try { this.httprequest.uploadFile = fs.openSync(filepath, 'wbN'); } catch (e) { this.write(Buffer.from(JSON.stringify({ action: 'uploaderror', reqid: cmd.reqid }))); break; }
|
try { this.httprequest.uploadFile = fs.openSync(filepath, cmd.append ? 'abN' : 'wbN'); } catch (e) { this.write(Buffer.from(JSON.stringify({ action: 'uploaderror', reqid: cmd.reqid }))); break; }
|
||||||
this.httprequest.uploadFileid = cmd.reqid;
|
this.httprequest.uploadFileid = cmd.reqid;
|
||||||
if (this.httprequest.uploadFile) { this.write(Buffer.from(JSON.stringify({ action: 'uploadstart', reqid: this.httprequest.uploadFileid }))); }
|
if (this.httprequest.uploadFile) { this.write(Buffer.from(JSON.stringify({ action: 'uploadstart', reqid: this.httprequest.uploadFileid }))); }
|
||||||
break;
|
break;
|
||||||
|
@ -2681,6 +2680,15 @@ function onTunnelData(data) {
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case 'uploadhash':
|
||||||
|
{
|
||||||
|
// Hash a file
|
||||||
|
var filepath = cmd.name ? obj.path.join(cmd.path, cmd.name) : cmd.path;
|
||||||
|
var h = null;
|
||||||
|
try { h = getSHA384FileHash(filepath); } catch (ex) { sendConsoleText(ex); }
|
||||||
|
this.write(Buffer.from(JSON.stringify({ action: 'uploadhash', reqid: cmd.reqid, path: cmd.path, name: cmd.name, tag: cmd.tag, hash: (h ? h.toString('hex') : null) })));
|
||||||
|
break
|
||||||
|
}
|
||||||
case 'copy':
|
case 'copy':
|
||||||
{
|
{
|
||||||
// Copy a bunch of files from scpath to dspath
|
// Copy a bunch of files from scpath to dspath
|
||||||
|
|
|
@ -1,20 +1,20 @@
|
||||||
<html><head></head><body><div>[[[SERVERNAME]]] - Device Notification</div>
|
<html><head></head><body><div>[[[SERVERNAME]]] - Apparaatmelding</div>
|
||||||
<div style="font-family:Arial,Helvetica,sans-serif">
|
<div style="font-family:Arial,Helvetica,sans-serif">
|
||||||
<table style="background-color:#003366;color:lightgray;width:100%" cellpadding="8">
|
<table style="background-color:#003366;color:lightgray;width:100%" cellpadding="8">
|
||||||
<tbody><tr>
|
<tbody><tr>
|
||||||
<td>
|
<td>
|
||||||
<b style="font-size:20px;font-family:Arial,Helvetica,sans-serif">[[[SERVERNAME]]] - Device Notification</b>
|
<b style="font-size:20px;font-family:Arial,Helvetica,sans-serif">[[[SERVERNAME]]] - Apparaatmelding</b>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody></table>
|
</tbody></table>
|
||||||
<area-header>
|
<area-header>
|
||||||
<p>
|
<p>
|
||||||
The following devices have changed their connection state.
|
De volgende apparaten hebben hun verbindingsstatus gewijzigd.
|
||||||
</p>
|
</p>
|
||||||
</area-header>
|
</area-header>
|
||||||
<area-connections>
|
<area-connections>
|
||||||
<p>
|
<p>
|
||||||
Connected devices:
|
Verbonden apparaten:
|
||||||
</p>
|
</p>
|
||||||
<p><b>
|
<p><b>
|
||||||
[[[CONNECTIONS]]]
|
[[[CONNECTIONS]]]
|
||||||
|
@ -22,7 +22,7 @@
|
||||||
</area-connections>
|
</area-connections>
|
||||||
<area-disconnections>
|
<area-disconnections>
|
||||||
<p>
|
<p>
|
||||||
Disconnected devices:
|
Losgekoppelde apparaten:
|
||||||
</p>
|
</p>
|
||||||
<p><b>
|
<p><b>
|
||||||
[[[DISCONNECTIONS]]]
|
[[[DISCONNECTIONS]]]
|
||||||
|
@ -30,7 +30,7 @@
|
||||||
</area-disconnections>
|
</area-disconnections>
|
||||||
<area-footer>
|
<area-footer>
|
||||||
<p>
|
<p>
|
||||||
To unsubscribe, <a href="[[[SERVERURL]]][[[UNSUBSCRIBELINK]]]">Klik hier</a> within 1 hour of getting this message.
|
Uitschrijven, <a href="[[[SERVERURL]]][[[UNSUBSCRIBELINK]]]">Klik hier</a> binnen 1 uur na ontvangst van dit bericht.
|
||||||
</p>
|
</p>
|
||||||
</area-footer>
|
</area-footer>
|
||||||
</div></body></html>
|
</div></body></html>
|
|
@ -1,22 +1,22 @@
|
||||||
[[[SERVERNAME]]] - Device Notification
|
[[[SERVERNAME]]] - Apparaatmelding
|
||||||
~<area-header>
|
~<area-header>
|
||||||
The following devices have changed their connection state.
|
De volgende apparaten hebben hun verbindingsstatus gewijzigd.
|
||||||
~</area-header>
|
~</area-header>
|
||||||
~<area-connections>
|
~<area-connections>
|
||||||
~
|
~
|
||||||
Connected devices:
|
Verbonden apparaten:
|
||||||
~
|
~
|
||||||
~[[[CONNECTIONS]]]
|
~[[[CONNECTIONS]]]
|
||||||
~
|
~
|
||||||
~</area-connections>
|
~</area-connections>
|
||||||
~<area-disconnections>
|
~<area-disconnections>
|
||||||
~
|
~
|
||||||
Disconnected devices:
|
Losgekoppelde apparaten:
|
||||||
~
|
~
|
||||||
~[[[DISCONNECTIONS]]]
|
~[[[DISCONNECTIONS]]]
|
||||||
~
|
~
|
||||||
~</area-disconnections>
|
~</area-disconnections>
|
||||||
~<area-footer>
|
~<area-footer>
|
||||||
|
|
||||||
To unsubscribe, load this link within 1 hour of getting this message: [[[SERVERURL]]][[[UNSUBSCRIBELINK]]]
|
Om je af te melden, open je deze link binnen 1 uur nadat je dit bericht hebt ontvangen: [[[SERVERURL]]][[[UNSUBSCRIBELINK]]]
|
||||||
~</area-footer>
|
~</area-footer>
|
|
@ -43,19 +43,19 @@ function ArrayElementMove(arr, from, to) { arr.splice(to, 0, arr.splice(from, 1)
|
||||||
// Print object for HTML
|
// Print object for HTML
|
||||||
function ObjectToStringEx(x, c) {
|
function ObjectToStringEx(x, c) {
|
||||||
var r = "";
|
var r = "";
|
||||||
if (x != 0 && (!x || x == null)) return "(Null)";
|
if (x != 0 && (!x || x == null)) return '(Null)';
|
||||||
if (x instanceof Array) { for (var i in x) { r += '<br />' + gap(c) + "Item #" + i + ": " + ObjectToStringEx(x[i], c + 1); } }
|
if (x instanceof Array) { for (var i in x) { r += '<br />' + gap(c) + 'Item #' + i + ": " + ObjectToStringEx(x[i], c + 1); } }
|
||||||
else if (x instanceof Object) { for (var i in x) { r += '<br />' + gap(c) + i + " = " + ObjectToStringEx(x[i], c + 1); } }
|
else if (x instanceof Object) { for (var i in x) { r += '<br />' + gap(c) + i + ' = ' + ObjectToStringEx(x[i], c + 1); } }
|
||||||
else { r += EscapeHtml(x); }
|
else { r += EscapeHtml(x); }
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Print object for console
|
// Print object for console
|
||||||
function ObjectToStringEx2(x, c) {
|
function ObjectToStringEx2(x, c) {
|
||||||
var r = "";
|
var r = '';
|
||||||
if (x != 0 && (!x || x == null)) return "(Null)";
|
if (x != 0 && (!x || x == null)) return '(Null)';
|
||||||
if (x instanceof Array) { for (var i in x) { r += '\r\n' + gap2(c) + "Item #" + i + ": " + ObjectToStringEx2(x[i], c + 1); } }
|
if (x instanceof Array) { for (var i in x) { r += '\r\n' + gap2(c) + 'Item #' + i + ': ' + ObjectToStringEx2(x[i], c + 1); } }
|
||||||
else if (x instanceof Object) { for (var i in x) { r += '\r\n' + gap2(c) + i + " = " + ObjectToStringEx2(x[i], c + 1); } }
|
else if (x instanceof Object) { for (var i in x) { r += '\r\n' + gap2(c) + i + ' = ' + ObjectToStringEx2(x[i], c + 1); } }
|
||||||
else { r += EscapeHtml(x); }
|
else { r += EscapeHtml(x); }
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
@ -70,7 +70,7 @@ function ObjectToString2(x) { return ObjectToStringEx2(x, 0); }
|
||||||
|
|
||||||
// Convert a hex string to a raw string
|
// Convert a hex string to a raw string
|
||||||
function hex2rstr(d) {
|
function hex2rstr(d) {
|
||||||
if (typeof d != "string" || d.length == 0) return '';
|
if (typeof d != 'string' || d.length == 0) return '';
|
||||||
var r = '', m = ('' + d).match(/../g), t;
|
var r = '', m = ('' + d).match(/../g), t;
|
||||||
while (t = m.shift()) r += String.fromCharCode('0x' + t);
|
while (t = m.shift()) r += String.fromCharCode('0x' + t);
|
||||||
return r
|
return r
|
||||||
|
@ -107,7 +107,7 @@ function random(max) { return Math.floor(Math.random() * max); }
|
||||||
function trademarks(x) { return x.replace(/\(R\)/g, '®').replace(/\(TM\)/g, '™'); }
|
function trademarks(x) { return x.replace(/\(R\)/g, '®').replace(/\(TM\)/g, '™'); }
|
||||||
|
|
||||||
// Pad a number with zeros on the left
|
// Pad a number with zeros on the left
|
||||||
function zeroPad(num, c) { if (c == null) { c = 2; } var s = "00000000" + num; return s.substr(s.length - c); }
|
function zeroPad(num, c) { if (c == null) { c = 2; } var s = '00000000' + num; return s.substr(s.length - c); }
|
||||||
|
|
||||||
// String validation
|
// String validation
|
||||||
function isAlphaNumeric(str) { if (typeof str == 'number') { return true; } return (str.match(/^[A-Za-z0-9]+$/) != null); };
|
function isAlphaNumeric(str) { if (typeof str == 'number') { return true; } return (str.match(/^[A-Za-z0-9]+$/) != null); };
|
||||||
|
@ -135,10 +135,10 @@ function parseUriArgs() {
|
||||||
// From: https://stackoverflow.com/questions/5573096/detecting-webp-support
|
// From: https://stackoverflow.com/questions/5573096/detecting-webp-support
|
||||||
function check_webp_feature(feature, callback) {
|
function check_webp_feature(feature, callback) {
|
||||||
var kTestImages = {
|
var kTestImages = {
|
||||||
lossy: "UklGRiIAAABXRUJQVlA4IBYAAAAwAQCdASoBAAEADsD+JaQAA3AAAAAA"//,
|
lossy: 'UklGRiIAAABXRUJQVlA4IBYAAAAwAQCdASoBAAEADsD+JaQAA3AAAAAA'//,
|
||||||
//lossless: "UklGRhoAAABXRUJQVlA4TA0AAAAvAAAAEAcQERGIiP4HAA==",
|
//lossless: 'UklGRhoAAABXRUJQVlA4TA0AAAAvAAAAEAcQERGIiP4HAA==',
|
||||||
//alpha: "UklGRkoAAABXRUJQVlA4WAoAAAAQAAAAAAAAAAAAQUxQSAwAAAARBxAR/Q9ERP8DAABWUDggGAAAABQBAJ0BKgEAAQAAAP4AAA3AAP7mtQAAAA==",
|
//alpha: 'UklGRkoAAABXRUJQVlA4WAoAAAAQAAAAAAAAAAAAQUxQSAwAAAARBxAR/Q9ERP8DAABWUDggGAAAABQBAJ0BKgEAAQAAAP4AAA3AAP7mtQAAAA==',
|
||||||
//animation: "UklGRlIAAABXRUJQVlA4WAoAAAASAAAAAAAAAAAAQU5JTQYAAAD/////AABBTk1GJgAAAAAAAAAAAAAAAAAAAGQAAABWUDhMDQAAAC8AAAAQBxAREYiI/gcA"
|
//animation: 'UklGRlIAAABXRUJQVlA4WAoAAAASAAAAAAAAAAAAQU5JTQYAAAD/////AABBTk1GJgAAAAAAAAAAAAAAAAAAAGQAAABWUDhMDQAAAC8AAAAQBxAREYiI/gcA'
|
||||||
};
|
};
|
||||||
var img = new Image();
|
var img = new Image();
|
||||||
img.onload = function () {
|
img.onload = function () {
|
||||||
|
@ -148,5 +148,5 @@ function check_webp_feature(feature, callback) {
|
||||||
img.onerror = function () {
|
img.onerror = function () {
|
||||||
callback(feature, false);
|
callback(feature, false);
|
||||||
};
|
};
|
||||||
img.src = "data:image/webp;base64," + kTestImages[feature];
|
img.src = 'data:image/webp;base64,' + kTestImages[feature];
|
||||||
}
|
}
|
|
@ -68697,4 +68697,4 @@
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
|
@ -9689,7 +9689,7 @@
|
||||||
function p13folderup(x) {
|
function p13folderup(x) {
|
||||||
if (x == null) { p13filetreelocation.pop(); } else { while (p13filetreelocation.length > x) { p13filetreelocation.pop(); } }
|
if (x == null) { p13filetreelocation.pop(); } else { while (p13filetreelocation.length > x) { p13filetreelocation.pop(); } }
|
||||||
p13targetpath = p13filetreelocation.join('/');
|
p13targetpath = p13filetreelocation.join('/');
|
||||||
files.sendText({ action: 'ls', reqid: 1, path: p13targetpath });
|
if (files) { files.sendText({ action: 'ls', reqid: 1, path: p13targetpath }); }
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -10004,6 +10004,12 @@
|
||||||
p13uploadNextFile();
|
p13uploadNextFile();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Perform SHA-384 hashing
|
||||||
|
const byteToHex = [];
|
||||||
|
for (var n = 0; n <= 0xff; ++n) { var hexOctet = n.toString(16).padStart(2, '0'); byteToHex.push(hexOctet); }
|
||||||
|
function arrayBufferToHex(arrayBuffer) { return Array.prototype.map.call( new Uint8Array(arrayBuffer), n => byteToHex[n] ).join(''); }
|
||||||
|
function performHash(data, f) { window.crypto.subtle.digest('SHA-384', data).then(function (v) { f(arrayBufferToHex(v)); }, function() { f(null); }); }
|
||||||
|
|
||||||
// Push the next file
|
// Push the next file
|
||||||
function p13uploadNextFile() {
|
function p13uploadNextFile() {
|
||||||
uploadFile.xfilePtr++;
|
uploadFile.xfilePtr++;
|
||||||
|
@ -10018,7 +10024,17 @@
|
||||||
uploadFile.xreader = new FileReader();
|
uploadFile.xreader = new FileReader();
|
||||||
uploadFile.xreader.onload = function () {
|
uploadFile.xreader.onload = function () {
|
||||||
uploadFile.xdata = uploadFile.xreader.result;
|
uploadFile.xdata = uploadFile.xreader.result;
|
||||||
files.sendText(JSON.stringify({ action: 'upload', reqid: uploadFile.xfilePtr, path: uploadFile.xpath, name: file.name, size: uploadFile.xdata.byteLength }));
|
|
||||||
|
// If the remote file already exists and is smaller then our file, see if we can resume the trasfer
|
||||||
|
var f = null;
|
||||||
|
for (var i in p13filetree.dir) { if (p13filetree.dir[i].n == file.name) { f = p13filetree.dir[i]; } }
|
||||||
|
if ((f != null) && (f.s <= uploadFile.xreader.result.byteLength)) {
|
||||||
|
performHash(uploadFile.xreader.result.slice(0, f.s), function(hash) {
|
||||||
|
files.sendText(JSON.stringify({ action: 'uploadhash', reqid: uploadFile.xfilePtr, path: uploadFile.xpath, name: file.name, tag: { h: hash.toUpperCase(), s: f.s, skip: f.s == uploadFile.xreader.result.byteLength } }));
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
files.sendText(JSON.stringify({ action: 'upload', reqid: uploadFile.xfilePtr, path: uploadFile.xpath, name: file.name, size: uploadFile.xdata.byteLength }));
|
||||||
|
}
|
||||||
};
|
};
|
||||||
uploadFile.xreader.readAsArrayBuffer(file);
|
uploadFile.xreader.readAsArrayBuffer(file);
|
||||||
} else {
|
} else {
|
||||||
|
@ -10052,6 +10068,22 @@
|
||||||
case 'uploadack': { p13uploadNextPart(false); break; }
|
case 'uploadack': { p13uploadNextPart(false); break; }
|
||||||
case 'uploaddone': { if (uploadFile.xfiles.length > uploadFile.xfilePtr + 1) { p13uploadNextFile(); } else { p13uploadFileTransferDone(); } break; }
|
case 'uploaddone': { if (uploadFile.xfiles.length > uploadFile.xfilePtr + 1) { p13uploadNextFile(); } else { p13uploadFileTransferDone(); } break; }
|
||||||
case 'uploaderror': { p13uploadFileCancel(); break; }
|
case 'uploaderror': { p13uploadFileCancel(); break; }
|
||||||
|
case 'uploadhash': {
|
||||||
|
var file = uploadFile.xfiles[uploadFile.xfilePtr];
|
||||||
|
if (file) {
|
||||||
|
if (cmd.tag.h === cmd.hash) {
|
||||||
|
if (cmd.tag.skip) {
|
||||||
|
p13uploadNextFile();
|
||||||
|
} else {
|
||||||
|
uploadFile.xptr = cmd.tag.s;
|
||||||
|
files.sendText(JSON.stringify({ action: 'upload', reqid: uploadFile.xfilePtr, path: uploadFile.xpath, name: file.name, size: uploadFile.xdata.byteLength, append: true }));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
files.sendText(JSON.stringify({ action: 'upload', reqid: uploadFile.xfilePtr, path: uploadFile.xpath, name: file.name, size: uploadFile.xdata.byteLength, append: false }));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue