/** * System commands * Pokemon Showdown - http://pokemonshowdown.com/ * * These are system commands - commands required for Pokemon Showdown * to run. A lot of these are sent by the client. * * System commands should not be modified, added, or removed. If you'd * like to modify or add commands, add or edit files in chat-plugins/ * * For the API, see chat-plugins/COMMANDS.md * * @license MIT license */ var crypto = require('crypto'); var fs = require('fs'); const MAX_REASON_LENGTH = 300; const MUTE_LENGTH = 7 * 60 * 1000; const HOURMUTE_LENGTH = 60 * 60 * 1000; var commands = exports.commands = { version: function (target, room, user) { if (!this.canBroadcast()) return; this.sendReplyBox("Server version: " + CommandParser.package.version + ""); }, auth: 'authority', authlist: 'authority', authority: function (target, room, user, connection) { var rankLists = {}; var ranks = Object.keys(Config.groups); for (var u in Users.usergroups) { var rank = Users.usergroups[u].charAt(0); if (rank === ' ' || rank === '+') continue; // In case the usergroups.csv file is not proper, we check for the server ranks. if (ranks.indexOf(rank) >= 0) { var name = Users.usergroups[u].substr(1); if (!rankLists[rank]) rankLists[rank] = []; if (name) rankLists[rank].push(name); } } var buffer = []; Object.keys(rankLists).sort(function (a, b) { return (Config.groups[b] || {rank: 0}).rank - (Config.groups[a] || {rank: 0}).rank; }).forEach(function (r) { buffer.push((Config.groups[r] ? Config.groups[r].name + "s (" + r + ")" : r) + ":\n" + rankLists[r].sort().join(", ")); }); if (!buffer.length) buffer = "This server has no auth."; connection.popup(buffer.join("\n\n")); }, me: function (target, room, user, connection) { // By default, /me allows a blank message if (target) target = this.canTalk(target); if (!target) return; return '/me ' + target; }, mee: function (target, room, user, connection) { // By default, /mee allows a blank message if (target) target = this.canTalk(target); if (!target) return; return '/mee ' + target; }, avatar: function (target, room, user) { if (!target) return this.parse('/avatars'); var parts = target.split(','); var avatar = parseInt(parts[0]); if (!avatar || avatar > 294 || avatar < 1) { if (!parts[1]) { this.sendReply("Invalid avatar."); } return false; } user.avatar = avatar; if (!parts[1]) { this.sendReply("Avatar changed to:\n" + '|raw|'); } }, avatarhelp: ["/avatar [new avatar number] - Change your trainer sprite."], logout: function (target, room, user) { user.resetName(); }, requesthelp: 'report', report: function (target, room, user) { if (room.id === 'help') { this.sendReply("Ask one of the Moderators (@) in the Help room."); } else { this.parse('/join help'); } }, r: 'reply', reply: function (target, room, user) { if (!target) return this.parse('/help reply'); if (!user.lastPM) { return this.sendReply("No one has PMed you yet."); } return this.parse('/msg ' + (user.lastPM || '') + ', ' + target); }, replyhelp: ["/reply OR /r [message] - Send a private message to the last person you received a message from, or sent a message to."], pm: 'msg', whisper: 'msg', w: 'msg', msg: function (target, room, user, connection) { if (!target) return this.parse('/help msg'); target = this.splitTarget(target); var targetUser = this.targetUser; if (!target) { this.sendReply("You forgot the comma."); return this.parse('/help msg'); } if (!targetUser || !targetUser.connected) { if (targetUser && !targetUser.connected) { this.popupReply("User " + this.targetUsername + " is offline."); } else { this.popupReply("User " + this.targetUsername + " not found. Did you misspell their name?"); } return this.parse('/help msg'); } if (Config.pmmodchat) { var userGroup = user.group; if (Config.groupsranking.indexOf(userGroup) < Config.groupsranking.indexOf(Config.pmmodchat)) { var groupName = Config.groups[Config.pmmodchat].name || Config.pmmodchat; this.popupReply("Because moderated chat is set, you must be of rank " + groupName + " or higher to PM users."); return false; } } if (user.locked && !targetUser.can('lock')) { return this.popupReply("You can only private message members of the moderation team (users marked by %, @, &, or ~) when locked."); } if (targetUser.locked && !user.can('lock')) { return this.popupReply("This user is locked and cannot PM."); } if (targetUser.ignorePMs && targetUser.ignorePMs !== user.group && !user.can('lock')) { if (!targetUser.can('lock')) { return this.popupReply("This user is blocking private messages right now."); } else if (targetUser.can('bypassall')) { return this.popupReply("This admin is too busy to answer private messages right now. Please contact a different staff member."); } } if (user.ignorePMs && user.ignorePMs !== targetUser.group && !targetUser.can('lock')) { return this.popupReply("You are blocking private messages right now."); } target = this.canTalk(target, null); if (!target) return false; if (target.charAt(0) === '/' && target.charAt(1) !== '/') { // PM command var innerCmdIndex = target.indexOf(' '); var innerCmd = (innerCmdIndex >= 0 ? target.slice(1, innerCmdIndex) : target.slice(1)); var innerTarget = (innerCmdIndex >= 0 ? target.slice(innerCmdIndex + 1) : ''); switch (innerCmd) { case 'me': case 'mee': case 'announce': break; case 'invite': case 'inv': var targetRoom = Rooms.search(innerTarget); if (!targetRoom || targetRoom === Rooms.global) return connection.send('|pm|' + user.getIdentity() + '|' + targetUser.getIdentity() + '|/text The room "' + innerTarget + '" does not exist.'); if (targetRoom.staffRoom && !targetUser.isStaff) return connection.send('|pm|' + user.getIdentity() + '|' + targetUser.getIdentity() + '|/text User "' + this.targetUsername + '" requires global auth to join room "' + targetRoom.id + '".'); if (targetRoom.isPrivate === true && targetRoom.modjoin && targetRoom.auth) { if (Config.groupsranking.indexOf(targetRoom.auth[targetUser.userid] || ' ') < Config.groupsranking.indexOf(targetRoom.modjoin) && !targetUser.can('bypassall')) { return connection.send('|pm|' + user.getIdentity() + '|' + targetUser.getIdentity() + '|/text The room "' + innerTarget + '" does not exist.'); } } target = '/invite ' + targetRoom.id; break; default: return connection.send('|pm|' + user.getIdentity() + '|' + targetUser.getIdentity() + "|/text The command '/" + innerCmd + "' was unrecognized or unavailable in private messages. To send a message starting with '/" + innerCmd + "', type '//" + innerCmd + "'."); } } var message = '|pm|' + user.getIdentity() + '|' + targetUser.getIdentity() + '|' + target; user.send(message); if (targetUser !== user) targetUser.send(message); targetUser.lastPM = user.userid; user.lastPM = targetUser.userid; }, msghelp: ["/msg OR /whisper OR /w [username], [message] - Send a private message."], blockpm: 'ignorepms', blockpms: 'ignorepms', ignorepm: 'ignorepms', ignorepms: function (target, room, user) { if (user.ignorePMs === (target || true)) return this.sendReply("You are already blocking private messages!"); if (user.can('lock') && !user.can('bypassall')) return this.sendReply("You are not allowed to block private messages."); user.ignorePMs = true; if (target in Config.groups) { user.ignorePMs = target; return this.sendReply("You are now blocking private messages, except from staff and " + target + "."); } return this.sendReply("You are now blocking private messages, except from staff."); }, ignorepmshelp: ["/blockpms - Blocks private messages. Unblock them with /unignorepms."], unblockpm: 'unignorepms', unblockpms: 'unignorepms', unignorepm: 'unignorepms', unignorepms: function (target, room, user) { if (!user.ignorePMs) return this.sendReply("You are not blocking private messages!"); user.ignorePMs = false; return this.sendReply("You are no longer blocking private messages."); }, unignorepmshelp: ["/unblockpms - Unblocks private messages. Block them with /blockpms."], idle: 'away', away: function (target, room, user) { this.parse('/blockchallenges'); this.parse('/blockpms ' + target); }, awayhelp: ["/away - Blocks challenges and private messages. Unblock them with /back."], unaway: 'back', back: function () { this.parse('/unblockpms'); this.parse('/unblockchallenges'); }, backhelp: ["/back - Unblocks challenges and/or private messages, if either are blocked."], makechatroom: function (target, room, user) { if (!this.can('makeroom')) return; // `,` is a delimiter used by a lot of /commands // `|` and `[` are delimiters used by the protocol // `-` has special meaning in roomids if (target.includes(',') || target.includes('|') || target.includes('[') || target.includes('-')) { return this.sendReply("Room titles can't contain any of: ,|[-"); } var id = toId(target); if (!id) return this.parse('/help makechatroom'); if (Rooms.rooms[id]) return this.sendReply("The room '" + target + "' already exists."); if (Rooms.global.addChatRoom(target)) { return this.sendReply("The room '" + target + "' was created."); } return this.sendReply("An error occurred while trying to create the room '" + target + "'."); }, makechatroomhelp: ["/makechatroom [roomname] - Creates a new room named [roomname]. Requires: ~"], deregisterchatroom: function (target, room, user) { if (!this.can('makeroom')) return; var id = toId(target); if (!id) return this.parse('/help deregisterchatroom'); var targetRoom = Rooms.search(id); if (!targetRoom) return this.sendReply("The room '" + target + "' doesn't exist."); target = targetRoom.title || targetRoom.id; if (Rooms.global.deregisterChatRoom(id)) { this.sendReply("The room '" + target + "' was deregistered."); this.sendReply("It will be deleted as of the next server restart."); return; } return this.sendReply("The room '" + target + "' isn't registered."); }, deregisterchatroomhelp: ["/deregisterchatroom [roomname] - Deletes room [roomname] after the next server restart. Requires: ~"], hideroom: 'privateroom', hiddenroom: 'privateroom', privateroom: function (target, room, user, connection, cmd) { var setting; switch (cmd) { case 'privateroom': if (!this.can('makeroom')) return; setting = true; break; default: if (!this.can('privateroom', null, room)) return; if (room.isPrivate === true) { if (this.can('makeroom')) this.sendReply("This room is a secret room. Use /privateroom to toggle instead."); return; } setting = 'hidden'; break; } if (target === 'off') { delete room.isPrivate; this.addModCommand("" + user.name + " made this room public."); if (room.chatRoomData) { delete room.chatRoomData.isPrivate; Rooms.global.writeChatRoomData(); } } else { room.isPrivate = setting; this.addModCommand("" + user.name + " made this room " + (setting === true ? 'secret' : setting) + "."); if (room.chatRoomData) { room.chatRoomData.isPrivate = setting; Rooms.global.writeChatRoomData(); } } }, privateroomhelp: ["/privateroom [on/off] - Makes or unmakes a room private. Requires: ~", "/hiddenroom [on/off] - Makes or unmakes a room hidden. Hidden rooms will maintain global ranks of users. Requires: \u2605 ~"], modjoin: function (target, room, user) { if (!this.can('privateroom', null, room)) return; if (target === 'off' || target === 'false') { delete room.modjoin; this.addModCommand("" + user.name + " turned off modjoin."); if (room.chatRoomData) { delete room.chatRoomData.modjoin; Rooms.global.writeChatRoomData(); } } else { if ((target === 'on' || target === 'true' || !target) || !user.can('privateroom')) { room.modjoin = true; this.addModCommand("" + user.name + " turned on modjoin."); } else if (target in Config.groups) { room.modjoin = target; this.addModCommand("" + user.name + " set modjoin to " + target + "."); } else { this.sendReply("Unrecognized modjoin setting."); return false; } if (room.chatRoomData) { room.chatRoomData.modjoin = room.modjoin; Rooms.global.writeChatRoomData(); } if (!room.modchat) this.parse('/modchat ' + Config.groupsranking[1]); if (!room.isPrivate) this.parse('/hiddenroom'); } }, officialchatroom: 'officialroom', officialroom: function (target, room, user) { if (!this.can('makeroom')) return; if (!room.chatRoomData) { return this.sendReply("/officialroom - This room can't be made official"); } if (target === 'off') { delete room.isOfficial; this.addModCommand("" + user.name + " made this chat room unofficial."); delete room.chatRoomData.isOfficial; Rooms.global.writeChatRoomData(); } else { room.isOfficial = true; this.addModCommand("" + user.name + " made this chat room official."); room.chatRoomData.isOfficial = true; Rooms.global.writeChatRoomData(); } }, roomdesc: function (target, room, user) { if (!target) { if (!this.canBroadcast()) return; var re = /(https?:\/\/(([-\w\.]+)+(:\d+)?(\/([\w/_\.]*(\?\S+)?)?)?))/g; if (!room.desc) return this.sendReply("This room does not have a description set."); this.sendReplyBox("The room description is: " + room.desc.replace(re, '$1')); return; } if (!this.can('declare')) return false; if (target.length > 80) return this.sendReply("Error: Room description is too long (must be at most 80 characters)."); var normalizedTarget = ' ' + target.toLowerCase().replace('[^a-zA-Z0-9]+', ' ').trim() + ' '; if (normalizedTarget.includes(' welcome ')) { return this.sendReply("Error: Room description must not contain the word 'welcome'."); } if (normalizedTarget.slice(0, 9) === ' discuss ') { return this.sendReply("Error: Room description must not start with the word 'discuss'."); } if (normalizedTarget.slice(0, 12) === ' talk about ' || normalizedTarget.slice(0, 17) === ' talk here about ') { return this.sendReply("Error: Room description must not start with the phrase 'talk about'."); } room.desc = target; this.sendReply("(The room description is now: " + target + ")"); this.privateModCommand("(" + user.name + " changed the roomdesc to: \"" + target + "\".)"); if (room.chatRoomData) { room.chatRoomData.desc = room.desc; Rooms.global.writeChatRoomData(); } }, topic: 'roomintro', roomintro: function (target, room, user) { if (!target) { if (!this.canBroadcast()) return; if (!room.introMessage) return this.sendReply("This room does not have an introduction set."); this.sendReply('|raw|
' + room.introMessage + '
'); if (!this.broadcasting && user.can('declare', null, room)) { this.sendReply('Source:'); this.sendReplyBox('' + Tools.escapeHTML(room.introMessage) + ''); } return; } if (!this.can('declare', null, room)) return false; if (!this.canHTML(target)) return; if (!/$1'); } if (!target.trim()) target = ''; room.introMessage = target; this.sendReply("(The room introduction has been changed to:)"); this.sendReply('|raw|
' + target + '
'); this.privateModCommand("(" + user.name + " changed the roomintro.)"); if (room.chatRoomData) { room.chatRoomData.introMessage = room.introMessage; Rooms.global.writeChatRoomData(); } }, roomalias: function (target, room, user) { if (!room.chatRoomData) return this.sendReply("This room isn't designed for aliases."); if (!target) { if (!room.chatRoomData.aliases || !room.chatRoomData.aliases.length) return this.sendReplyBox("This room does not have any aliases."); return this.sendReplyBox("This room has the following aliases: " + room.chatRoomData.aliases.join(", ") + ""); } if (!this.can('setalias')) return false; var alias = toId(target); if (!alias.length) return this.sendReply("Only alphanumeric characters are valid in an alias."); if (Rooms.get(alias) || Rooms.aliases[alias]) return this.sendReply("You cannot set an alias to an existing room or alias."); this.privateModCommand("(" + user.name + " added the room alias '" + target + "'.)"); if (!room.chatRoomData.aliases) room.chatRoomData.aliases = []; room.chatRoomData.aliases.push(alias); Rooms.aliases[alias] = room; Rooms.global.writeChatRoomData(); }, removeroomalias: function (target, room, user) { if (!room.chatRoomData) return this.sendReply("This room isn't designed for aliases."); if (!room.chatRoomData.aliases) return this.sendReply("This room does not have any aliases."); if (!this.can('setalias')) return false; var alias = toId(target); if (!alias.length || !Rooms.aliases[alias]) return this.sendReply("Please specify an existing alias."); if (Rooms.aliases[alias] !== room) return this.sendReply("You may only remove an alias from the current room."); this.privateModCommand("(" + user.name + " removed the room alias '" + target + "'.)"); var aliasIndex = room.chatRoomData.aliases.indexOf(alias); if (aliasIndex >= 0) { room.chatRoomData.aliases.splice(aliasIndex, 1); delete Rooms.aliases[alias]; Rooms.global.writeChatRoomData(); } }, roomowner: function (target, room, user) { if (!room.chatRoomData) { return this.sendReply("/roomowner - This room isn't designed for per-room moderation to be added"); } target = this.splitTarget(target, true); var targetUser = this.targetUser; if (!targetUser) return this.sendReply("User '" + this.targetUsername + "' is not online."); if (!this.can('makeroom', targetUser, room)) return false; if (!room.auth) room.auth = room.chatRoomData.auth = {}; var name = targetUser.name; room.auth[targetUser.userid] = '#'; this.addModCommand("" + name + " was appointed Room Owner by " + user.name + "."); room.onUpdateIdentity(targetUser); Rooms.global.writeChatRoomData(); }, roomownerhelp: ["/roomowner [username] - Appoints [username] as a room owner. Removes official status. Requires: ~"], roomdeowner: 'deroomowner', deroomowner: function (target, room, user) { if (!room.auth) { return this.sendReply("/roomdeowner - This room isn't designed for per-room moderation"); } target = this.splitTarget(target, true); var targetUser = this.targetUser; var name = this.targetUsername; var userid = toId(name); if (!userid || userid === '') return this.sendReply("User '" + name + "' does not exist."); if (room.auth[userid] !== '#') return this.sendReply("User '" + name + "' is not a room owner."); if (!this.can('makeroom', null, room)) return false; delete room.auth[userid]; this.sendReply("(" + name + " is no longer Room Owner.)"); if (targetUser) targetUser.updateIdentity(); if (room.chatRoomData) { Rooms.global.writeChatRoomData(); } }, deroomownerhelp: ["/roomdeowner [username] - Removes [username]'s status as a room owner. Requires: ~"], roomdemote: 'roompromote', roompromote: function (target, room, user, connection, cmd) { if (!room.auth) { this.sendReply("/roompromote - This room isn't designed for per-room moderation"); return this.sendReply("Before setting room mods, you need to set it up with /roomowner"); } if (!target) return this.parse('/help roompromote'); target = this.splitTarget(target, true); var targetUser = this.targetUser; var userid = toId(this.targetUsername); var name = targetUser ? targetUser.name : this.targetUsername; if (!userid) return this.parse('/help roompromote'); if (!targetUser && (!room.auth || !room.auth[userid])) { return this.sendReply("User '" + name + "' is offline and unauthed, and so can't be promoted."); } var currentGroup = ((room.auth && room.auth[userid]) || (room.isPrivate !== true && targetUser.group) || ' '); var nextGroup = target || Users.getNextGroupSymbol(currentGroup, cmd === 'roomdemote', true); if (target === 'deauth') nextGroup = Config.groupsranking[0]; if (!Config.groups[nextGroup]) { return this.sendReply("Group '" + nextGroup + "' does not exist."); } if (Config.groups[nextGroup].globalonly) { return this.sendReply("Group 'room" + Config.groups[nextGroup].id + "' does not exist as a room rank."); } var groupName = Config.groups[nextGroup].name || "regular user"; if (currentGroup === nextGroup) { return this.sendReply("User '" + name + "' is already a " + groupName + " in this room."); } if (currentGroup !== ' ' && !user.can('room' + (Config.groups[currentGroup] ? Config.groups[currentGroup].id : 'voice'), null, room)) { return this.sendReply("/" + cmd + " - Access denied for promoting/demoting from " + (Config.groups[currentGroup] ? Config.groups[currentGroup].name : "an undefined group") + "."); } if (nextGroup !== ' ' && !user.can('room' + Config.groups[nextGroup].id, null, room)) { return this.sendReply("/" + cmd + " - Access denied for promoting/demoting to " + Config.groups[nextGroup].name + "."); } if (nextGroup === ' ') { delete room.auth[userid]; } else { room.auth[userid] = nextGroup; } if (Config.groups[nextGroup].rank < Config.groups[currentGroup].rank) { this.privateModCommand("(" + name + " was demoted to Room " + groupName + " by " + user.name + ".)"); if (targetUser && Rooms.rooms[room.id].users[targetUser.userid]) targetUser.popup("You were demoted to Room " + groupName + " by " + user.name + "."); } else if (nextGroup === '#') { this.addModCommand("" + name + " was promoted to " + groupName + " by " + user.name + "."); } else { this.addModCommand("" + name + " was promoted to Room " + groupName + " by " + user.name + "."); } if (targetUser) targetUser.updateIdentity(room.id); if (room.chatRoomData) Rooms.global.writeChatRoomData(); }, roompromotehelp: ["/roompromote [username], [group] - Promotes the user to the specified group or next ranked group. Requires: @ # & ~"], roomdemotehelp: ["/roomdemote [username], [group] - Demotes the user to the specified group or previous ranked group. Requires: @ # & ~"], roomauth: function (target, room, user, connection) { var targetRoom = room; if (target) targetRoom = Rooms.search(target); if (!targetRoom || (targetRoom !== room && targetRoom.modjoin && !user.can('bypassall'))) return this.sendReply("The room '" + target + "' does not exist."); if (!targetRoom.auth) return this.sendReply("/roomauth - The room '" + (targetRoom.title ? targetRoom.title : target) + "' isn't designed for per-room moderation and therefore has no auth list."); var rankLists = {}; for (var u in targetRoom.auth) { if (!rankLists[targetRoom.auth[u]]) rankLists[targetRoom.auth[u]] = []; rankLists[targetRoom.auth[u]].push(u); } var buffer = []; Object.keys(rankLists).sort(function (a, b) { return (Config.groups[b] || {rank:0}).rank - (Config.groups[a] || {rank:0}).rank; }).forEach(function (r) { buffer.push((Config.groups[r] ? Config.groups[r] .name + "s (" + r + ")" : r) + ":\n" + rankLists[r].sort().join(", ")); }); if (!buffer.length) { connection.popup("The room '" + targetRoom.title + "' has no auth."); return; } if (targetRoom !== room) buffer.unshift("" + targetRoom.title + " room auth:"); connection.popup(buffer.join("\n\n")); }, userauth: function (target, room, user, connection) { var targetId = toId(target) || user.userid; var targetUser = Users.getExact(targetId); var targetUsername = (targetUser ? targetUser.name : target); var buffer = []; var innerBuffer = []; var group = Users.usergroups[targetId]; if (group) { buffer.push('Global auth: ' + group.charAt(0)); } for (var i = 0; i < Rooms.global.chatRooms.length; i++) { var curRoom = Rooms.global.chatRooms[i]; if (!curRoom.auth || curRoom.isPrivate) continue; group = curRoom.auth[targetId]; if (!group) continue; innerBuffer.push(group + curRoom.id); } if (innerBuffer.length) { buffer.push('Room auth: ' + innerBuffer.join(', ')); } if (targetId === user.userid || user.can('makeroom')) { innerBuffer = []; for (var i = 0; i < Rooms.global.chatRooms.length; i++) { var curRoom = Rooms.global.chatRooms[i]; if (!curRoom.auth || !curRoom.isPrivate) continue; var auth = curRoom.auth[targetId]; if (!auth) continue; innerBuffer.push(auth + curRoom.id); } if (innerBuffer.length) { buffer.push('Private room auth: ' + innerBuffer.join(', ')); } } if (!buffer.length) { buffer.push("No global or room auth."); } buffer.unshift("" + targetUsername + " user auth:"); connection.popup(buffer.join("\n\n")); }, rb: 'roomban', roomban: function (target, room, user, connection) { if (!target) return this.parse('/help roomban'); if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); target = this.splitTarget(target, true); var targetUser = this.targetUser; var name = this.targetUsername; var userid = toId(name); if (!userid || !targetUser) return this.sendReply("User '" + name + "' does not exist."); if (!this.can('ban', targetUser, room)) return false; if (!room.bannedUsers || !room.bannedIps) { return this.sendReply("Room bans are not meant to be used in room " + room.id + "."); } if (room.bannedUsers[userid] && room.bannedIps[targetUser.latestIp]) return this.sendReply("User " + targetUser.name + " is already banned from room " + room.id + "."); targetUser.popup("" + user.name + " has banned you from the room " + room.id + "." + (target ? "\n\nReason: " + target + "" : "") + "\n\nTo appeal the ban, PM the staff member that banned you or a room owner. If you are unsure who the room owners are, type this into any room: /roomauth " + room.id); this.addModCommand("" + targetUser.name + " was banned from room " + room.id + " by " + user.name + "." + (target ? " (" + target + ")" : "")); var alts = room.roomBan(targetUser); if (alts.length) { this.privateModCommand("(" + targetUser.name + "'s alts were also banned from room " + room.id + ": " + alts.join(", ") + ")"); for (var i = 0; i < alts.length; ++i) { this.add('|unlink|' + toId(alts[i])); } } this.add('|unlink|' + this.getLastIdOf(targetUser)); }, roombanhelp: ["/roomban [username] - Bans the user from the room you are in. Requires: @ & ~"], unroomban: 'roomunban', roomunban: function (target, room, user, connection) { if (!target) return this.parse('/help roomunban'); if (!room.bannedUsers || !room.bannedIps) { return this.sendReply("Room bans are not meant to be used in room " + room.id + "."); } if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); this.splitTarget(target, true); var targetUser = this.targetUser; var userid = room.isRoomBanned(targetUser) || toId(target); if (!userid) return this.sendReply("User '" + target + "' is an invalid username."); if (!this.can('ban', targetUser, room)) return false; var unbannedUserid = room.unRoomBan(userid); if (!unbannedUserid) return this.sendReply("User " + userid + " is not banned from room " + room.id + "."); if (targetUser) targetUser.popup("" + user.name + " has unbanned you from the room " + room.id + "."); this.addModCommand("" + unbannedUserid + " was unbanned from room " + room.id + " by " + user.name + "."); }, roomunbanhelp: ["/roomunban [username] - Unbans the user from the room you are in. Requires: @ & ~"], autojoin: function (target, room, user, connection) { Rooms.global.autojoinRooms(user, connection); }, joim: 'join', j: 'join', join: function (target, room, user, connection) { if (!target) return false; var targetRoom = Rooms.search(target); if (!targetRoom) { return connection.sendTo(target, "|noinit|nonexistent|The room '" + target + "' does not exist."); } if (targetRoom.modjoin && !user.can('bypassall')) { var userGroup = user.group; if (targetRoom.auth) { if (targetRoom.isPrivate === true) { userGroup = ' '; } userGroup = targetRoom.auth[user.userid] || userGroup; } if (Config.groupsranking.indexOf(userGroup) < Config.groupsranking.indexOf(targetRoom.modjoin !== true ? targetRoom.modjoin : targetRoom.modchat)) { return connection.sendTo(target, "|noinit|nonexistent|The room '" + target + "' does not exist."); } } if (targetRoom.isPrivate) { if (!user.named) { return connection.sendTo(target, "|noinit|namerequired|You must have a name in order to join the room '" + target + "'."); } } if (toId(target) !== targetRoom.id) { connection.send(">" + toId(target) + "\n|deinit"); } var joinResult = user.joinRoom(targetRoom, connection); if (!joinResult) { if (joinResult === null) { return connection.sendTo(target, "|noinit|joinfailed|You are banned from the room '" + target + "'."); } return connection.sendTo(target, "|noinit|joinfailed|You do not have permission to join '" + target + "'."); } }, leave: 'part', part: function (target, room, user, connection) { if (room.id === 'global') return false; var targetRoom = Rooms.search(target); if (target && !targetRoom) { return this.sendReply("The room '" + target + "' does not exist."); } user.leaveRoom(targetRoom || room, connection); }, /********************************************************* * Moderating: Punishments *********************************************************/ kick: 'warn', k: 'warn', warn: function (target, room, user) { if (!target) return this.parse('/help warn'); if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); target = this.splitTarget(target); var targetUser = this.targetUser; if (!targetUser || !targetUser.connected) return this.sendReply("User '" + this.targetUsername + "' does not exist."); if (room.isPrivate === true && room.auth) { return this.sendReply("You can't warn here: This is a privately-owned room not subject to global rules."); } if (!Rooms.rooms[room.id].users[targetUser.userid]) { return this.sendReply("User " + this.targetUsername + " is not in the room " + room.id + "."); } if (target.length > MAX_REASON_LENGTH) { return this.sendReply("The reason is too long. It cannot exceed " + MAX_REASON_LENGTH + " characters."); } if (!this.can('warn', targetUser, room)) return false; this.addModCommand("" + targetUser.name + " was warned by " + user.name + "." + (target ? " (" + target + ")" : "")); targetUser.send('|c|~|/warn ' + target); this.add('|unlink|' + this.getLastIdOf(targetUser)); }, warnhelp: ["/warn OR /k [username], [reason] - Warns a user showing them the Pokemon Showdown Rules and [reason] in an overlay. Requires: % @ & ~"], redirect: 'redir', redir: function (target, room, user, connection) { if (!target) return this.parse('/help redirect'); if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); target = this.splitTarget(target); var targetUser = this.targetUser; var targetRoom = Rooms.search(target); if (!targetRoom) { return this.sendReply("The room '" + target + "' does not exist."); } if (!this.can('warn', targetUser, room) || !this.can('warn', targetUser, targetRoom)) return false; if (!targetUser || !targetUser.connected) { return this.sendReply("User " + this.targetUsername + " not found."); } if (targetRoom.id === "global") return this.sendReply("Users cannot be redirected to the global room."); if (Rooms.rooms[targetRoom.id].users[targetUser.userid]) { return this.sendReply("User " + targetUser.name + " is already in the room " + targetRoom.title + "!"); } if (!Rooms.rooms[room.id].users[targetUser.userid]) { return this.sendReply("User " + this.targetUsername + " is not in the room " + room.id + "."); } if (targetUser.joinRoom(targetRoom.id) === false) return this.sendReply("User " + targetUser.name + " could not be joined to room " + targetRoom.title + ". They could be banned from the room."); var roomName = (targetRoom.isPrivate) ? "a private room" : "room " + targetRoom.title; this.addModCommand("" + targetUser.name + " was redirected to " + roomName + " by " + user.name + "."); targetUser.leaveRoom(room); }, redirhelp: ["/redirect OR /redir [username], [roomname] - Attempts to redirect the user [username] to the room [roomname]. Requires: % @ & ~"], m: 'mute', mute: function (target, room, user, connection, cmd) { if (!target) return this.parse('/help mute'); if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); target = this.splitTarget(target); var targetUser = this.targetUser; if (!targetUser) return this.sendReply("User '" + this.targetUsername + "' does not exist."); if (target.length > MAX_REASON_LENGTH) { return this.sendReply("The reason is too long. It cannot exceed " + MAX_REASON_LENGTH + " characters."); } var muteDuration = ((cmd === 'hm' || cmd === 'hourmute') ? HOURMUTE_LENGTH : MUTE_LENGTH); if (!this.can('mute', targetUser, room)) return false; var canBeMutedFurther = (targetUser.mutedRooms[room.id] && (targetUser.muteDuration[room.id] || 0) <= (muteDuration * 5 / 6)); if ((targetUser.mutedRooms[room.id] && !canBeMutedFurther) || targetUser.locked || !targetUser.connected) { var problem = " but was already " + (!targetUser.connected ? "offline" : targetUser.locked ? "locked" : "muted"); if (!target) { return this.privateModCommand("(" + targetUser.name + " would be muted by " + user.name + problem + ".)"); } return this.addModCommand("" + targetUser.name + " would be muted by " + user.name + problem + "." + (target ? " (" + target + ")" : "")); } targetUser.popup("" + user.name + " has muted you for " + (muteDuration / (60 * 1000)) + " minutes. " + target); this.addModCommand("" + targetUser.name + " was muted by " + user.name + " for " + (muteDuration / (60 * 1000)) + " minutes." + (target ? " (" + target + ")" : "")); var alts = targetUser.getAlts(); if (alts.length) this.privateModCommand("(" + targetUser.name + "'s alts were also muted: " + alts.join(", ") + ")"); this.add('|unlink|' + this.getLastIdOf(targetUser)); targetUser.mute(room.id, muteDuration, true); }, mutehelp: ["/mute OR /m [username], [reason] - Mutes a user with reason for 7 minutes. Requires: % @ & ~"], hm: 'hourmute', hourmute: function () { CommandParser.commands.mute.apply(this, arguments); }, hourmutehelp: ["/hourmute OR /hm [username], [reason] - Mutes a user with reason for an hour. Requires: % @ & ~"], um: 'unmute', unmute: function (target, room, user) { if (!target) return this.parse('/help unmute'); if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); var targetUser = Users.get(target); if (!targetUser) return this.sendReply("User '" + target + "' does not exist."); if (!this.can('mute', targetUser, room)) return false; if (!targetUser.mutedRooms[room.id]) { return this.sendReply("" + targetUser.name + " is not muted."); } this.addModCommand("" + targetUser.name + " was unmuted by " + user.name + "."); targetUser.unmute(room.id); }, unmutehelp: ["/unmute [username] - Removes mute from user. Requires: % @ & ~"], forcelock: 'lock', l: 'lock', ipmute: 'lock', lock: function (target, room, user, connection, cmd) { if (!target) return this.parse('/help lock'); if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); target = this.splitTarget(target); var targetUser = this.targetUser; if (!targetUser) return this.sendReply("User '" + this.targetUsername + "' does not exist."); if (target.length > MAX_REASON_LENGTH) { return this.sendReply("The reason is too long. It cannot exceed " + MAX_REASON_LENGTH + " characters."); } if (!this.can('lock', targetUser)) return false; if ((targetUser.locked || Users.checkBanned(targetUser.latestIp)) && !target) { var problem = " but was already " + (targetUser.locked ? "locked" : "banned"); return this.privateModCommand("(" + targetUser.name + " would be locked by " + user.name + problem + ".)"); } if (targetUser.confirmed) { if (cmd === 'forcelock') { var from = targetUser.deconfirm(); ResourceMonitor.log("[CrisisMonitor] " + targetUser.name + " was locked by " + user.name + " and demoted from " + from.join(", ") + "."); } else { return this.sendReply("" + targetUser.name + " is a confirmed user. If you are sure you would like to lock them use /forcelock."); } } else if (cmd === 'forcelock') { return this.sendReply("Use /lock; " + targetUser.name + " is not a confirmed user."); } targetUser.popup("" + user.name + " has locked you from talking in chats, battles, and PMing regular users." + (target ? "\n\nReason: " + target : "") + "\n\nIf you feel that your lock was unjustified, you can still PM staff members (%, @, &, and ~) to discuss it" + (Config.appealurl ? " or you can appeal:\n" + Config.appealurl : ".") + "\n\nYour lock will expire in a few days."); this.addModCommand("" + targetUser.name + " was locked from talking by " + user.name + "." + (target ? " (" + target + ")" : "")); var alts = targetUser.getAlts(); var acAccount = (targetUser.autoconfirmed !== targetUser.userid && targetUser.autoconfirmed); if (alts.length) { this.privateModCommand("(" + targetUser.name + "'s " + (acAccount ? " ac account: " + acAccount + ", " : "") + "locked alts: " + alts.join(", ") + ")"); } else if (acAccount) { this.privateModCommand("(" + targetUser.name + "'s ac account: " + acAccount + ")"); } this.add('|unlink|hide|' + this.getLastIdOf(targetUser)); targetUser.lock(); }, lockhelp: ["/lock OR /l [username], [reason] - Locks the user from talking in all chats. Requires: % @ & ~"], unlock: function (target, room, user) { if (!target) return this.parse('/help unlock'); if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); if (!this.can('lock')) return false; var unlocked = Users.unlock(target); if (unlocked) { var names = Object.keys(unlocked); this.addModCommand(names.join(", ") + " " + ((names.length > 1) ? "were" : "was") + " unlocked by " + user.name + "."); } else { this.sendReply("User '" + target + "' is not locked."); } }, unlockhelp: ["/unlock [username] - Unlocks the user. Requires: % @ & ~"], forceban: 'ban', b: 'ban', ban: function (target, room, user, connection, cmd) { if (!target) return this.parse('/help ban'); if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); target = this.splitTarget(target); var targetUser = this.targetUser; if (!targetUser) return this.sendReply("User '" + this.targetUsername + "' does not exist."); if (target.length > MAX_REASON_LENGTH) { return this.sendReply("The reason is too long. It cannot exceed " + MAX_REASON_LENGTH + " characters."); } if (!this.can('ban', targetUser)) return false; if (Users.checkBanned(targetUser.latestIp) && !target && !targetUser.connected) { var problem = " but was already banned"; return this.privateModCommand("(" + targetUser.name + " would be banned by " + user.name + problem + ".)"); } if (targetUser.confirmed) { if (cmd === 'forceban') { var from = targetUser.deconfirm(); ResourceMonitor.log("[CrisisMonitor] " + targetUser.name + " was banned by " + user.name + " and demoted from " + from.join(", ") + "."); } else { return this.sendReply("" + targetUser.name + " is a confirmed user. If you are sure you would like to ban them use /forceban."); } } else if (cmd === 'forceban') { return this.sendReply("Use /ban; " + targetUser.name + " is not a confirmed user."); } targetUser.popup("" + user.name + " has banned you." + (target ? "\n\nReason: " + target : "") + (Config.appealurl ? "\n\nIf you feel that your ban was unjustified, you can appeal:\n" + Config.appealurl : "") + "\n\nYour ban will expire in a few days."); this.addModCommand("" + targetUser.name + " was banned by " + user.name + "." + (target ? " (" + target + ")" : ""), " (" + targetUser.latestIp + ")"); var alts = targetUser.getAlts(); var acAccount = (targetUser.autoconfirmed !== targetUser.userid && targetUser.autoconfirmed); if (alts.length) { this.privateModCommand("(" + targetUser.name + "'s " + (acAccount ? " ac account: " + acAccount + ", " : "") + "banned alts: " + alts.join(", ") + ")"); for (var i = 0; i < alts.length; ++i) { this.add('|unlink|' + toId(alts[i])); } } else if (acAccount) { this.privateModCommand("(" + targetUser.name + "'s ac account: " + acAccount + ")"); } this.add('|unlink|hide|' + this.getLastIdOf(targetUser)); targetUser.ban(); }, banhelp: ["/ban OR /b [username], [reason] - Kick user from all rooms and ban user's IP address with reason. Requires: @ & ~"], unban: function (target, room, user) { if (!target) return this.parse('/help unban'); if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); if (!this.can('ban')) return false; var name = Users.unban(target); if (name) { this.addModCommand("" + name + " was unbanned by " + user.name + "."); } else { this.sendReply("User '" + target + "' is not banned."); } }, unbanhelp: ["/unban [username] - Unban a user. Requires: @ & ~"], unbanall: function (target, room, user) { if (!this.can('rangeban')) return false; if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); // we have to do this the hard way since it's no longer a global for (var i in Users.bannedIps) { delete Users.bannedIps[i]; } for (var i in Users.lockedIps) { delete Users.lockedIps[i]; } this.addModCommand("All bans and locks have been lifted by " + user.name + "."); }, unbanallhelp: ["/unbanall - Unban all IP addresses. Requires: & ~"], banip: function (target, room, user) { if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); target = target.trim(); if (!target) { return this.parse('/help banip'); } if (!this.can('rangeban')) return false; if (Users.bannedIps[target] === '#ipban') return this.sendReply("The IP " + (target.charAt(target.length - 1) === '*' ? "range " : "") + target + " has already been temporarily banned."); Users.bannedIps[target] = '#ipban'; this.addModCommand("" + user.name + " temporarily banned the " + (target.charAt(target.length - 1) === '*' ? "IP range" : "IP") + ": " + target); }, baniphelp: ["/banip [ip] - Kick users on this IP or IP range from all rooms and bans it. Accepts wildcards to ban ranges. Requires: & ~"], unbanip: function (target, room, user) { if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); target = target.trim(); if (!target) { return this.parse('/help unbanip'); } if (!this.can('rangeban')) return false; if (!Users.bannedIps[target]) { return this.sendReply("" + target + " is not a banned IP or IP range."); } delete Users.bannedIps[target]; this.addModCommand("" + user.name + " unbanned the " + (target.charAt(target.length - 1) === '*' ? "IP range" : "IP") + ": " + target); }, unbaniphelp: ["/unbanip [ip] - Kick users on this IP or IP range from all rooms and bans it. Accepts wildcards to ban ranges. Requires: & ~"], rangelock: function (target, room, user) { if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); if (!target) return this.sendReply("Please specify a range to lock."); if (!this.can('rangeban')) return false; var isIp = (target.slice(-1) === '*' ? true : false); var range = (isIp ? target : Users.shortenHost(target)); if (Users.lockedRanges[range]) return this.sendReply("The range " + range + " has already been temporarily locked."); Users.lockRange(range, isIp); this.addModCommand("" + user.name + " temporarily locked the range " + range + "."); }, unrangelock: 'rangeunlock', rangeunlock: function (target, room, user) { if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); if (!target) return this.sendReply("Please specify a range to unlock."); if (!this.can('rangeban')) return false; var range = (target.slice(-1) === '*' ? target : Users.shortenHost(target)); if (!Users.lockedRanges[range]) return this.sendReply("The range " + range + " is not locked."); Users.unlockRange(range); this.addModCommand("" + user.name + " unlocked the range " + range + "."); }, /********************************************************* * Moderating: Other *********************************************************/ mn: 'modnote', modnote: function (target, room, user, connection) { if (!target) return this.parse('/help modnote'); if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); if (target.length > MAX_REASON_LENGTH) { return this.sendReply("The note is too long. It cannot exceed " + MAX_REASON_LENGTH + " characters."); } if (!this.can('receiveauthmessages', null, room)) return false; return this.privateModCommand("(" + user.name + " notes: " + target + ")"); }, modnotehelp: ["/modnote [note] - Adds a moderator note that can be read through modlog. Requires: % @ & ~"], globalpromote: 'promote', promote: function (target, room, user, connection, cmd) { if (!target) return this.parse('/help promote'); target = this.splitTarget(target, true); var targetUser = this.targetUser; var userid = toId(this.targetUsername); var name = targetUser ? targetUser.name : this.targetUsername; if (!userid) return this.parse('/help promote'); var currentGroup = ((targetUser && targetUser.group) || Users.usergroups[userid] || ' ')[0]; var nextGroup = target ? target : Users.getNextGroupSymbol(currentGroup, cmd === 'demote', true); if (target === 'deauth') nextGroup = Config.groupsranking[0]; if (!Config.groups[nextGroup]) { return this.sendReply("Group '" + nextGroup + "' does not exist."); } if (Config.groups[nextGroup].roomonly) { return this.sendReply("Group '" + nextGroup + "' does not exist as a global rank."); } var groupName = Config.groups[nextGroup].name || "regular user"; if (currentGroup === nextGroup) { return this.sendReply("User '" + name + "' is already a " + groupName); } if (!user.canPromote(currentGroup, nextGroup)) { return this.sendReply("/" + cmd + " - Access denied."); } if (!Users.setOfflineGroup(name, nextGroup)) { return this.sendReply("/promote - WARNING: This user is offline and could be unregistered. Use /forcepromote if you're sure you want to risk it."); } if (Config.groups[nextGroup].rank < Config.groups[currentGroup].rank) { this.privateModCommand("(" + name + " was demoted to " + groupName + " by " + user.name + ".)"); if (targetUser) targetUser.popup("You were demoted to " + groupName + " by " + user.name + "."); } else { this.addModCommand("" + name + " was promoted to " + groupName + " by " + user.name + "."); } if (targetUser) targetUser.updateIdentity(); }, promotehelp: ["/promote [username], [group] - Promotes the user to the specified group or next ranked group. Requires: & ~"], globaldemote: 'demote', demote: function () { CommandParser.commands.promote.apply(this, arguments); }, demotehelp: ["/demote [username], [group] - Demotes the user to the specified group or previous ranked group. Requires: & ~"], forcepromote: function (target, room, user) { // warning: never document this command in /help if (!this.can('forcepromote')) return false; target = this.splitTarget(target, true); var name = this.targetUsername; var nextGroup = target || Users.getNextGroupSymbol(' ', false); if (!Config.groups[nextGroup]) return this.sendReply("Group '" + nextGroup + "' does not exist."); if (!Users.setOfflineGroup(name, nextGroup, true)) { return this.sendReply("/forcepromote - Don't forcepromote unless you have to."); } this.addModCommand("" + name + " was promoted to " + (Config.groups[nextGroup].name || "regular user") + " by " + user.name + "."); }, deauth: function (target, room, user) { return this.parse('/demote ' + target + ', deauth'); }, deroomauth: 'roomdeauth', roomdeauth: function (target, room, user) { return this.parse('/roomdemote ' + target + ', deauth'); }, modchat: function (target, room, user) { if (!target) return this.sendReply("Moderated chat is currently set to: " + room.modchat); if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); if (!this.can('modchat', null, room)) return false; if (room.modchat && room.modchat.length <= 1 && Config.groupsranking.indexOf(room.modchat) > 1 && !user.can('modchatall', null, room)) { return this.sendReply("/modchat - Access denied for removing a setting higher than " + Config.groupsranking[1] + "."); } target = target.toLowerCase(); var currentModchat = room.modchat; switch (target) { case 'off': case 'false': case 'no': case ' ': room.modchat = false; break; case 'ac': case 'autoconfirmed': room.modchat = 'autoconfirmed'; break; case '*': case 'player': target = '\u2605'; /* falls through */ default: if (!Config.groups[target]) { return this.parse('/help modchat'); } if (Config.groupsranking.indexOf(target) > 1 && !user.can('modchatall', null, room)) { return this.sendReply("/modchat - Access denied for setting higher than " + Config.groupsranking[1] + "."); } room.modchat = target; break; } if (currentModchat === room.modchat) { return this.sendReply("Modchat is already set to " + currentModchat + "."); } if (!room.modchat) { this.add("|raw|
Moderated chat was disabled!
Anyone may talk now.
"); } else { var modchat = Tools.escapeHTML(room.modchat); this.add("|raw|
Moderated chat was set to " + modchat + "!
Only users of rank " + modchat + " and higher can talk.
"); } this.logModCommand(user.name + " set modchat to " + room.modchat); if (room.chatRoomData) { room.chatRoomData.modchat = room.modchat; Rooms.global.writeChatRoomData(); } }, modchathelp: ["/modchat [off/autoconfirmed/+/%/@/&/~] - Set the level of moderated chat. Requires: @ for off/autoconfirmed/+ options, & ~ for all the options"], declare: function (target, room, user) { if (!target) return this.parse('/help declare'); if (!this.can('declare', null, room)) return false; if (!this.canTalk()) return; this.add('|raw|
' + Tools.escapeHTML(target) + '
'); this.logModCommand(user.name + " declared " + target); }, declarehelp: ["/declare [message] - Anonymously announces a message. Requires: & ~"], htmldeclare: function (target, room, user) { if (!target) return this.parse('/help htmldeclare'); if (!this.can('gdeclare', null, room)) return false; if (!this.canTalk()) return; this.add('|raw|
' + target + '
'); this.logModCommand(user.name + " declared " + target); }, gdeclare: 'globaldeclare', globaldeclare: function (target, room, user) { if (!target) return this.parse('/help globaldeclare'); if (!this.can('gdeclare')) return false; for (var id in Rooms.rooms) { if (id !== 'global') Rooms.rooms[id].addRaw('
' + target + '
'); } this.logModCommand(user.name + " globally declared " + target); }, globaldeclarehelp: ["/globaldeclare [message] - Anonymously announces a message to every room on the server. Requires: ~"], cdeclare: 'chatdeclare', chatdeclare: function (target, room, user) { if (!target) return this.parse('/help chatdeclare'); if (!this.can('gdeclare')) return false; for (var id in Rooms.rooms) { if (id !== 'global') if (Rooms.rooms[id].type !== 'battle') Rooms.rooms[id].addRaw('
' + target + '
'); } this.logModCommand(user.name + " globally declared (chat level) " + target); }, chatdeclarehelp: ["/cdeclare [message] - Anonymously announces a message to all chatrooms on the server. Requires: ~"], wall: 'announce', announce: function (target, room, user) { if (!target) return this.parse('/help announce'); if (!this.can('announce', null, room)) return false; target = this.canTalk(target); if (!target) return; return '/announce ' + target; }, announcehelp: ["/announce OR /wall [message] - Makes an announcement. Requires: % @ & ~"], fr: 'forcerename', forcerename: function (target, room, user) { if (!target) return this.parse('/help forcerename'); if ((user.locked || user.mutedRooms[room.id]) && !user.can('bypassall')) return this.sendReply("You cannot do this while unable to talk."); var commaIndex = target.indexOf(','); var targetUser, reason; if (commaIndex >= 0) { reason = target.substr(commaIndex + 1).trim(); target = target.substr(0, commaIndex).trim(); } targetUser = Users.get(target); if (!targetUser) return this.sendReply("User '" + target + "' not found."); if (!this.can('forcerename', targetUser)) return false; if (targetUser.userid !== toId(target)) { return this.sendReply("User '" + target + "' had already changed its name to '" + targetUser.name + "'."); } var entry = targetUser.name + " was forced to choose a new name by " + user.name + (reason ? ": " + reason : ""); this.privateModCommand("(" + entry + ")"); Rooms.global.cancelSearch(targetUser); targetUser.resetName(); targetUser.send("|nametaken||" + user.name + " considers your name inappropriate" + (reason ? ": " + reason : ".")); }, forcerenamehelp: ["/forcerename OR /fr [username], [reason] - Forcibly change a user's name and shows them the [reason]. Requires: % @ & ~"], modlog: function (target, room, user, connection) { var lines = 0; // Specific case for modlog command. Room can be indicated with a comma, lines go after the comma. // Otherwise, the text is defaulted to text search in current room's modlog. var roomId = room.id; var hideIps = !user.can('ban'); var path = require('path'); var isWin = process.platform === 'win32'; var logPath = 'logs/modlog/'; if (target.includes(',')) { var targets = target.split(','); target = targets[1].trim(); roomId = toId(targets[0]) || room.id; } // Let's check the number of lines to retrieve or if it's a word instead if (!target.match('[^0-9]')) { lines = parseInt(target || 15, 10); if (lines > 100) lines = 100; } var wordSearch = (!lines || lines < 0); // Control if we really, really want to check all modlogs for a word. var roomNames = ''; var filename = ''; var command = ''; if (roomId === 'all' && wordSearch) { if (!this.can('modlog')) return; roomNames = "all rooms"; // Get a list of all the rooms var fileList = fs.readdirSync('logs/modlog'); for (var i = 0; i < fileList.length; ++i) { filename += path.normalize(__dirname + '/' + logPath + fileList[i]) + ' '; } } else { if (!this.can('modlog', null, Rooms.get(roomId))) return; roomNames = "the room " + roomId; filename = path.normalize(__dirname + '/' + logPath + 'modlog_' + roomId + '.txt'); } // Seek for all input rooms for the lines or text if (isWin) { command = path.normalize(__dirname + '/lib/winmodlog') + ' tail ' + lines + ' ' + filename; } else { command = 'tail -' + lines + ' ' + filename; } var grepLimit = 100; if (wordSearch) { // searching for a word instead if (target.match(/^["'].+["']$/)) target = target.substring(1, target.length - 1); if (isWin) { command = path.normalize(__dirname + '/lib/winmodlog') + ' ws ' + grepLimit + ' "' + target.replace(/%/g, "%%").replace(/([\^"&<>\|])/g, "^$1") + '" ' + filename; } else { command = "awk '{print NR,$0}' " + filename + " | sort -nr | cut -d' ' -f2- | grep -m" + grepLimit + " -i '" + target.replace(/\\/g, '\\\\\\\\').replace(/["'`]/g, '\'\\$&\'').replace(/[\{\}\[\]\(\)\$\^\.\?\+\-\*]/g, '[$&]') + "'"; } } // Execute the file search to see modlog require('child_process').exec(command, function (error, stdout, stderr) { if (error && stderr) { connection.popup("/modlog empty on " + roomNames + " or erred"); console.log("/modlog error: " + error); return false; } if (stdout && hideIps) { stdout = stdout.replace(/\([0-9]+\.[0-9]+\.[0-9]+\.[0-9]+\)/g, ''); } if (lines) { if (!stdout) { connection.popup("The modlog is empty. (Weird.)"); } else { connection.popup("Displaying the last " + lines + " lines of the Moderator Log of " + roomNames + ":\n\n" + stdout); } } else { if (!stdout) { connection.popup("No moderator actions containing '" + target + "' were found on " + roomNames + "."); } else { connection.popup("Displaying the last " + grepLimit + " logged actions containing '" + target + "' on " + roomNames + ":\n\n" + stdout); } } }); }, modloghelp: ["/modlog [roomid|all], [n] - Roomid defaults to current room.", "If n is a number or omitted, display the last n lines of the moderator log. Defaults to 15.", "If n is not a number, search the moderator log for 'n' on room's log [roomid]. If you set [all] as [roomid], searches for 'n' on all rooms's logs. Requires: % @ & ~"], /********************************************************* * Server management commands *********************************************************/ hotpatch: function (target, room, user) { if (!target) return this.parse('/help hotpatch'); if (!this.can('hotpatch')) return false; this.logEntry(user.name + " used /hotpatch " + target); if (target === 'chat' || target === 'commands') { try { CommandParser.uncacheTree('./command-parser.js'); global.CommandParser = require('./command-parser.js'); var runningTournaments = Tournaments.tournaments; CommandParser.uncacheTree('./tournaments'); global.Tournaments = require('./tournaments'); Tournaments.tournaments = runningTournaments; return this.sendReply("Chat commands have been hot-patched."); } catch (e) { return this.sendReply("Something failed while trying to hotpatch chat: \n" + e.stack); } } else if (target === 'tournaments') { try { var runningTournaments = Tournaments.tournaments; CommandParser.uncacheTree('./tournaments'); global.Tournaments = require('./tournaments'); Tournaments.tournaments = runningTournaments; return this.sendReply("Tournaments have been hot-patched."); } catch (e) { return this.sendReply("Something failed while trying to hotpatch tournaments: \n" + e.stack); } } else if (target === 'battles') { Simulator.SimulatorProcess.respawn(); return this.sendReply("Battles have been hotpatched. Any battles started after now will use the new code; however, in-progress battles will continue to use the old code."); } else if (target === 'formats') { try { // uncache the tools.js dependency tree CommandParser.uncacheTree('./tools.js'); // reload tools.js global.Tools = require('./tools.js'); // note: this will lock up the server for a few seconds // rebuild the formats list Rooms.global.formatListText = Rooms.global.getFormatListText(); // respawn validator processes TeamValidator.ValidatorProcess.respawn(); // respawn simulator processes Simulator.SimulatorProcess.respawn(); // broadcast the new formats list to clients Rooms.global.send(Rooms.global.formatListText); return this.sendReply("Formats have been hotpatched."); } catch (e) { return this.sendReply("Something failed while trying to hotpatch formats: \n" + e.stack); } } else if (target === 'learnsets') { try { // uncache the tools.js dependency tree CommandParser.uncacheTree('./tools.js'); // reload tools.js global.Tools = require('./tools.js'); // note: this will lock up the server for a few seconds return this.sendReply("Learnsets have been hotpatched."); } catch (e) { return this.sendReply("Something failed while trying to hotpatch learnsets: \n" + e.stack); } } this.sendReply("Your hot-patch command was unrecognized."); }, hotpatchhelp: ["Hot-patching the game engine allows you to update parts of Showdown without interrupting currently-running battles. Requires: ~", "Hot-patching has greater memory requirements than restarting.", "/hotpatch chat - reload chat-commands.js", "/hotpatch battles - spawn new simulator processes", "/hotpatch formats - reload the tools.js tree, rebuild and rebroad the formats list, and also spawn new simulator processes"], savelearnsets: function (target, room, user) { if (!this.can('hotpatch')) return false; fs.writeFile('data/learnsets.js', 'exports.BattleLearnsets = ' + JSON.stringify(Tools.data.Learnsets) + ";\n"); this.sendReply("learnsets.js saved."); }, disableladder: function (target, room, user) { if (!this.can('disableladder')) return false; if (LoginServer.disabled) { return this.sendReply("/disableladder - Ladder is already disabled."); } LoginServer.disabled = true; this.logModCommand("The ladder was disabled by " + user.name + "."); this.add("|raw|
Due to high server load, the ladder has been temporarily disabled
Rated games will no longer update the ladder. It will be back momentarily.
"); }, enableladder: function (target, room, user) { if (!this.can('disableladder')) return false; if (!LoginServer.disabled) { return this.sendReply("/enable - Ladder is already enabled."); } LoginServer.disabled = false; this.logModCommand("The ladder was enabled by " + user.name + "."); this.add("|raw|
The ladder is now back.
Rated games will update the ladder now.
"); }, lockdown: function (target, room, user) { if (!this.can('lockdown')) return false; Rooms.global.lockdown = true; for (var id in Rooms.rooms) { if (id === 'global') continue; var curRoom = Rooms.rooms[id]; curRoom.addRaw("
The server is restarting soon.
Please finish your battles quickly. No new battles can be started until the server resets in a few minutes.
"); if (curRoom.requestKickInactive && !curRoom.battle.ended) { curRoom.requestKickInactive(user, true); if (curRoom.modchat !== '+') { curRoom.modchat = '+'; curRoom.addRaw("
Moderated chat was set to +!
Only users of rank + and higher can talk.
"); } } } this.logEntry(user.name + " used /lockdown"); }, lockdownhelp: ["/lockdown - locks down the server, which prevents new battles from starting so that the server can eventually be restarted. Requires: ~"], prelockdown: function (target, room, user) { if (!this.can('lockdown')) return false; Rooms.global.lockdown = 'pre'; this.sendReply("Tournaments have been disabled in preparation for the server restart."); this.logEntry(user.name + " used /prelockdown"); }, slowlockdown: function (target, room, user) { if (!this.can('lockdown')) return false; Rooms.global.lockdown = true; for (var id in Rooms.rooms) { if (id === 'global') continue; var curRoom = Rooms.rooms[id]; if (curRoom.battle) continue; curRoom.addRaw("
The server is restarting soon.
Please finish your battles quickly. No new battles can be started until the server resets in a few minutes.
"); } this.logEntry(user.name + " used /slowlockdown"); }, endlockdown: function (target, room, user) { if (!this.can('lockdown')) return false; if (!Rooms.global.lockdown) { return this.sendReply("We're not under lockdown right now."); } if (Rooms.global.lockdown === true) { for (var id in Rooms.rooms) { if (id !== 'global') Rooms.rooms[id].addRaw("
The server shutdown was canceled.
"); } } else { this.sendReply("Preparation for the server shutdown was canceled."); } Rooms.global.lockdown = false; this.logEntry(user.name + " used /endlockdown"); }, emergency: function (target, room, user) { if (!this.can('lockdown')) return false; if (Config.emergency) { return this.sendReply("We're already in emergency mode."); } Config.emergency = true; for (var id in Rooms.rooms) { if (id !== 'global') Rooms.rooms[id].addRaw("
The server has entered emergency mode. Some features might be disabled or limited.
"); } this.logEntry(user.name + " used /emergency"); }, endemergency: function (target, room, user) { if (!this.can('lockdown')) return false; if (!Config.emergency) { return this.sendReply("We're not in emergency mode."); } Config.emergency = false; for (var id in Rooms.rooms) { if (id !== 'global') Rooms.rooms[id].addRaw("
The server is no longer in emergency mode.
"); } this.logEntry(user.name + " used /endemergency"); }, kill: function (target, room, user) { if (!this.can('lockdown')) return false; if (Rooms.global.lockdown !== true) { return this.sendReply("For safety reasons, /kill can only be used during lockdown."); } if (CommandParser.updateServerLock) { return this.sendReply("Wait for /updateserver to finish before using /kill."); } for (var i in Sockets.workers) { Sockets.workers[i].kill(); } if (!room.destroyLog) { process.exit(); return; } room.destroyLog(function () { room.logEntry(user.name + " used /kill"); }, function () { process.exit(); }); // Just in the case the above never terminates, kill the process // after 10 seconds. setTimeout(function () { process.exit(); }, 10000); }, killhelp: ["/kill - kills the server. Can't be done unless the server is in lockdown state. Requires: ~"], loadbanlist: function (target, room, user, connection) { if (!this.can('hotpatch')) return false; connection.sendTo(room, "Loading ipbans.txt..."); fs.readFile('config/ipbans.txt', function (err, data) { if (err) return; data = ('' + data).split('\n'); var rangebans = []; for (var i = 0; i < data.length; ++i) { var line = data[i].split('#')[0].trim(); if (!line) continue; if (line.includes('/')) { rangebans.push(line); } else if (line && !Users.bannedIps[line]) { Users.bannedIps[line] = '#ipban'; } } Users.checkRangeBanned = Cidr.checker(rangebans); connection.sendTo(room, "ipbans.txt has been reloaded."); }); }, loadbanlisthelp: ["/loadbanlist - Loads the bans located at ipbans.txt. The command is executed automatically at startup. Requires: ~"], refreshpage: function (target, room, user) { if (!this.can('hotpatch')) return false; Rooms.global.send('|refresh|'); this.logEntry(user.name + " used /refreshpage"); }, updateserver: function (target, room, user, connection) { if (!user.hasConsoleAccess(connection)) { return this.sendReply("/updateserver - Access denied."); } if (CommandParser.updateServerLock) { return this.sendReply("/updateserver - Another update is already in progress."); } CommandParser.updateServerLock = true; var logQueue = []; logQueue.push(user.name + " used /updateserver"); connection.sendTo(room, "updating..."); var exec = require('child_process').exec; exec('git diff-index --quiet HEAD --', function (error) { var cmd = 'git pull --rebase'; if (error) { if (error.code === 1) { // The working directory or index have local changes. cmd = 'git stash && ' + cmd + ' && git stash pop'; } else { // The most likely case here is that the user does not have // `git` on the PATH (which would be error.code === 127). connection.sendTo(room, "" + error); logQueue.push("" + error); logQueue.forEach(function (line) { room.logEntry(line); }); CommandParser.updateServerLock = false; return; } } var entry = "Running `" + cmd + "`"; connection.sendTo(room, entry); logQueue.push(entry); exec(cmd, function (error, stdout, stderr) { ("" + stdout + stderr).split("\n").forEach(function (s) { connection.sendTo(room, s); logQueue.push(s); }); logQueue.forEach(function (line) { room.logEntry(line); }); CommandParser.updateServerLock = false; }); }); }, crashfixed: function (target, room, user) { if (Rooms.global.lockdown !== true) { return this.sendReply('/crashfixed - There is no active crash.'); } if (!this.can('hotpatch')) return false; Rooms.global.lockdown = false; if (Rooms.lobby) { Rooms.lobby.modchat = false; Rooms.lobby.addRaw("
We fixed the crash without restarting the server!
You may resume talking in the lobby and starting new battles.
"); } this.logEntry(user.name + " used /crashfixed"); }, 'memusage': 'memoryusage', memoryusage: function (target) { if (!this.can('hotpatch')) return false; target = toId(target) || 'all'; if (target === 'all') { this.sendReply("Loading memory usage, this might take a while."); } var roomSize, configSize, rmSize, cpSize, simSize, usersSize, toolsSize; if (target === 'all' || target === 'rooms' || target === 'room') { this.sendReply("Calculating Room size..."); roomSize = ResourceMonitor.sizeOfObject(Rooms); this.sendReply("Rooms are using " + roomSize + " bytes of memory."); } if (target === 'all' || target === 'config') { this.sendReply("Calculating config size..."); configSize = ResourceMonitor.sizeOfObject(Config); this.sendReply("Config is using " + configSize + " bytes of memory."); } if (target === 'all' || target === 'resourcemonitor' || target === 'rm') { this.sendReply("Calculating Resource Monitor size..."); rmSize = ResourceMonitor.sizeOfObject(ResourceMonitor); this.sendReply("The Resource Monitor is using " + rmSize + " bytes of memory."); } if (target === 'all' || target === 'cmdp' || target === 'cp' || target === 'commandparser') { this.sendReply("Calculating Command Parser size..."); cpSize = ResourceMonitor.sizeOfObject(CommandParser); this.sendReply("Command Parser is using " + cpSize + " bytes of memory."); } if (target === 'all' || target === 'sim' || target === 'simulator') { this.sendReply("Calculating Simulator size..."); simSize = ResourceMonitor.sizeOfObject(Simulator); this.sendReply("Simulator is using " + simSize + " bytes of memory."); } if (target === 'all' || target === 'users') { this.sendReply("Calculating Users size..."); usersSize = ResourceMonitor.sizeOfObject(Users); this.sendReply("Users is using " + usersSize + " bytes of memory."); } if (target === 'all' || target === 'tools') { this.sendReply("Calculating Tools size..."); toolsSize = ResourceMonitor.sizeOfObject(Tools); this.sendReply("Tools are using " + toolsSize + " bytes of memory."); } if (target === 'all' || target === 'v8') { this.sendReply("Retrieving V8 memory usage..."); var o = process.memoryUsage(); this.sendReply("Resident set size: " + o.rss + ", " + o.heapUsed + " heap used of " + o.heapTotal + " total heap. " + (o.heapTotal - o.heapUsed) + " heap left."); } if (target === 'all') { this.sendReply("Calculating Total size..."); var total = (roomSize + configSize + rmSize + cpSize + simSize + usersSize + toolsSize) || 0; var units = ["bytes", "K", "M", "G"]; var converted = total; var unit = 0; while (converted > 1024) { converted /= 1024; ++unit; } converted = Math.round(converted); this.sendReply("Total memory used: " + converted + units[unit] + " (" + total + " bytes)."); } return; }, bash: function (target, room, user, connection) { if (!user.hasConsoleAccess(connection)) { return this.sendReply("/bash - Access denied."); } var exec = require('child_process').exec; exec(target, function (error, stdout, stderr) { connection.sendTo(room, ("" + stdout + stderr)); }); }, eval: function (target, room, user, connection) { if (!user.hasConsoleAccess(connection)) { return this.sendReply("/eval - Access denied."); } if (!this.canBroadcast()) return; if (!this.broadcasting) this.sendReply('||>> ' + target); try { var battle = room.battle; var me = user; this.sendReply('||<< ' + eval(target)); } catch (e) { this.sendReply('||<< error: ' + e.message); var stack = '||' + ('' + e.stack).replace(/\n/g, '\n||'); connection.sendTo(room, stack); } }, evalbattle: function (target, room, user, connection) { if (!user.hasConsoleAccess(connection)) { return this.sendReply("/evalbattle - Access denied."); } if (!this.canBroadcast()) return; if (!room.battle) { return this.sendReply("/evalbattle - This isn't a battle room."); } room.battle.send('eval', target.replace(/\n/g, '\f')); }, /********************************************************* * Battle commands *********************************************************/ forfeit: function (target, room, user) { if (!room.battle) { return this.sendReply("There's nothing to forfeit here."); } if (!room.forfeit(user)) { return this.sendReply("You can't forfeit this battle."); } }, savereplay: function (target, room, user, connection) { if (!room || !room.battle) return; var logidx = 2; // spectator log (no exact HP) if (room.battle.ended) { // If the battle is finished when /savereplay is used, include // exact HP in the replay log. logidx = 3; } var data = room.getLog(logidx).join("\n"); var datahash = crypto.createHash('md5').update(data.replace(/[^(\x20-\x7F)]+/g, '')).digest('hex'); LoginServer.request('prepreplay', { id: room.id.substr(7), loghash: datahash, p1: room.p1.name, p2: room.p2.name, format: room.format }, function (success) { if (success && success.errorip) { connection.popup("This server's request IP " + success.errorip + " is not a registered server."); return; } connection.send('|queryresponse|savereplay|' + JSON.stringify({ log: data, id: room.id.substr(7) })); }); }, mv: 'move', attack: 'move', move: function (target, room, user) { if (!room.decision) return this.sendReply("You can only do this in battle rooms."); room.decision(user, 'choose', 'move ' + target); }, sw: 'switch', switch: function (target, room, user) { if (!room.decision) return this.sendReply("You can only do this in battle rooms."); room.decision(user, 'choose', 'switch ' + parseInt(target, 10)); }, choose: function (target, room, user) { if (!room.decision) return this.sendReply("You can only do this in battle rooms."); room.decision(user, 'choose', target); }, undo: function (target, room, user) { if (!room.decision) return this.sendReply("You can only do this in battle rooms."); room.decision(user, 'undo', target); }, team: function (target, room, user) { if (!room.decision) return this.sendReply("You can only do this in battle rooms."); room.decision(user, 'choose', 'team ' + target); }, addplayer: function (target, room, user) { if (!target) return this.parse('/help addplayer'); target = this.splitTarget(target, true); var userid = toId(this.targetUsername); var targetUser = this.targetUser; var name = this.targetUsername; if (!targetUser) return this.sendReply("User " + name + " not found."); if (!room.joinBattle) return this.sendReply("You can only do this in battle rooms."); if (targetUser.can('joinbattle', null, room)) { return this.sendReply("" + name + " can already join battles as a Player."); } if (!this.can('joinbattle', null, room)) return; room.auth[targetUser.userid] = '\u2605'; this.addModCommand("" + name + " was promoted to Player by " + user.name + "."); }, addplayerhelp: ["/addplayer [username] - Allow the specified user to join the battle as a player."], joinbattle: function (target, room, user) { if (!room.joinBattle) return this.sendReply("You can only do this in battle rooms."); if (!user.can('joinbattle', null, room)) return this.popupReply("You must be a set as a player to join a battle you didn't start. Ask a player to use /addplayer on you to join this battle."); room.joinBattle(user); }, partbattle: 'leavebattle', leavebattle: function (target, room, user) { if (!room.leaveBattle) return this.sendReply("You can only do this in battle rooms."); room.leaveBattle(user); }, kickbattle: function (target, room, user) { if (!room.leaveBattle) return this.sendReply("You can only do this in battle rooms."); target = this.splitTarget(target); var targetUser = this.targetUser; if (!targetUser || !targetUser.connected) { return this.sendReply("User " + this.targetUsername + " not found."); } if (!this.can('kick', targetUser)) return false; if (room.leaveBattle(targetUser)) { this.addModCommand("" + targetUser.name + " was kicked from a battle by " + user.name + (target ? " (" + target + ")" : "")); } else { this.sendReply("/kickbattle - User isn't in battle."); } }, kickbattlehelp: ["/kickbattle [username], [reason] - Kicks a user from a battle with reason. Requires: % @ & ~"], kickinactive: function (target, room, user) { if (room.requestKickInactive) { room.requestKickInactive(user); } else { this.sendReply("You can only kick inactive players from inside a room."); } }, timer: function (target, room, user) { target = toId(target); if (room.requestKickInactive) { if (target === 'off' || target === 'false' || target === 'stop') { room.stopKickInactive(user, user.can('timer')); } else if (target === 'on' || target === 'true' || !target) { room.requestKickInactive(user, user.can('timer')); } else { this.sendReply("'" + target + "' is not a recognized timer state."); } } else { this.sendReply("You can only set the timer from inside a room."); } }, autotimer: 'forcetimer', forcetimer: function (target, room, user) { target = toId(target); if (!this.can('autotimer')) return; if (target === 'off' || target === 'false' || target === 'stop') { Config.forcetimer = false; this.addModCommand("Forcetimer is now OFF: The timer is now opt-in. (set by " + user.name + ")"); } else if (target === 'on' || target === 'true' || !target) { Config.forcetimer = true; this.addModCommand("Forcetimer is now ON: All battles will be timed. (set by " + user.name + ")"); } else { this.sendReply("'" + target + "' is not a recognized forcetimer setting."); } }, forcetie: 'forcewin', forcewin: function (target, room, user) { if (!this.can('forcewin')) return false; if (!room.battle) { this.sendReply("/forcewin - This is not a battle room."); return false; } room.battle.endType = 'forced'; if (!target) { room.battle.tie(); this.logModCommand(user.name + " forced a tie."); return false; } target = Users.get(target); if (target) target = target.userid; else target = ''; if (target) { room.battle.win(target); this.logModCommand(user.name + " forced a win for " + target + "."); } }, forcewinhelp: ["/forcetie - Forces the current match to tie. Requires: & ~"], /********************************************************* * Challenging and searching commands *********************************************************/ cancelsearch: 'search', search: function (target, room, user) { if (target) { if (Config.pmmodchat) { var userGroup = user.group; if (Config.groupsranking.indexOf(userGroup) < Config.groupsranking.indexOf(Config.pmmodchat)) { var groupName = Config.groups[Config.pmmodchat].name || Config.pmmodchat; this.popupReply("Because moderated chat is set, you must be of rank " + groupName + " or higher to search for a battle."); return false; } } Rooms.global.searchBattle(user, target); } else { Rooms.global.cancelSearch(user); } }, chall: 'challenge', challenge: function (target, room, user, connection) { target = this.splitTarget(target); var targetUser = this.targetUser; if (!targetUser || !targetUser.connected) { return this.popupReply("The user '" + this.targetUsername + "' was not found."); } if (targetUser.blockChallenges && !user.can('bypassblocks', targetUser)) { return this.popupReply("The user '" + this.targetUsername + "' is not accepting challenges right now."); } if (Config.pmmodchat) { var userGroup = user.group; if (Config.groupsranking.indexOf(userGroup) < Config.groupsranking.indexOf(Config.pmmodchat)) { var groupName = Config.groups[Config.pmmodchat].name || Config.pmmodchat; this.popupReply("Because moderated chat is set, you must be of rank " + groupName + " or higher to challenge users."); return false; } } user.prepBattle(target, 'challenge', connection, function (result) { if (result) user.makeChallenge(targetUser, target); }); }, bch: 'blockchallenges', blockchall: 'blockchallenges', blockchalls: 'blockchallenges', blockchallenges: function (target, room, user) { if (user.blockChallenges) return this.sendReply("You are already blocking challenges!"); user.blockChallenges = true; this.sendReply("You are now blocking all incoming challenge requests."); }, blockchallengeshelp: ["/blockchallenges - Blocks challenges so no one can challenge you. Unblock them with /unblockchallenges."], unbch: 'allowchallenges', unblockchall: 'allowchallenges', unblockchalls: 'allowchallenges', unblockchallenges: 'allowchallenges', allowchallenges: function (target, room, user) { if (!user.blockChallenges) return this.sendReply("You are already available for challenges!"); user.blockChallenges = false; this.sendReply("You are available for challenges from now on."); }, allowchallengeshelp: ["/unblockchallenges - Unblocks challenges so you can be challenged again. Block them with /blockchallenges."], cchall: 'cancelChallenge', cancelchallenge: function (target, room, user) { user.cancelChallengeTo(target); }, accept: function (target, room, user, connection) { var userid = toId(target); var format = ''; if (user.challengesFrom[userid]) format = user.challengesFrom[userid].format; if (!format) { this.popupReply(target + " cancelled their challenge before you could accept it."); return false; } user.prepBattle(format, 'challenge', connection, function (result) { if (result) user.acceptChallengeFrom(userid); }); }, reject: function (target, room, user) { user.rejectChallengeFrom(toId(target)); }, saveteam: 'useteam', utm: 'useteam', useteam: function (target, room, user) { user.team = target; }, /********************************************************* * Low-level *********************************************************/ cmd: 'query', query: function (target, room, user, connection) { // Avoid guest users to use the cmd errors to ease the app-layer attacks in emergency mode var trustable = (!Config.emergency || (user.named && user.registered)); if (Config.emergency && ResourceMonitor.countCmd(connection.ip, user.name)) return false; var spaceIndex = target.indexOf(' '); var cmd = target; if (spaceIndex > 0) { cmd = target.substr(0, spaceIndex); target = target.substr(spaceIndex + 1); } else { target = ''; } if (cmd === 'userdetails') { var targetUser = Users.get(target); if (!trustable || !targetUser) { connection.send('|queryresponse|userdetails|' + JSON.stringify({ userid: toId(target), rooms: false })); return false; } var roomList = {}; for (var i in targetUser.roomCount) { if (i === 'global') continue; var targetRoom = Rooms.get(i); if (!targetRoom || targetRoom.isPrivate) continue; var roomData = {}; if (targetRoom.battle) { var battle = targetRoom.battle; roomData.p1 = battle.p1 ? ' ' + battle.p1 : ''; roomData.p2 = battle.p2 ? ' ' + battle.p2 : ''; } roomList[i] = roomData; } if (!targetUser.roomCount['global']) roomList = false; var userdetails = { userid: targetUser.userid, avatar: targetUser.avatar, rooms: roomList }; connection.send('|queryresponse|userdetails|' + JSON.stringify(userdetails)); } else if (cmd === 'roomlist') { if (!trustable) return false; connection.send('|queryresponse|roomlist|' + JSON.stringify({ rooms: Rooms.global.getRoomList(target) })); } else if (cmd === 'rooms') { if (!trustable) return false; connection.send('|queryresponse|rooms|' + JSON.stringify( Rooms.global.getRooms(user) )); } }, trn: function (target, room, user, connection) { var commaIndex = target.indexOf(','); var targetName = target; var targetAuth = false; var targetToken = ''; if (commaIndex >= 0) { targetName = target.substr(0, commaIndex); target = target.substr(commaIndex + 1); commaIndex = target.indexOf(','); targetAuth = target; if (commaIndex >= 0) { targetAuth = !!parseInt(target.substr(0, commaIndex), 10); targetToken = target.substr(commaIndex + 1); } } user.rename(targetName, targetToken, targetAuth, connection); }, a: function (target, room, user) { if (!this.can('rawpacket')) return false; // secret sysop command room.add(target); }, /********************************************************* * Help commands *********************************************************/ commands: 'help', h: 'help', '?': 'help', help: function (target, room, user) { target = target.toLowerCase(); // overall if (target === 'help' || target === 'h' || target === '?' || target === 'commands') { this.sendReply("/help OR /h OR /? - Gives you help."); } else if (!target) { this.sendReply("COMMANDS: /nick, /avatar, /rating, /whois, /msg, /reply, /ignore, /away, /back, /timestamps, /highlight"); this.sendReply("INFORMATIONAL COMMANDS: /data, /dexsearch, /movesearch, /groups, /faq, /rules, /intro, /tiers, /othermetas, /learn, /analysis, /calc (replace / with ! to broadcast. Broadcasting requires: + % @ & ~)"); if (user.group !== Config.groupsranking[0]) { this.sendReply("DRIVER COMMANDS: /warn, /mute, /unmute, /alts, /forcerename, /modlog, /lock, /unlock, /announce, /redirect"); this.sendReply("MODERATOR COMMANDS: /ban, /unban, /ip"); this.sendReply("LEADER COMMANDS: /declare, /forcetie, /forcewin, /promote, /demote, /banip, /unbanall"); } this.sendReply("For an overview of room commands, use /roomhelp"); this.sendReply("For details of a specific command, use something like: /help data"); } else { var altCommandHelp; var helpCmd; var targets = target.split(' '); if (typeof commands[target] === 'string') { // If a function changes with command name, help for that command name will be searched first. altCommandHelp = target + 'help'; if (altCommandHelp in commands) { helpCmd = altCommandHelp; } else { helpCmd = commands[target] + 'help'; } } else if (targets.length > 1 && typeof commands[targets[0]] === 'object') { // Handle internal namespace commands var helpCmd = targets[targets.length - 1] + 'help'; var namespace = commands[targets[0]]; for (var i = 1; i < targets.length - 1; i++) { if (!namespace[targets[i]]) return this.sendReply("Help for the command '" + target + "' was not found. Try /help for general help"); namespace = namespace[targets[i]]; } if (typeof namespace[helpCmd] === 'object') { return this.sendReply(namespace[helpCmd].join('\n')); } else if (typeof namespace[helpCmd] === 'function') { return this.parse('/' + targets.slice(0, targets.length - 1).concat(helpCmd).join(' ')); } else { return this.sendReply("Help for the command '" + target + "' was not found. Try /help for general help"); } } else { helpCmd = target + 'help'; } if (helpCmd in commands) { if (typeof commands[helpCmd] === 'function') { // If the help command is a function, parse it instead this.parse('/' + helpCmd); } else if (Array.isArray(commands[helpCmd])) { this.sendReply(commands[helpCmd].join('\n')); } } else { this.sendReply("Help for the command '" + target + "' was not found. Try /help for general help"); } } } };