aboutsummaryrefslogtreecommitdiffstats
path: root/server/src/commands/mod/dumb.js
blob: ecb4e0d56411d1cdf153750cddcdd3f40327a412 (plain)
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'];