1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
|
/*
* Description: Make a user (spammer) dumb (mute)
* Author: simple
*/
// module constructor
exports.init = (core) => {
if (typeof core.muzzledHashes === 'undefined') {
core.muzzledHashes = {};
}
};
// module main
exports.run = async (core, server, socket, data) => {
// increase rate limit chance and ignore if not admin or mod
if (socket.uType === 'user') {
return server._police.frisk(socket.remoteAddress, 10);
}
// check user input
if (typeof data.nick !== 'string') {
return;
}
// find target user
let badClient = server.findSockets({ channel: socket.channel, nick: data.nick });
if (badClient.length === 0) {
return server.reply({
cmd: 'warn',
text: 'Could not find user in channel'
}, socket);
}
badClient = badClient[0];
// likely dont need this, muting mods and admins is fine
if (badClient.uType !== 'user') {
return server.reply({
cmd: 'warn',
text: 'This trick wont work on mods and admin'
}, socket);
}
// store hash in mute list
let record = core.muzzledHashes[badClient.hash] = {
dumb: true
}
// store allies if needed
if(data.allies && Array.isArray(data.allies)){
record.allies = data.allies;
}
// notify mods
server.broadcast({
cmd: 'info',
text: `${socket.nick} muzzled ${data.nick} in ${socket.channel}, userhash: ${badClient.hash}`
}, { uType: 'mod' });
};
// module hook functions
exports.initHooks = (server) => {
server.registerHook('in', 'chat', this.chatCheck);
server.registerHook('in', 'invite', this.inviteCheck);
// TODO: add whisper hook, need hook priorities todo finished first
};
// hook incoming chat commands, shadow-prevent chat if they are muzzled
exports.chatCheck = (core, server, socket, payload) => {
if (typeof payload.text !== 'string') {
return false;
}
if(core.muzzledHashes[socket.hash]){
// build fake chat payload
mutedPayload = {
cmd: 'chat',
nick: socket.nick,
text: payload.text
};
if (socket.trip) {
mutedPayload.trip = socket.trip;
}
// broadcast to any duplicate connections in channel
server.broadcast( mutedPayload, { channel: socket.channel, hash: socket.hash });
// broadcast to allies, if any
if(core.muzzledHashes[socket.hash].allies){
server.broadcast( mutedPayload, { channel: socket.channel, nick: core.muzzledHashes[socket.hash].allies });
}
// blanket "spam" protection, may expose the ratelimiting lines from `chat` and use that, TODO: one day #lazydev
server._police.frisk(socket.remoteAddress, 9);
return false;
}
return payload;
};
// shadow-prevent all invites from muzzled users
exports.inviteCheck = (core, server, socket, payload) => {
if (typeof payload.nick !== 'string') {
return false;
}
if(core.muzzledHashes[socket.hash]){
// generate common channel
let channel = Math.random().toString(36).substr(2, 8);
// send fake reply
server.reply({
cmd: 'info',
text: `You invited ${payload.nick} to ?${channel}`
}, socket);
return false;
}
return payload;
};
// module meta
exports.requiredData = ['nick'];
exports.info = {
name: 'dumb',
description: 'Globally shadow mute a connection. Optional allies array will see muted messages.',
usage: `
API: { cmd: 'dumb', nick: '<target nick>', allies: ['<optional nick array>', ...] }`
};
exports.info.aliases = ['muzzle', 'mute'];
|