Files
discord.js/src/Client/Resolver/Resolver.js
2016-03-12 21:11:21 -08:00

242 lines
6.3 KiB
JavaScript

"use strict";
/* global Buffer */
/**
* Resolves supplied data type to a Channel. If a String, it should be a Channel ID.
* @typedef {(Channel|Server|Message|User|String)} ChannelResolvable
*/
/**
* Resolves supplied data type to a TextChannel or PMChannel. If a String, it should be a Channel ID.
* @typedef {(TextChannel|PMChannel|Server|Message|User|String)} TextChannelResolvable
*/
/**
* If given an array, turns it into a newline-separated string.
* @typedef {(String|Array)} StringResolvable
*/
/**
* Resolves supplied data type to a Message. If a channel, it is the latest message from that channel.
* @typedef {(Message|TextChannel|PMChannel)} MessageResolvable
*/
/**
* Resolves supplied data type to a Server. If a String, it should be the server's ID.
* @typedef {(Server|ServerChannel|Message|String)} ServerResolvable
*/
/**
* Resolves supplied data type to something that can be attached to a message. If a String, it can be an URL or a path to a local file.
* @typedef {(String|ReadableStream|Buffer)} FileResolvable
*/
/**
* Resolves supplied data type to an invite ID. If a String, it should be an ID or a direct URL to the invite.
* @typedef {(Invite|String)} InviteIDResolvable
*/
import fs from "fs";
import request from "superagent";
import User from "../../Structures/User";
import Channel from "../../Structures/Channel";
import TextChannel from "../../Structures/TextChannel";
import VoiceChannel from "../../Structures/VoiceChannel";
import ServerChannel from "../../Structures/ServerChannel";
import PMChannel from "../../Structures/PMChannel";
import Role from "../../Structures/Role";
import Server from "../../Structures/Server";
import Message from "../../Structures/Message";
import Invite from "../../Structures/Invite";
export default class Resolver {
constructor(internal) {
this.internal = internal;
}
resolveToBase64(resource) {
if (resource instanceof Buffer) {
resource = resource.toString("base64");
resource = "data:image/jpg;base64," + resource;
}
return resource;
}
resolveInviteID(resource) {
if (resource instanceof Invite) {
return resource.id;
}
if (typeof resource === "string" || resource instanceof String) {
if (resource.indexOf("http") === 0) {
var split = resource.split("/");
return split.pop();
}
return resource;
}
return null;
}
resolveServer(resource) {
if (resource instanceof Server) {
return resource;
}
if (resource instanceof ServerChannel) {
return resource.server;
}
if (resource instanceof String || typeof resource === "string") {
return this.internal.servers.get("id", resource);
}
if (resource instanceof Message) {
if (resource.channel instanceof TextChannel) {
return resource.channel.server;
}
}
return null;
}
resolveRole(resource) {
if (resource instanceof Role) {
return resource;
}
if (resource instanceof String || typeof resource === "string") {
var role = null;
for (var server of this.internal.servers) {
if (role = server.roles.find(r => r.id == resource)) {
return role;
}
}
}
return null;
}
resolveFile(resource) {
if (typeof resource === "string" || resource instanceof String) {
if (/^https?:\/\//.test(resource)) {
return new Promise((resolve, reject) => {
request.get(resource).buffer().parse((res, cb) => {
res.setEncoding("binary");
res.data = "";
res.on("data", (chunk) => {
res.data += chunk;
});
res.on("end", () => {
cb(null, new Buffer(res.data, "binary"));
});
}).end((err, res) => {
if (err) {
return reject(err);
}
return resolve(res.body);
});
});
} else {
return Promise.resolve(resource);
}
}
return Promise.resolve(resource);
}
resolveMentions(resource) {
// resource is a string
var _mentions = [];
var changed = resource;
for (var mention of (resource.match(/<@[0-9]+>/g) || [])) {
var userID = mention.substring(2, mention.length - 1);
var user = this.internal.client.users.get("id", userID);
if (user) {
_mentions.push(user);
changed = changed.replace(new RegExp(mention, "g"), `@${user.username}`);
}
}
return [_mentions, changed];
}
resolveString(resource) {
// accepts Array, Channel, Server, User, Message, String and anything
// toString()-able
var final = resource;
if (resource instanceof Array) {
final = resource.join("\n");
}
return final.toString();
}
resolveUser(resource) {
/*
accepts a Message, Channel, Server, String ID, User, PMChannel
*/
if (resource instanceof User) {
return resource;
}
if (resource instanceof Message) {
return resource.author;
}
if (resource instanceof TextChannel) {
var lmsg = resource.lastMessage;
if (lmsg) {
return lmsg.author;
}
}
if (resource instanceof Server) {
return resource.owner;
}
if (resource instanceof PMChannel) {
return resource.recipient;
}
if (resource instanceof String || typeof resource === "string") {
return this.internal.users.get("id", resource);
}
return null;
}
resolveMessage(resource) {
// accepts a Message, PMChannel & TextChannel
if (resource instanceof TextChannel || resource instanceof PMChannel) {
return resource.lastMessage;
}
if (resource instanceof Message) {
return resource;
}
return null;
}
resolveChannel(resource) {
/*
accepts a Message, Channel, Server, String ID, User
*/
if (resource instanceof Message) {
return Promise.resolve(resource.channel);
}
if (resource instanceof Channel) {
return Promise.resolve(resource);
}
if (resource instanceof Server) {
return Promise.resolve(resource.defaultChannel);
}
if (resource instanceof String || typeof resource === "string") {
var user = this.internal.users.get("id", resource);
if (user) {
resource = user;
} else {
return Promise.resolve(this.internal.channels.get("id", resource) || this.internal.private_channels.get("id", resource));
}
}
if (resource instanceof User) {
// see if a PM exists
for (var pmchat of this.internal.private_channels) {
if (pmchat.recipient.equals(resource)) {
return Promise.resolve(pmchat);
}
}
// PM does not exist :\
return this.internal.startPM(resource);
}
var error = new Error("Could not resolve channel");
error.resource = resource;
return Promise.reject(error);
}
}