2019-03-31 18:07:03 +00:00
|
|
|
declare const native: any; //FIXME: Read client declarations!
|
2019-02-23 13:15:22 +00:00
|
|
|
namespace connection {
|
|
|
|
export interface HandshakeIdentityHandler {
|
|
|
|
connection: AbstractServerConnection;
|
|
|
|
|
|
|
|
start_handshake();
|
|
|
|
register_callback(callback: (success: boolean, message?: string) => any);
|
|
|
|
}
|
|
|
|
|
|
|
|
export class HandshakeHandler {
|
2019-04-15 13:33:51 +00:00
|
|
|
private connection: AbstractServerConnection;
|
2019-02-23 13:15:22 +00:00
|
|
|
private handshake_handler: HandshakeIdentityHandler;
|
|
|
|
private failed = false;
|
|
|
|
|
|
|
|
readonly profile: profiles.ConnectionProfile;
|
|
|
|
readonly name: string;
|
|
|
|
readonly server_password: string;
|
|
|
|
|
|
|
|
constructor(profile: profiles.ConnectionProfile, name: string, password: string) {
|
|
|
|
this.profile = profile;
|
|
|
|
this.server_password = password;
|
|
|
|
this.name = name;
|
|
|
|
}
|
|
|
|
|
2019-04-15 13:33:51 +00:00
|
|
|
setConnection(con: AbstractServerConnection) {
|
2019-02-23 13:15:22 +00:00
|
|
|
this.connection = con;
|
|
|
|
}
|
|
|
|
|
2019-04-15 13:33:51 +00:00
|
|
|
initialize() {
|
2019-02-23 13:15:22 +00:00
|
|
|
this.handshake_handler = this.profile.spawn_identity_handshake_handler(this.connection);
|
|
|
|
if(!this.handshake_handler) {
|
|
|
|
this.handshake_failed("failed to create identity handler");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.handshake_handler.register_callback((flag, message) => {
|
|
|
|
if(flag)
|
|
|
|
this.handshake_finished();
|
|
|
|
else
|
|
|
|
this.handshake_failed(message);
|
|
|
|
});
|
2019-04-15 13:33:51 +00:00
|
|
|
}
|
2019-02-23 13:15:22 +00:00
|
|
|
|
2019-04-15 13:33:51 +00:00
|
|
|
get_identity_handler() : HandshakeIdentityHandler {
|
|
|
|
return this.handshake_handler;
|
|
|
|
}
|
|
|
|
|
|
|
|
startHandshake() {
|
2019-02-23 13:15:22 +00:00
|
|
|
this.handshake_handler.start_handshake();
|
|
|
|
}
|
|
|
|
|
2019-04-15 14:58:42 +00:00
|
|
|
on_teamspeak() {
|
|
|
|
const type = this.profile.selected_type();
|
|
|
|
if(type == profiles.identities.IdentitifyType.TEAMSPEAK)
|
|
|
|
this.handshake_finished();
|
|
|
|
else {
|
|
|
|
|
|
|
|
if(this.failed) return;
|
|
|
|
|
|
|
|
this.failed = true;
|
|
|
|
this.connection.client.handleDisconnect(DisconnectReason.HANDSHAKE_TEAMSPEAK_REQUIRED);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-23 13:15:22 +00:00
|
|
|
private handshake_failed(message: string) {
|
|
|
|
if(this.failed) return;
|
|
|
|
|
|
|
|
this.failed = true;
|
|
|
|
this.connection.client.handleDisconnect(DisconnectReason.HANDSHAKE_FAILED, message);
|
|
|
|
}
|
|
|
|
|
|
|
|
private handshake_finished(version?: string) {
|
|
|
|
if(native_client && window["native"] && native.client_version && !version) {
|
|
|
|
native.client_version()
|
|
|
|
.then( this.handshake_finished.bind(this))
|
|
|
|
.catch(error => {
|
|
|
|
console.error(tr("Failed to get version:"));
|
|
|
|
console.error(error);
|
|
|
|
this.handshake_finished("?.?.?");
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const git_version = settings.static_global("version", "unknown");
|
|
|
|
const browser_name = (navigator.browserSpecs || {})["name"] || " ";
|
|
|
|
let data = {
|
|
|
|
client_nickname: this.name,
|
|
|
|
client_platform: (browser_name ? browser_name + " " : "") + navigator.platform,
|
|
|
|
client_version: "TeaWeb " + git_version + " (" + navigator.userAgent + ")",
|
2019-04-15 13:33:51 +00:00
|
|
|
client_version_sign: undefined,
|
2019-02-23 13:15:22 +00:00
|
|
|
|
|
|
|
client_server_password: this.server_password,
|
2019-04-04 19:47:52 +00:00
|
|
|
client_browser_engine: navigator.product,
|
|
|
|
|
|
|
|
client_input_hardware: this.connection.client.client_status.input_hardware,
|
|
|
|
client_output_hardware: false,
|
|
|
|
client_input_muted: this.connection.client.client_status.input_muted,
|
|
|
|
client_output_muted: this.connection.client.client_status.output_muted,
|
2019-02-23 13:15:22 +00:00
|
|
|
};
|
|
|
|
|
2019-04-15 13:33:51 +00:00
|
|
|
//0.0.1 [Build: 1549713549] Linux 7XvKmrk7uid2ixHFeERGqcC8vupeQqDypLtw2lY9slDNPojEv//F47UaDLG+TmVk4r6S0TseIKefzBpiRtLDAQ==
|
|
|
|
|
2019-02-23 13:15:22 +00:00
|
|
|
if(version) {
|
|
|
|
data.client_version = "TeaClient ";
|
|
|
|
data.client_version += " " + version;
|
|
|
|
|
|
|
|
const os = require("os");
|
|
|
|
const arch_mapping = {
|
|
|
|
"x32": "32bit",
|
|
|
|
"x64": "64bit"
|
|
|
|
};
|
|
|
|
|
|
|
|
data.client_version += " " + (arch_mapping[os.arch()] || os.arch());
|
|
|
|
|
|
|
|
const os_mapping = {
|
|
|
|
"win32": "Windows",
|
|
|
|
"linux": "Linux"
|
|
|
|
};
|
|
|
|
data.client_platform = (os_mapping[os.platform()] || os.platform());
|
|
|
|
}
|
|
|
|
|
2019-04-15 13:33:51 +00:00
|
|
|
/* required to keep compatibility */
|
|
|
|
if(this.profile.selected_type() === profiles.identities.IdentitifyType.TEAMSPEAK) {
|
|
|
|
data["client_key_offset"] = (this.profile.selected_identity() as profiles.identities.TeaSpeakIdentity).hash_number;
|
|
|
|
}
|
|
|
|
|
2019-02-23 13:15:22 +00:00
|
|
|
this.connection.send_command("clientinit", data).catch(error => {
|
|
|
|
if(error instanceof CommandResult) {
|
|
|
|
if(error.id == 1028) {
|
|
|
|
this.connection.client.handleDisconnect(DisconnectReason.SERVER_REQUIRES_PASSWORD);
|
2019-04-15 13:33:51 +00:00
|
|
|
} else if(error.id == 783 || error.id == 519) {
|
|
|
|
error.extra_message = parseInt(error.extra_message) == NaN ? "8" : error.extra_message;
|
|
|
|
this.connection.client.handleDisconnect(DisconnectReason.IDENTITY_TOO_LOW, error);
|
2019-04-15 14:58:42 +00:00
|
|
|
} else if(error.id == 3329) {
|
|
|
|
this.connection.client.handleDisconnect(DisconnectReason.HANDSHAKE_BANNED, error);
|
2019-02-23 13:15:22 +00:00
|
|
|
} else {
|
|
|
|
this.connection.client.handleDisconnect(DisconnectReason.CLIENT_KICKED, error);
|
|
|
|
}
|
2019-04-15 13:33:51 +00:00
|
|
|
} else
|
|
|
|
this.connection.disconnect();
|
2019-02-23 13:15:22 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|