mizzkey/src/api/bot/core.ts

89 lines
1.9 KiB
TypeScript
Raw Normal View History

2017-10-06 20:36:46 +02:00
import * as EventEmitter from 'events';
import * as bcrypt from 'bcryptjs';
import User, { IUser } from '../models/user';
export default class BotCore extends EventEmitter {
2017-10-06 21:30:57 +02:00
public user: IUser = null;
2017-10-06 20:36:46 +02:00
private context: Context = null;
2017-10-06 21:30:57 +02:00
constructor(user?: IUser) {
2017-10-06 20:36:46 +02:00
super();
this.user = user;
}
public async q(query: string): Promise<string> {
if (this.context != null) {
return await this.context.q(query);
}
switch (query) {
case 'ping':
return 'PONG';
case 'ログイン':
case 'サインイン':
this.context = new SigninContext(this);
return await this.context.greet();
default:
return '?';
}
}
public setUser(user: IUser) {
this.user = user;
this.emit('set-user', user);
}
}
abstract class Context {
protected core: BotCore;
public abstract async greet(): Promise<string>;
public abstract async q(query: string): Promise<string>;
constructor(core: BotCore) {
this.core = core;
}
}
class SigninContext extends Context {
private temporaryUser: IUser;
public async greet(): Promise<string> {
return 'まずユーザー名を教えてください:';
}
public async q(query: string): Promise<string> {
if (this.temporaryUser == null) {
// Fetch user
const user: IUser = await User.findOne({
username_lower: query.toLowerCase()
}, {
fields: {
data: false,
profile: false
}
});
if (user === null) {
return `${query}というユーザーは存在しませんでした... もう一度教えてください:`;
} else {
this.temporaryUser = user;
return `パスワードを教えてください:`;
}
} else {
// Compare password
const same = bcrypt.compareSync(query, this.temporaryUser.password);
if (same) {
this.core.setUser(this.temporaryUser);
return `${this.temporaryUser.name}さん、おかえりなさい!`;
} else {
return `パスワードが違います... もう一度教えてください:`;
}
}
}
}