TeaWeb/shared/js/client.ts

260 lines
10 KiB
TypeScript
Raw Normal View History

/// <reference path="log.ts" />
2018-03-24 23:38:01 +01:00
/// <reference path="voice/AudioController.ts" />
2018-02-27 17:20:49 +01:00
/// <reference path="proto.ts" />
/// <reference path="ui/view.ts" />
/// <reference path="connection.ts" />
/// <reference path="settings.ts" />
/// <reference path="ui/frames/SelectedItemInfo.ts" />
2018-02-27 17:20:49 +01:00
/// <reference path="FileManager.ts" />
/// <reference path="permission/PermissionManager.ts" />
/// <reference path="permission/GroupManager.ts" />
2018-07-03 12:33:31 +02:00
/// <reference path="ui/frames/ControlBar.ts" />
2018-02-27 17:20:49 +01:00
2018-03-07 19:06:52 +01:00
enum DisconnectReason {
2018-03-07 20:14:36 +01:00
REQUESTED,
2018-03-07 19:06:52 +01:00
CONNECT_FAILURE,
CONNECTION_CLOSED,
CONNECTION_FATAL_ERROR,
CONNECTION_PING_TIMEOUT,
CLIENT_KICKED,
CLIENT_BANNED,
HANDSHAKE_FAILED,
2018-03-07 19:06:52 +01:00
SERVER_CLOSED,
2018-09-24 21:14:36 +02:00
SERVER_REQUIRES_PASSWORD,
2018-03-07 19:06:52 +01:00
UNKNOWN
}
enum ConnectionState {
UNCONNECTED,
CONNECTING,
INITIALISING,
CONNECTED,
DISCONNECTING
}
2018-02-27 17:20:49 +01:00
enum ViewReasonId {
VREASON_USER_ACTION = 0,
VREASON_MOVED = 1,
VREASON_SYSTEM = 2,
VREASON_TIMEOUT = 3,
VREASON_CHANNEL_KICK = 4,
VREASON_SERVER_KICK = 5,
VREASON_BAN = 6,
VREASON_SERVER_STOPPED = 7,
VREASON_SERVER_LEFT = 8,
VREASON_CHANNEL_UPDATED = 9,
VREASON_EDITED = 10,
VREASON_SERVER_SHUTDOWN = 11
}
class TSClient {
channelTree: ChannelTree;
serverConnection: ServerConnection;
voiceConnection: VoiceConnection;
fileManager: FileManager;
selectInfo: InfoBar;
permissions: PermissionManager;
groups: GroupManager;
controlBar: ControlBar;
private _clientId: number = 0;
private _ownEntry: LocalClientEntry;
constructor() {
this.selectInfo = new InfoBar(this, $("#select_info"));
this.channelTree = new ChannelTree(this, $("#channelTree"));
2018-03-07 19:06:52 +01:00
this.serverConnection = new ServerConnection(this);
2018-02-27 17:20:49 +01:00
this.fileManager = new FileManager(this);
this.permissions = new PermissionManager(this);
this.groups = new GroupManager(this);
this.voiceConnection = new VoiceConnection(this);
this._ownEntry = new LocalClientEntry(this);
this.controlBar = new ControlBar(this, $("#control_bar"));
this.channelTree.registerClient(this._ownEntry);
}
setup() {
this.controlBar.initialise();
}
startConnection(addr: string, profile: profiles.ConnectionProfile, name?: string, password?: {password: string, hashed: boolean}) {
2018-03-07 20:14:36 +01:00
if(this.serverConnection)
this.handleDisconnect(DisconnectReason.REQUESTED);
2018-02-27 17:20:49 +01:00
let idx = addr.lastIndexOf(':');
let port: number;
let host: string;
if(idx != -1) {
2018-03-24 23:38:01 +01:00
port = parseInt(addr.substr(idx + 1));
2018-02-27 17:20:49 +01:00
host = addr.substr(0, idx);
} else {
host = addr;
2018-04-16 20:38:35 +02:00
port = 9987;
2018-02-27 17:20:49 +01:00
}
console.log(tr("Start connection to %s:%d"), host, port);
this.channelTree.initialiseHead(addr, {host, port});
2018-09-24 21:14:36 +02:00
if(password && !password.hashed) {
helpers.hashPassword(password.password).then(password => {
this.serverConnection.startConnection({host, port}, new HandshakeHandler(profile, name, password));
2018-09-24 21:14:36 +02:00
}).catch(error => {
createErrorModal(tr("Error while hashing password"), tr("Failed to hash server password!<br>") + error).open();
2018-09-24 21:14:36 +02:00
})
} else
this.serverConnection.startConnection({host, port}, new HandshakeHandler(profile, name, password ? password.password : undefined));
2018-02-27 17:20:49 +01:00
}
getClient() : LocalClientEntry { return this._ownEntry; }
getClientId() { return this._clientId; } //TODO here
2018-02-27 17:20:49 +01:00
set clientId(id: number) {
this._clientId = id;
this._ownEntry["_clientId"] = id;
}
2018-03-07 19:06:52 +01:00
get clientId() {
return this._clientId;
}
2018-02-27 17:20:49 +01:00
getServerConnection() : ServerConnection { return this.serverConnection; }
/**
* LISTENER
*/
onConnected() {
2018-03-07 19:06:52 +01:00
console.log("Client connected!");
2018-03-07 20:14:36 +01:00
this.channelTree.registerClient(this._ownEntry);
2018-04-16 20:38:35 +02:00
settings.setServer(this.channelTree.server);
2018-03-07 19:06:52 +01:00
this.permissions.requestPermissionList();
2018-04-16 20:38:35 +02:00
this.serverConnection.sendCommand("channelsubscribeall");
2018-03-07 19:06:52 +01:00
if(this.groups.serverGroups.length == 0)
this.groups.requestGroups();
this.controlBar.updateProperties();
2018-09-25 17:39:38 +02:00
if(!this.voiceConnection.current_encoding_supported())
createErrorModal(tr("Codec encode type not supported!"), tr("Codec encode type " + VoiceConnectionType[this.voiceConnection.type] + " not supported by this browser!<br>Choose another one!")).open(); //TODO tr
2018-02-27 17:20:49 +01:00
}
get connected() : boolean {
return !!this.serverConnection && this.serverConnection.connected;
}
private certAcceptUrl() {
const properties = {
2019-01-02 12:18:05 +01:00
connect_default: true,
connect_profile: this.serverConnection._handshakeHandler.profile.id,
2019-01-02 12:18:05 +01:00
connect_address: this.serverConnection._remote_address.host + ":" + this.serverConnection._remote_address.port
};
2019-01-02 12:18:05 +01:00
const parameters: string[] = [];
for(const key in properties)
2019-01-20 18:43:14 +01:00
parameters.push(key + "=" + encodeURIComponent(properties[key]));
2019-01-02 12:18:05 +01:00
// document.URL
let callback = document.URL;
if(document.location.search.length == 0)
2019-01-02 12:18:05 +01:00
callback += "?" + parameters.join("&");
else
2019-01-02 12:18:05 +01:00
callback += "&" + parameters.join("&");
return "https://" + this.serverConnection._remote_address.host + ":" + this.serverConnection._remote_address.port + "/?forward_url=" + encodeURIComponent(callback);
}
2018-03-07 19:06:52 +01:00
handleDisconnect(type: DisconnectReason, data: any = {}) {
switch (type) {
2018-03-07 20:14:36 +01:00
case DisconnectReason.REQUESTED:
break;
2018-03-07 19:06:52 +01:00
case DisconnectReason.CONNECT_FAILURE:
console.error(tr("Could not connect to remote host! Exception: %o"), data);
2018-03-07 19:06:52 +01:00
2018-10-06 15:13:45 +02:00
if(native_client) {
createErrorModal(
tr("Could not connect"),
tr("Could not connect to remote host (Connection refused)")
2018-10-06 15:13:45 +02:00
).open();
} else {
//TODO tr
2018-10-06 15:13:45 +02:00
createErrorModal(
tr("Could not connect"),
2018-10-06 15:13:45 +02:00
"Could not connect to remote host (Connection refused)<br>" +
"If you're sure that the remote host is up, than you may not allow unsigned certificates.<br>" +
"Click <a href='" + this.certAcceptUrl() + "'>here</a> to accept the remote certificate"
).open();
}
2018-11-04 00:39:29 +01:00
sound.play(Sound.CONNECTION_REFUSED);
2018-03-07 19:06:52 +01:00
break;
case DisconnectReason.HANDSHAKE_FAILED:
//TODO sound
console.error(tr("Failed to process handshake: %o"), data);
createErrorModal(
tr("Could not connect"),
tr("Failed to process handshake: ") + data as string
).open();
break;
2018-03-07 19:06:52 +01:00
case DisconnectReason.CONNECTION_CLOSED:
console.error(tr("Lost connection to remote server!"));
2018-03-07 19:06:52 +01:00
createErrorModal(
tr("Connection closed"),
tr("The connection was closed by remote host")
2018-03-07 19:06:52 +01:00
).open();
2018-11-04 00:39:29 +01:00
sound.play(Sound.CONNECTION_DISCONNECTED);
2018-03-07 19:06:52 +01:00
break;
case DisconnectReason.CONNECTION_PING_TIMEOUT:
console.error(tr("Connection ping timeout"));
2018-11-04 00:39:29 +01:00
sound.play(Sound.CONNECTION_DISCONNECTED_TIMEOUT);
2018-03-07 19:06:52 +01:00
createErrorModal(
tr("Connection lost"),
tr("Lost connection to remote host (Ping timeout)<br>Even possible?")
2018-03-07 19:06:52 +01:00
).open();
break;
case DisconnectReason.SERVER_CLOSED:
chat.serverChat().appendError(tr("Server closed ({0})"), data.reasonmsg);
2018-03-07 19:06:52 +01:00
createErrorModal(
tr("Server closed"),
"The server is closed.<br>" + //TODO tr
2018-03-07 19:06:52 +01:00
"Reason: " + data.reasonmsg
).open();
2018-11-04 00:39:29 +01:00
sound.play(Sound.CONNECTION_DISCONNECTED);
2018-03-07 19:06:52 +01:00
break;
2018-09-24 21:14:36 +02:00
case DisconnectReason.SERVER_REQUIRES_PASSWORD:
chat.serverChat().appendError(tr("Server requires password"));
createInputModal(tr("Server password"), tr("Enter server password:"), password => password.length != 0, password => {
2018-09-24 21:14:36 +02:00
if(!(typeof password === "string")) return;
this.startConnection(this.serverConnection._remote_address.host + ":" + this.serverConnection._remote_address.port,
this.serverConnection._handshakeHandler.profile,
2018-09-24 21:14:36 +02:00
this.serverConnection._handshakeHandler.name,
{password: password as string, hashed: false});
}).open();
2018-11-04 00:39:29 +01:00
break;
case DisconnectReason.CLIENT_KICKED:
chat.serverChat().appendError(tr("You got kicked from the server by {0}{1}"),
2018-11-04 00:39:29 +01:00
ClientEntry.chatTag(data["invokerid"], data["invokername"], data["invokeruid"]),
data["reasonmsg"] ? " (" + data["reasonmsg"] + ")" : "");
sound.play(Sound.SERVER_KICKED);
break;
case DisconnectReason.CLIENT_BANNED:
chat.serverChat().appendError(tr("You got banned from the server by {0}{1}"),
2018-11-04 00:39:29 +01:00
ClientEntry.chatTag(data["invokerid"], data["invokername"], data["invokeruid"]),
data["reasonmsg"] ? " (" + data["reasonmsg"] + ")" : "");
sound.play(Sound.CONNECTION_BANNED); //TODO findout if it was a disconnect or a connect refuse
break;
2018-03-07 19:06:52 +01:00
default:
console.error(tr("Got uncaught disconnect!"));
console.error(tr("Type: %o Data:"), type);
2018-03-07 19:06:52 +01:00
console.error(data);
break;
}
2018-02-27 17:20:49 +01:00
2018-03-07 19:06:52 +01:00
this.channelTree.reset();
this.voiceConnection.dropSession();
2018-03-07 20:14:36 +01:00
if(this.serverConnection) this.serverConnection.disconnect();
2018-09-25 12:57:47 +02:00
this.controlBar.update_connection_state();
2018-11-25 13:57:48 +01:00
this.selectInfo.setCurrentSelected(null);
this.selectInfo.update_banner();
2018-02-27 17:20:49 +01:00
}
}