Add moderation rules: * You can apply rules in the settings.toml to enforce moderator restrictions on certain users, e.g. to force their camera to always be NSFW or bar them from sharing their webcam at all anymore. Chat UI improvements around users blocking admin accounts: * When a main website block is in place, the DMs button in the Who List shows as greyed out with a cross through, as if that user had closed their DMs. * Admin users are always able to watch the camera of people who have blocked them. The broadcaster is not notified about the watch. New operator commands: * /cut username: to tell a user to turn off their webcam. * /unmute-all: to lift all mutes on your side, e.g. so your moderator chatbot can still see public messages from users who have blocked it. * /help-advanced: moved the more dangerous admin command documentation here. Miscellaneous fixes: * The admin commands now tolerate an @ prefix in front of usernames. * The /nsfw command won't fire unless the user's camera is actually active and not marked as explicit.
345 lines
10 KiB
JavaScript
345 lines
10 KiB
JavaScript
// WebSocket chat client handler.
|
|
class ChatClient {
|
|
/**
|
|
* Constructor for the client.
|
|
*
|
|
* @param usePolling: instead of WebSocket use the ajax polling API.
|
|
* @param onClientError: function to receive 'ChatClient' messages to
|
|
* add to the chat room (this.ChatClient())
|
|
*/
|
|
constructor({
|
|
usePolling=false,
|
|
onClientError,
|
|
|
|
username,
|
|
jwt, // JWT token for authorization
|
|
prefs, // User preferences for 'me' action (close DMs, etc)
|
|
|
|
// Chat Protocol handler functions for the caller.
|
|
onWho,
|
|
onMe,
|
|
onMessage,
|
|
onTakeback,
|
|
onReact,
|
|
onPresence,
|
|
onRing,
|
|
onOpen,
|
|
onCandidate,
|
|
onSDP,
|
|
onWatch,
|
|
onUnwatch,
|
|
onBlock,
|
|
onCut,
|
|
|
|
// Misc function registrations for callback.
|
|
onNewJWT, // new JWT token from ping response
|
|
bulkMuteUsers, // Upload our blocklist on connect.
|
|
focusMessageBox, // Tell caller to focus the message entry box.
|
|
pushHistory,
|
|
}) {
|
|
this.usePolling = usePolling;
|
|
|
|
// Pointer to the 'ChatClient(message)' command from the main app.
|
|
this.ChatClient = onClientError;
|
|
|
|
this.username = username;
|
|
this.jwt = jwt;
|
|
this.prefs = prefs;
|
|
|
|
// Register the handler functions.
|
|
this.onWho = onWho;
|
|
this.onMe = onMe;
|
|
this.onMessage = onMessage;
|
|
this.onTakeback = onTakeback;
|
|
this.onReact = onReact;
|
|
this.onPresence = onPresence;
|
|
this.onRing = onRing;
|
|
this.onOpen = onOpen;
|
|
this.onCandidate = onCandidate;
|
|
this.onSDP = onSDP;
|
|
this.onWatch = onWatch;
|
|
this.onUnwatch = onUnwatch;
|
|
this.onBlock = onBlock;
|
|
this.onCut = onCut;
|
|
|
|
this.onNewJWT = onNewJWT;
|
|
this.bulkMuteUsers = bulkMuteUsers;
|
|
this.focusMessageBox = focusMessageBox;
|
|
this.pushHistory = pushHistory;
|
|
|
|
// WebSocket connection.
|
|
this.ws = {
|
|
conn: null,
|
|
connected: false,
|
|
|
|
// Disconnect spamming: don't retry too many times.
|
|
reconnect: true, // unless told to go away
|
|
disconnectLimit: 2,
|
|
disconnectCount: 0,
|
|
};
|
|
|
|
// Polling connection.
|
|
this.polling = {
|
|
username: "",
|
|
sessionID: "",
|
|
timeout: null, // setTimeout for next poll.
|
|
}
|
|
}
|
|
|
|
// Connected polls if the client is connected.
|
|
connected() {
|
|
if (this.usePolling) {
|
|
return this.polling.timeout != null && this.polling.sessionID != "";
|
|
}
|
|
return this.ws.connected;
|
|
}
|
|
|
|
// Disconnect from the server.
|
|
disconnect() {
|
|
if (this.usePolling) {
|
|
this.polling.sessionID = "";
|
|
this.polling.username = "";
|
|
this.stopPolling();
|
|
this.ChatClient("You have disconnected from the server.");
|
|
return;
|
|
}
|
|
|
|
this.ws.connected = false;
|
|
this.ws.conn.close(1000, "server asked to close the connection");
|
|
}
|
|
|
|
// Common function to send a message to the server. The message
|
|
// is a JSON object before stringify.
|
|
send(message) {
|
|
if (this.usePolling) {
|
|
fetch("/poll", {
|
|
method: "POST",
|
|
mode: "same-origin",
|
|
cache: "no-cache",
|
|
credentials: "same-origin",
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
},
|
|
body: JSON.stringify({
|
|
username: this.polling.username,
|
|
session_id: this.polling.sessionID,
|
|
msg: message,
|
|
})
|
|
}).then(resp => resp.json()).then(resp => {
|
|
console.log(resp);
|
|
|
|
// Store sessionID information.
|
|
this.polling.sessionID = resp.session_id;
|
|
this.polling.username = resp.username;
|
|
for (let msg of resp.messages) {
|
|
this.handle(msg);
|
|
}
|
|
}).catch(err => {
|
|
this.ChatClient("Error from polling API: " + err);
|
|
});
|
|
return;
|
|
}
|
|
|
|
if (!this.ws.connected) {
|
|
this.ChatClient("Couldn't send WebSocket message: not connected.");
|
|
return;
|
|
}
|
|
|
|
if (typeof(message) !== "string") {
|
|
message = JSON.stringify(message);
|
|
}
|
|
this.ws.conn.send(message);
|
|
}
|
|
|
|
// Common function to handle a message from the server.
|
|
handle(msg) {
|
|
switch (msg.action) {
|
|
case "who":
|
|
this.onWho(msg);
|
|
break;
|
|
case "me":
|
|
this.onMe(msg);
|
|
break;
|
|
case "message":
|
|
this.onMessage(msg);
|
|
break;
|
|
case "takeback":
|
|
this.onTakeback(msg);
|
|
break;
|
|
case "react":
|
|
this.onReact(msg);
|
|
break;
|
|
case "presence":
|
|
this.onPresence(msg);
|
|
break;
|
|
case "ring":
|
|
this.onRing(msg);
|
|
break;
|
|
case "open":
|
|
this.onOpen(msg);
|
|
break;
|
|
case "candidate":
|
|
this.onCandidate(msg);
|
|
break;
|
|
case "sdp":
|
|
this.onSDP(msg);
|
|
break;
|
|
case "watch":
|
|
this.onWatch(msg);
|
|
break;
|
|
case "unwatch":
|
|
this.onUnwatch(msg);
|
|
break;
|
|
case "block":
|
|
this.onBlock(msg);
|
|
break;
|
|
case "cut":
|
|
this.onCut(msg);
|
|
break;
|
|
case "error":
|
|
this.pushHistory({
|
|
channel: msg.channel,
|
|
username: msg.username || 'Internal Server Error',
|
|
message: msg.message,
|
|
isChatServer: true,
|
|
});
|
|
break;
|
|
case "disconnect":
|
|
this.onWho({ whoList: [] });
|
|
this.ws.reconnect = false;
|
|
this.disconnect();
|
|
break;
|
|
case "ping":
|
|
// New JWT token?
|
|
if (msg.jwt) {
|
|
this.onNewJWT(msg.jwt);
|
|
}
|
|
|
|
// Reset disconnect retry counter: if we were on long enough to get
|
|
// a ping, we're well connected and can reconnect no matter how many
|
|
// times the chat server is rebooted.
|
|
this.ws.disconnectCount = 0;
|
|
break;
|
|
default:
|
|
console.error("Unexpected action: %s", JSON.stringify(msg));
|
|
}
|
|
}
|
|
|
|
// Dial the WebSocket.
|
|
dial() {
|
|
// Polling API?
|
|
if (this.usePolling) {
|
|
this.ChatClient("Connecting to the server via polling API...");
|
|
this.startPolling();
|
|
|
|
// Log in now.
|
|
this.send({
|
|
action: "login",
|
|
username: this.username,
|
|
jwt: this.jwt.token,
|
|
dnd: this.prefs.closeDMs,
|
|
});
|
|
return;
|
|
}
|
|
|
|
this.ChatClient("Establishing connection to server...");
|
|
|
|
const proto = location.protocol === 'https:' ? 'wss' : 'ws';
|
|
const conn = new WebSocket(`${proto}://${location.host}/ws`);
|
|
|
|
conn.addEventListener("close", ev => {
|
|
// Lost connection to server - scrub who list.
|
|
this.onWho({ whoList: [] });
|
|
|
|
this.ws.connected = false;
|
|
this.ChatClient(`WebSocket Disconnected code: ${ev.code}, reason: ${ev.reason}`);
|
|
|
|
this.ws.disconnectCount++;
|
|
if (this.ws.disconnectCount > this.ws.disconnectLimit) {
|
|
this.ChatClient(
|
|
`It seems there's a problem connecting to the server. Please try some other time.<br><br>` +
|
|
`If you experience this problem frequently, try going into the Chat Settings 'Misc' tab ` +
|
|
`and switch to the 'Polling' Server Connection method.`
|
|
);
|
|
return;
|
|
}
|
|
|
|
if (this.ws.reconnect) {
|
|
if (ev.code !== 1001 && ev.code !== 1000) {
|
|
this.ChatClient("Reconnecting in 5s");
|
|
setTimeout(() => {
|
|
this.dial();
|
|
}, 5000);
|
|
}
|
|
}
|
|
});
|
|
|
|
conn.addEventListener("open", () => {
|
|
this.ws.connected = true;
|
|
this.ChatClient("Websocket connected!");
|
|
|
|
// Upload our blocklist to the server before login. This resolves a bug where if a block
|
|
// was added recently (other user still online in chat), that user would briefly see your
|
|
// "has entered the room" message followed by you immediately not being online.
|
|
this.bulkMuteUsers();
|
|
|
|
// Tell the server our username.
|
|
this.send({
|
|
action: "login",
|
|
username: this.username,
|
|
jwt: this.jwt.token,
|
|
dnd: this.prefs.closeDMs,
|
|
});
|
|
|
|
// Focus the message entry box.
|
|
window.requestAnimationFrame(() => {
|
|
this.focusMessageBox();
|
|
});
|
|
});
|
|
|
|
conn.addEventListener("message", ev => {
|
|
if (typeof ev.data !== "string") {
|
|
console.error("unexpected message type", typeof ev.data);
|
|
return;
|
|
}
|
|
|
|
let msg = JSON.parse(ev.data);
|
|
this.handle(msg);
|
|
});
|
|
|
|
this.ws.conn = conn;
|
|
}
|
|
|
|
// Start the polling interval.
|
|
startPolling() {
|
|
if (!this.usePolling) return;
|
|
this.stopPolling();
|
|
|
|
this.polling.timeout = setTimeout(() => {
|
|
this.poll();
|
|
this.startPolling();
|
|
}, 5000);
|
|
}
|
|
|
|
// Poll the API.
|
|
poll() {
|
|
if (!this.usePolling) {
|
|
this.stopPolling();
|
|
return;
|
|
}
|
|
this.send({
|
|
action: "ping",
|
|
});
|
|
this.startPolling();
|
|
}
|
|
|
|
// Stop polling.
|
|
stopPolling() {
|
|
if (this.polling.timeout != null) {
|
|
clearTimeout(this.polling.timeout);
|
|
}
|
|
}
|
|
}
|
|
|
|
export default ChatClient;
|