- numerous bug fixes - wtv-mail system - user account updates - viewergen alpha (experimental webtv viewer patcher on pc_service) - implement wtv-favorites with huge help from @JarHead4 - add wtv-ticket store api - Bump vm2 from 3.9.5 to 3.9.7 in /zefie_wtvp_minisrv - fix bf0app default rom - Add wtv-1800 service to wtv-1800:/noflash - handle webtvism: - allow get/post variables to be the same name multiple times - rather than overwrite, the server will now change the variable from a string to an array. - Rewrite script processing a bit - Instead of using eval() we now use a proper VM Context - As a result, any scripting errors will now give a more useful filename and line number. - However, some things may break, if they are dependant on variables we are not allowing in the context. - BREAKING CHANGES: - `ssid_sessions[socket.ssid]` is now `session_data` - `require` is no longer allowed in user scripts - add star service - change how we handle modules for services in the VM - fixed wtv-disk:/sync always failed the first time - implement production-like wtv-star handling (when a service port becomes unavailable, it requests the url over the wtv-star port to show an error page) - renamed WTVDownloadList.js to WTVDisk.js - a bit more work on WTVNews (created class) - probably more stuff I can't remember
357 lines
14 KiB
JavaScript
357 lines
14 KiB
JavaScript
/**
|
|
* wtv/download-list creation helper class
|
|
* By: zefie
|
|
*/
|
|
class WTVDownloadList {
|
|
|
|
download_list = "";
|
|
service_name = "";
|
|
content_type = "wtv/download-list";
|
|
wtvshared = null;
|
|
clientShowAlert = null;
|
|
minisrv_config = [];
|
|
|
|
/**
|
|
* Constructs the WTVDownloadList Class
|
|
* @param {string} service_name Service name to use in wtv-urls
|
|
*/
|
|
constructor(minisrv_config, service_name = "wtv-disk") {
|
|
var { WTVShared, clientShowAlert } = require("./WTVShared.js");
|
|
this.minisrv_config = minisrv_config;
|
|
this.wtvshared = new WTVShared(minisrv_config);
|
|
this.clientShowAlert = clientShowAlert;
|
|
this.service_name = service_name
|
|
this.clear();
|
|
}
|
|
|
|
/**
|
|
* Clears the download list
|
|
*/
|
|
clear() {
|
|
this.download_list = "";
|
|
}
|
|
|
|
/**
|
|
* Alias to clear() (clears the download list)
|
|
*/
|
|
reset() {
|
|
this.clear();
|
|
}
|
|
|
|
/**
|
|
* Returns the download list.
|
|
* @returns {string} Download list for client;
|
|
*/
|
|
getDownloadList() {
|
|
return this.download_list;
|
|
}
|
|
|
|
/**
|
|
* Adds a DISPLAY command to the download list
|
|
* @param {string} message Message to display to the client
|
|
*/
|
|
display(message) {
|
|
this.download_list += "DISPLAY " + message + "\n\n";
|
|
}
|
|
|
|
/**
|
|
* Adds an EXECUTE command to the download list
|
|
* @param {string} command client command to execute
|
|
*/
|
|
execute(command) {
|
|
this.download_list += "EXECUTE " + command + "\n\n";
|
|
}
|
|
|
|
/**
|
|
* Adds a CREATE partition command to the download list
|
|
* @param {string} path file://Disk/ path to desired partition
|
|
* @param {string} size Size of the desired partition
|
|
*/
|
|
createPartition(path, size) {
|
|
this.download_list += "CREATE " + path + "\n";
|
|
this.download_list += "partition-size: " + size + "\n\n";
|
|
}
|
|
|
|
/**
|
|
* Adds a CREATE-GROUP command to the download list
|
|
* @param {string} name Group name
|
|
* @param {string} path file://Disk/ path of desired group
|
|
* @param {string} state Group state
|
|
* @param {boolean|null} service_owned Sets service owned flag. (null = don't set)
|
|
*/
|
|
createGroup(name, path, state = 'invalid', service_owned = null) {
|
|
this.download_list += "CREATE-GROUP " + name + "\n";
|
|
this.download_list += "state: " + state + "\n";
|
|
if (service_owned !== null) this.download_list += "service-owned: " + service_owned + "\n";
|
|
this.download_list += "base: " + path + "\n\n";
|
|
}
|
|
|
|
/**
|
|
* An alias for createGroup() that handles creating the '-UPDATE' group for you
|
|
* @param {string} name Group name
|
|
* @param {string} path file://Disk/ path of desired group
|
|
* @param {string} state Group state
|
|
* @param {boolean} service_owned Sets service owned flag.
|
|
*/
|
|
createUpdateGroup(name, path, state = 'invalid', service_owned = false) {
|
|
this.createGroup(name + "-UPDATE", path, state);
|
|
this.createGroup(name, path, state, service_owned);
|
|
}
|
|
|
|
/**
|
|
* Adds a DELETE command to the download list
|
|
* @param {string} path Non-absolute path of client destination file (relative to group base) if group defined, otherwise absolute file://Disk/ path to delete
|
|
* @param {string|null} group Group to which it belongs
|
|
* @param {string|null} original_filename Use this filename (useful if WTV GZ)
|
|
*/
|
|
delete(path, group = null, original_filename = null) {
|
|
path = this.checkOriginalName(path, original_filename);
|
|
this.download_list += "DELETE " + path + "\n";
|
|
if (group) this.download_list += "group: " + group + "\n\n";
|
|
else (this.download_list) += "\n";
|
|
}
|
|
|
|
/**
|
|
* Adds a PUT command to the download list
|
|
* @param {string} path Absolute file://Disk/ path of a file to upload to the service
|
|
* @param {string} destination Destination address (wtv url on service) in which to POST upload the file to
|
|
*/
|
|
put(path, destination) {
|
|
this.download_list += "PUT " + path + "\n";
|
|
this.download_list += "location: " + destination + "\n\n";
|
|
}
|
|
|
|
/**
|
|
* Alias to put() for User Store
|
|
* @param {string} path Absolute file://Disk/ path of a file to upload to the service
|
|
* @param {string} destination Destination file path in the User Store
|
|
*/
|
|
putUserStoreDest(path, destination) {
|
|
this.put(path, this.service_name + ":/userstore?partialPath=" + escape(destination));
|
|
}
|
|
|
|
/**
|
|
* Alias to putUserStoreDest() that generates the destination
|
|
* @param {any} path
|
|
*/
|
|
putUserStore(path) {
|
|
var destination = path.replace("file://", "");
|
|
this.putUserStoreDest(path, destination);
|
|
}
|
|
/**
|
|
* Adds a GET command to the download list
|
|
* @param {string} file Non-absolute path of client destination file (relative to group base)
|
|
* @param {string} path Absolute file://Disk/ path of destination
|
|
* @param {string} source wtv-url to fetch file from
|
|
* @param {string} group Group this file belongs to
|
|
* @param {string|null} checksum md5sum of the file
|
|
* @param {string|null} uncompressed_size Uncompressed size of gzip file
|
|
* @param {string|null} original_filename Use this filename (useful if WTV GZ)
|
|
* @param {string} file_permission File permissions
|
|
*/
|
|
get(file, path, source, group, checksum = null, uncompressed_size = null, original_filename = null, file_permission = 'r') {
|
|
if (original_filename) {
|
|
file = file.split('/');
|
|
var file_name = file[file.length - 1];
|
|
path = path.replace(file_name, original_filename);
|
|
file.pop();
|
|
if (file.length > 0) file = file.join('/') + '/' + original_filename;
|
|
else file = original_filename;
|
|
}
|
|
this.download_list += "GET " + file + "\n";
|
|
|
|
source = source.replace(/\\/g, "/");
|
|
this.download_list += "group: " + group + "-UPDATE\n";
|
|
this.download_list += "location: " + source + "\n";
|
|
this.download_list += "file-permission: " + file_permission + "\n";
|
|
if (checksum != null) this.download_list += "wtv-checksum: " + checksum + "\n";
|
|
if (uncompressed_size != null) this.download_list += "wtv-uncompressed-filesize: " + uncompressed_size + "\n";
|
|
this.download_list += "service-source-location: /webtv/content/" + source.substr(source.indexOf('-') + 1, source.indexOf(':/') - source.indexOf('-') - 1) + "d/" + source.substr(source.indexOf(':/') + 2) + "\n";
|
|
this.download_list += "client-dest-location: " + path + "\n\n";
|
|
}
|
|
|
|
/**
|
|
* Helper function for WTV GZIP, if original_name is set, use this filename instead of the name in the path
|
|
* @param {string} path
|
|
* @param {string} original_name
|
|
* @returns {string} Path, with filename replaces by original_name, or just path if original_name = null
|
|
*/
|
|
checkOriginalName(path, original_name) {
|
|
if (original_name) {
|
|
var tmp = this.wtvshared.getFilePath(path);
|
|
if (tmp.length > 0) return tmp + "/" + original_name;
|
|
return original_name
|
|
} else return path;
|
|
}
|
|
|
|
getGroupDataFromClientPost(post_data) {
|
|
if (typeof post_data == 'string') post_data = post_data.split("\n\n");
|
|
var group_data = [];
|
|
var i = 0;
|
|
post_data.forEach(function (v) {
|
|
if (v.substr(0, 4) == "file") {
|
|
var block_split = v.split("\n");
|
|
var group_data_entry = {};
|
|
group_data_entry.path = block_split[0];
|
|
block_split.forEach(function (block_section) {
|
|
if (block_section.indexOf(": ") > 0) {
|
|
var block_section_split = block_section.split(": ");
|
|
group_data_entry[block_section_split[0]] = block_section_split[1];
|
|
}
|
|
});
|
|
group_data[group_data_entry.group] = group_data_entry;
|
|
}
|
|
});
|
|
return group_data;
|
|
}
|
|
|
|
/**
|
|
* Adds a RENAME command to the download list
|
|
* @param {string} srcfile Non-absolute path of client source file (relative to source group base)
|
|
* @param {string} destfile Non-absolute path of client destination file (relative to destination group base)
|
|
* @param {string} srcgroup Source Group
|
|
* @param {string} destgroup Destination Group
|
|
* @param {string} original_filename Use this filename (useful if WTV GZ)
|
|
*/
|
|
rename(srcfile, destfile, srcgroup, destgroup, original_filename = null) {
|
|
if (original_filename) {
|
|
srcfile = this.checkOriginalName(srcfile, original_filename);
|
|
destfile = this.checkOriginalName(srcfile, original_filename);
|
|
}
|
|
this.download_list += "RENAME " + srcfile + "\n";
|
|
this.download_list += "group: " + srcgroup + "-UPDATE\n";
|
|
this.download_list += "destination-group: " + destgroup + "\n";
|
|
this.download_list += "location: " + destfile + "\n\n";
|
|
}
|
|
|
|
/**
|
|
* Adds a SET-GROUP command to the download list
|
|
* @param {string} group Group to set state of
|
|
* @param {string} state State to set group to
|
|
* @param {string} version Version to set group to
|
|
*/
|
|
setGroup(group, state, version) {
|
|
this.download_list += "SET-GROUP " + group + "\n";
|
|
this.download_list += "state: " + state + "\n";
|
|
this.download_list += "version: " + version + "\n";
|
|
this.download_list += "last-checkup-time: " + new Date().toUTCString().replace("GMT", "+0000") + "\n\n";
|
|
}
|
|
|
|
/**
|
|
* Adds a DELETE-GROUP command to the download list
|
|
* @param {string} group Group to delete
|
|
*/
|
|
deleteGroup(group) {
|
|
this.download_list += "DELETE-GROUP " + group + "\n\n";
|
|
}
|
|
|
|
/**
|
|
* An alias for deleteGroup() that handles deleting the '-UPDATE' group files for you
|
|
* @param {string} group Group to delete
|
|
* @param {string} path Group base path
|
|
*/
|
|
deleteGroupUpdate(group, path) {
|
|
this.deleteGroup(group + "-UPDATE");
|
|
this.delete(path + ".GROUP-UPDATE/");
|
|
}
|
|
|
|
/**
|
|
* Generates the Download page
|
|
* @param {object} minisrv_config minisrv config object
|
|
* @param {string} title Page title
|
|
* @param {string} group
|
|
* @param {string|null} diskmap
|
|
* @param {string|null} main_message Message displayed in the center of the page
|
|
* @param {string|null} message Initial progress bar message
|
|
* @param {boolean|null} force_update Force this update even if the client reports the files are synced
|
|
* @param {string|null} success_url Where the client goes when the process succeeds
|
|
* @param {string|null} fail_url Where the client goes when the process fails.
|
|
* @param {string|null} url Use your own URL for client:fetch?source= instead of our generated one
|
|
* @returns {string} HTML Download Page
|
|
*/
|
|
getSyncPage(title, group, diskmap = null, main_message = null, message = null, force_update = null, dont_delete_files = null, success_url = null, fail_url = null, url = null) {
|
|
// Begin Set defaults
|
|
if (main_message === null) main_message = "Your receiver is downloading files.";
|
|
|
|
if (message === null) message = "Retrieving files";
|
|
|
|
if (force_update === null) force_update = false;
|
|
|
|
if (url === null) url = this.service_name + ":/sync?diskmap=" + escape(diskmap);
|
|
|
|
if (force_update) url += "&force=" + force_update;
|
|
if (dont_delete_files) url += "&dont_delete_files=" + dont_delete_files;
|
|
|
|
if (success_url === null) success_url = new this.clientShowAlert({
|
|
'image': this.minisrv_config.config.service_logo,
|
|
'message': "Download successful!",
|
|
'buttonlabel1': "Okay",
|
|
'buttonaction1': "client:goback",
|
|
'noback': true,
|
|
}).getURL();
|
|
|
|
if (fail_url === null) fail_url = new this.clientShowAlert({
|
|
'image': this.minisrv_config.config.service_logo,
|
|
'message': "Download failed...",
|
|
'buttonlabel1': "Okay",
|
|
'buttonaction1': "client:goback",
|
|
'noback': true,
|
|
}).getURL();
|
|
// End set defaults
|
|
return `<html>
|
|
<head>
|
|
<meta
|
|
http-equiv=refresh
|
|
content="0;url=client:Fetch?group=${encodeURIComponent(group)}&source=${encodeURIComponent(url)}&message=${encodeURIComponent(message)}"
|
|
>
|
|
<display downloadsuccess="${success_url}" downloadfail="${fail_url}">
|
|
<title>${title}</title>
|
|
</head>
|
|
<body bgcolor=#0 text=#42CC55 fontsize=large hspace=0 vspace=0>
|
|
<table cellspacing=0 cellpadding=0>
|
|
<tr>
|
|
<td width=104 height=74 valign=middle align=center bgcolor=3B3A4D>
|
|
<img src="${this.minisrv_config.config.service_logo}" width=86 height=64>
|
|
<td width=20 valign=top align=left bgcolor=3B3A4D>
|
|
<spacer>
|
|
<td colspan=2 width=436 valign=middle align=left bgcolor=3B3A4D>
|
|
<font color=D6DFD0 size=+2><blackface><shadow>
|
|
<spacer type=block width=1 height=4>
|
|
<br>
|
|
${message}
|
|
</shadow>
|
|
</blackface>
|
|
</font>
|
|
<tr>
|
|
<td width=104 height=20>
|
|
<td width=20>
|
|
<td width=416>
|
|
<td width=20>
|
|
<tr>
|
|
<td colspan=2>
|
|
<td>
|
|
<font size=+1>
|
|
${main_message}
|
|
<p>This may take a while.
|
|
</font>
|
|
<tr>
|
|
<td colspan=2>
|
|
<td>
|
|
<br><br>
|
|
<font color=white>
|
|
<progressindicator name="downloadprogress"
|
|
message="Preparing..."
|
|
height=40 width=250>
|
|
</font>
|
|
|
|
<br>
|
|
</table>
|
|
</body>
|
|
</html>
|
|
|
|
`
|
|
}
|
|
|
|
}
|
|
|
|
module.exports = WTVDownloadList; |