2015-09-18 20:54:20 +08:00
|
|
|
/*
|
2016-01-07 12:06:39 +08:00
|
|
|
Copyright 2015, 2016 OpenMarket Ltd
|
2015-09-18 20:54:20 +08:00
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
var MatrixClientPeg = require("./MatrixClientPeg");
|
2015-10-27 17:58:55 +08:00
|
|
|
var MatrixTools = require("./MatrixTools");
|
2015-09-18 20:54:20 +08:00
|
|
|
var dis = require("./dispatcher");
|
|
|
|
var encryption = require("./encryption");
|
2016-01-05 08:46:52 +08:00
|
|
|
var Tinter = require("./Tinter");
|
2015-09-18 20:54:20 +08:00
|
|
|
|
2016-01-14 22:39:58 +08:00
|
|
|
|
|
|
|
class Command {
|
|
|
|
constructor(name, paramArgs, runFn) {
|
|
|
|
this.name = name;
|
|
|
|
this.paramArgs = paramArgs;
|
|
|
|
this.runFn = runFn;
|
|
|
|
}
|
|
|
|
|
|
|
|
getCommand() {
|
|
|
|
return "/" + this.name;
|
|
|
|
}
|
|
|
|
|
|
|
|
getCommandWithArgs() {
|
|
|
|
return this.getCommand() + " " + this.paramArgs;
|
|
|
|
}
|
|
|
|
|
|
|
|
run(roomId, args) {
|
|
|
|
return this.runFn.bind(this)(roomId, args);
|
|
|
|
}
|
|
|
|
|
|
|
|
getUsage() {
|
|
|
|
return "Usage: " + this.getCommandWithArgs()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-18 20:54:20 +08:00
|
|
|
var reject = function(msg) {
|
|
|
|
return {
|
|
|
|
error: msg
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
var success = function(promise) {
|
|
|
|
return {
|
|
|
|
promise: promise
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
var commands = {
|
|
|
|
// Change your nickname
|
2016-01-14 22:39:58 +08:00
|
|
|
nick: new Command("nick", "<display_name>", function(room_id, args) {
|
2015-09-18 20:54:20 +08:00
|
|
|
if (args) {
|
|
|
|
return success(
|
|
|
|
MatrixClientPeg.get().setDisplayName(args)
|
|
|
|
);
|
|
|
|
}
|
2016-01-14 22:39:58 +08:00
|
|
|
return reject(this.getUsage());
|
|
|
|
}),
|
2015-09-18 20:54:20 +08:00
|
|
|
|
2016-01-08 11:22:38 +08:00
|
|
|
// Changes the colorscheme of your current room
|
2016-01-15 00:29:30 +08:00
|
|
|
tint: new Command("tint", "<color1> [<color2>]", function(room_id, args) {
|
2016-01-08 11:22:38 +08:00
|
|
|
if (args) {
|
|
|
|
var matches = args.match(/^(#([0-9a-fA-F]{3}|[0-9a-fA-F]{6}))( +(#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})))?$/);
|
|
|
|
if (matches) {
|
|
|
|
Tinter.tint(matches[1], matches[4]);
|
|
|
|
var colorScheme = {}
|
|
|
|
colorScheme.primary_color = matches[1];
|
|
|
|
if (matches[4]) {
|
|
|
|
colorScheme.secondary_color = matches[4];
|
|
|
|
}
|
|
|
|
return success(
|
|
|
|
MatrixClientPeg.get().setRoomAccountData(
|
2016-01-13 21:01:00 +08:00
|
|
|
room_id, "org.matrix.room.color_scheme", colorScheme
|
2016-01-08 11:22:38 +08:00
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2016-01-14 22:39:58 +08:00
|
|
|
return reject(this.getUsage());
|
|
|
|
}),
|
2016-01-05 08:46:52 +08:00
|
|
|
|
2016-01-15 00:29:30 +08:00
|
|
|
encrypt: new Command("encrypt", "<on|off>", function(room_id, args) {
|
2015-09-18 20:54:20 +08:00
|
|
|
if (args == "on") {
|
|
|
|
var client = MatrixClientPeg.get();
|
|
|
|
var members = client.getRoom(room_id).currentState.members;
|
|
|
|
var user_ids = Object.keys(members);
|
|
|
|
return success(
|
|
|
|
encryption.enableEncryption(client, room_id, user_ids)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (args == "off") {
|
|
|
|
var client = MatrixClientPeg.get();
|
|
|
|
return success(
|
|
|
|
encryption.disableEncryption(client, room_id)
|
|
|
|
);
|
|
|
|
|
|
|
|
}
|
2016-01-14 22:39:58 +08:00
|
|
|
return reject(this.getUsage());
|
|
|
|
}),
|
2015-09-18 20:54:20 +08:00
|
|
|
|
|
|
|
// Change the room topic
|
2016-01-14 22:39:58 +08:00
|
|
|
topic: new Command("topic", "<topic>", function(room_id, args) {
|
2015-09-18 20:54:20 +08:00
|
|
|
if (args) {
|
|
|
|
return success(
|
|
|
|
MatrixClientPeg.get().setRoomTopic(room_id, args)
|
|
|
|
);
|
|
|
|
}
|
2016-01-14 22:39:58 +08:00
|
|
|
return reject(this.getUsage());
|
|
|
|
}),
|
2015-09-18 20:54:20 +08:00
|
|
|
|
|
|
|
// Invite a user
|
2016-01-14 22:39:58 +08:00
|
|
|
invite: new Command("invite", "<userId>", function(room_id, args) {
|
2015-09-18 20:54:20 +08:00
|
|
|
if (args) {
|
|
|
|
var matches = args.match(/^(\S+)$/);
|
|
|
|
if (matches) {
|
|
|
|
return success(
|
|
|
|
MatrixClientPeg.get().invite(room_id, matches[1])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2016-01-14 22:39:58 +08:00
|
|
|
return reject(this.getUsage());
|
|
|
|
}),
|
2015-09-18 20:54:20 +08:00
|
|
|
|
|
|
|
// Join a room
|
2016-01-14 22:39:58 +08:00
|
|
|
join: new Command("join", "<room_alias>", function(room_id, args) {
|
2015-09-18 20:54:20 +08:00
|
|
|
if (args) {
|
|
|
|
var matches = args.match(/^(\S+)$/);
|
|
|
|
if (matches) {
|
|
|
|
var room_alias = matches[1];
|
|
|
|
if (room_alias[0] !== '#') {
|
|
|
|
return reject("Usage: /join #alias:domain");
|
|
|
|
}
|
|
|
|
if (!room_alias.match(/:/)) {
|
2016-01-14 02:15:59 +08:00
|
|
|
room_alias += ':' + MatrixClientPeg.get().getDomain();
|
2015-09-18 20:54:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Try to find a room with this alias
|
2015-12-28 10:58:40 +08:00
|
|
|
// XXX: do we need to do this? Doesn't the JS SDK suppress duplicate attempts to join the same room?
|
2015-10-27 17:58:55 +08:00
|
|
|
var foundRoom = MatrixTools.getRoomForAlias(
|
|
|
|
MatrixClientPeg.get().getRooms(),
|
|
|
|
room_alias
|
|
|
|
);
|
2015-12-28 10:58:40 +08:00
|
|
|
|
|
|
|
if (foundRoom) { // we've already joined this room, view it if it's not archived.
|
|
|
|
var me = foundRoom.getMember(MatrixClientPeg.get().credentials.userId);
|
|
|
|
if (me && me.membership !== "leave") {
|
|
|
|
dis.dispatch({
|
|
|
|
action: 'view_room',
|
|
|
|
room_id: foundRoom.roomId
|
|
|
|
});
|
|
|
|
return success();
|
|
|
|
}
|
2015-09-18 20:54:20 +08:00
|
|
|
}
|
2015-12-28 10:58:40 +08:00
|
|
|
|
|
|
|
// otherwise attempt to join this alias.
|
|
|
|
return success(
|
|
|
|
MatrixClientPeg.get().joinRoom(room_alias).then(
|
|
|
|
function(room) {
|
|
|
|
dis.dispatch({
|
|
|
|
action: 'view_room',
|
|
|
|
room_id: room.roomId
|
|
|
|
});
|
|
|
|
})
|
|
|
|
);
|
2015-09-18 20:54:20 +08:00
|
|
|
}
|
|
|
|
}
|
2016-01-14 22:39:58 +08:00
|
|
|
return reject(this.getUsage());
|
|
|
|
}),
|
2015-09-18 20:54:20 +08:00
|
|
|
|
2016-01-14 22:39:58 +08:00
|
|
|
part: new Command("part", "[#alias:domain]", function(room_id, args) {
|
2015-09-18 20:54:20 +08:00
|
|
|
var targetRoomId;
|
|
|
|
if (args) {
|
|
|
|
var matches = args.match(/^(\S+)$/);
|
|
|
|
if (matches) {
|
|
|
|
var room_alias = matches[1];
|
|
|
|
if (room_alias[0] !== '#') {
|
2016-01-14 22:39:58 +08:00
|
|
|
return reject(this.getUsage());
|
2015-09-18 20:54:20 +08:00
|
|
|
}
|
|
|
|
if (!room_alias.match(/:/)) {
|
2016-01-14 02:15:59 +08:00
|
|
|
room_alias += ':' + MatrixClientPeg.get().getDomain();
|
2015-09-18 20:54:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Try to find a room with this alias
|
|
|
|
var rooms = MatrixClientPeg.get().getRooms();
|
|
|
|
for (var i = 0; i < rooms.length; i++) {
|
|
|
|
var aliasEvents = rooms[i].currentState.getStateEvents(
|
|
|
|
"m.room.aliases"
|
|
|
|
);
|
|
|
|
for (var j = 0; j < aliasEvents.length; j++) {
|
|
|
|
var aliases = aliasEvents[j].getContent().aliases || [];
|
|
|
|
for (var k = 0; k < aliases.length; k++) {
|
|
|
|
if (aliases[k] === room_alias) {
|
|
|
|
targetRoomId = rooms[i].roomId;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (targetRoomId) { break; }
|
|
|
|
}
|
|
|
|
if (targetRoomId) { break; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!targetRoomId) {
|
|
|
|
return reject("Unrecognised room alias: " + room_alias);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!targetRoomId) targetRoomId = room_id;
|
|
|
|
return success(
|
|
|
|
MatrixClientPeg.get().leave(targetRoomId).then(
|
|
|
|
function() {
|
|
|
|
dis.dispatch({action: 'view_next_room'});
|
|
|
|
})
|
|
|
|
);
|
2016-01-14 22:39:58 +08:00
|
|
|
}),
|
2015-09-18 20:54:20 +08:00
|
|
|
|
|
|
|
// Kick a user from the room with an optional reason
|
2016-01-14 22:39:58 +08:00
|
|
|
kick: new Command("kick", "<userId> [<reason>]", function(room_id, args) {
|
2015-09-18 20:54:20 +08:00
|
|
|
if (args) {
|
|
|
|
var matches = args.match(/^(\S+?)( +(.*))?$/);
|
|
|
|
if (matches) {
|
|
|
|
return success(
|
|
|
|
MatrixClientPeg.get().kick(room_id, matches[1], matches[3])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2016-01-14 22:39:58 +08:00
|
|
|
return reject(this.getUsage());
|
|
|
|
}),
|
2015-09-18 20:54:20 +08:00
|
|
|
|
|
|
|
// Ban a user from the room with an optional reason
|
2016-01-14 22:39:58 +08:00
|
|
|
ban: new Command("ban", "<userId> [<reason>]", function(room_id, args) {
|
2015-09-18 20:54:20 +08:00
|
|
|
if (args) {
|
|
|
|
var matches = args.match(/^(\S+?)( +(.*))?$/);
|
|
|
|
if (matches) {
|
|
|
|
return success(
|
|
|
|
MatrixClientPeg.get().ban(room_id, matches[1], matches[3])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2016-01-14 22:39:58 +08:00
|
|
|
return reject(this.getUsage());
|
|
|
|
}),
|
2015-09-18 20:54:20 +08:00
|
|
|
|
|
|
|
// Unban a user from the room
|
2016-01-14 22:39:58 +08:00
|
|
|
unban: new Command("unban", "<userId>", function(room_id, args) {
|
2015-09-18 20:54:20 +08:00
|
|
|
if (args) {
|
|
|
|
var matches = args.match(/^(\S+)$/);
|
|
|
|
if (matches) {
|
|
|
|
// Reset the user membership to "leave" to unban him
|
|
|
|
return success(
|
|
|
|
MatrixClientPeg.get().unban(room_id, matches[1])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2016-01-14 22:39:58 +08:00
|
|
|
return reject(this.getUsage());
|
|
|
|
}),
|
2015-09-18 20:54:20 +08:00
|
|
|
|
|
|
|
// Define the power level of a user
|
2016-01-14 22:39:58 +08:00
|
|
|
op: new Command("op", "<userId> [<power level>]", function(room_id, args) {
|
2015-09-18 20:54:20 +08:00
|
|
|
if (args) {
|
|
|
|
var matches = args.match(/^(\S+?)( +(\d+))?$/);
|
|
|
|
var powerLevel = 50; // default power level for op
|
|
|
|
if (matches) {
|
|
|
|
var user_id = matches[1];
|
|
|
|
if (matches.length === 4 && undefined !== matches[3]) {
|
|
|
|
powerLevel = parseInt(matches[3]);
|
|
|
|
}
|
|
|
|
if (powerLevel !== NaN) {
|
|
|
|
var room = MatrixClientPeg.get().getRoom(room_id);
|
|
|
|
if (!room) {
|
|
|
|
return reject("Bad room ID: " + room_id);
|
|
|
|
}
|
|
|
|
var powerLevelEvent = room.currentState.getStateEvents(
|
|
|
|
"m.room.power_levels", ""
|
|
|
|
);
|
|
|
|
return success(
|
|
|
|
MatrixClientPeg.get().setPowerLevel(
|
|
|
|
room_id, user_id, powerLevel, powerLevelEvent
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-01-14 22:39:58 +08:00
|
|
|
return reject(this.getUsage());
|
|
|
|
}),
|
2015-09-18 20:54:20 +08:00
|
|
|
|
|
|
|
// Reset the power level of a user
|
2016-01-14 22:39:58 +08:00
|
|
|
deop: new Command("deop", "<userId>", function(room_id, args) {
|
2015-09-18 20:54:20 +08:00
|
|
|
if (args) {
|
|
|
|
var matches = args.match(/^(\S+)$/);
|
|
|
|
if (matches) {
|
|
|
|
var room = MatrixClientPeg.get().getRoom(room_id);
|
|
|
|
if (!room) {
|
|
|
|
return reject("Bad room ID: " + room_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
var powerLevelEvent = room.currentState.getStateEvents(
|
|
|
|
"m.room.power_levels", ""
|
|
|
|
);
|
|
|
|
return success(
|
|
|
|
MatrixClientPeg.get().setPowerLevel(
|
|
|
|
room_id, args, undefined, powerLevelEvent
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2016-01-14 22:39:58 +08:00
|
|
|
return reject(this.getUsage());
|
|
|
|
})
|
2015-09-18 20:54:20 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
// helpful aliases
|
2016-01-15 00:29:01 +08:00
|
|
|
var aliases = {
|
|
|
|
j: "join"
|
|
|
|
}
|
2015-09-18 20:54:20 +08:00
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
/**
|
|
|
|
* Process the given text for /commands and perform them.
|
|
|
|
* @param {string} roomId The room in which the command was performed.
|
|
|
|
* @param {string} input The raw text input by the user.
|
|
|
|
* @return {Object|null} An object with the property 'error' if there was an error
|
|
|
|
* processing the command, or 'promise' if a request was sent out.
|
|
|
|
* Returns null if the input didn't match a command.
|
|
|
|
*/
|
|
|
|
processInput: function(roomId, input) {
|
|
|
|
// trim any trailing whitespace, as it can confuse the parser for
|
|
|
|
// IRC-style commands
|
|
|
|
input = input.replace(/\s+$/, "");
|
|
|
|
if (input[0] === "/" && input[1] !== "/") {
|
|
|
|
var bits = input.match(/^(\S+?)( +(.*))?$/);
|
|
|
|
var cmd = bits[1].substring(1).toLowerCase();
|
|
|
|
var args = bits[3];
|
|
|
|
if (cmd === "me") return null;
|
2016-01-15 00:29:01 +08:00
|
|
|
if (aliases[cmd]) {
|
|
|
|
cmd = aliases[cmd];
|
|
|
|
}
|
2015-09-18 20:54:20 +08:00
|
|
|
if (commands[cmd]) {
|
2016-01-14 22:39:58 +08:00
|
|
|
return commands[cmd].run(roomId, args);
|
2015-09-18 20:54:20 +08:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
return reject("Unrecognised command: " + input);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null; // not a command
|
2016-01-14 01:46:36 +08:00
|
|
|
},
|
|
|
|
|
|
|
|
getCommandList: function() {
|
2016-02-14 19:45:52 +08:00
|
|
|
// Return all the commands plus /me and /markdown which aren't handled like normal commands
|
2016-01-15 01:33:52 +08:00
|
|
|
var cmds = Object.keys(commands).sort().map(function(cmdKey) {
|
2016-01-14 22:39:58 +08:00
|
|
|
return commands[cmdKey];
|
2016-01-15 01:33:52 +08:00
|
|
|
})
|
|
|
|
cmds.push(new Command("me", "<action>", function(){}));
|
2016-02-14 19:45:52 +08:00
|
|
|
cmds.push(new Command("markdown", "<on|off>", function(){}));
|
2016-01-15 01:33:52 +08:00
|
|
|
|
|
|
|
return cmds;
|
2015-09-18 20:54:20 +08:00
|
|
|
}
|
|
|
|
};
|