2019-02-28 11:29:26 +03:00
|
|
|
import { Injectable, Inject } from '@nestjs/common'
|
|
|
|
import { ModuleRef } from '@nestjs/core'
|
|
|
|
import Telegraf, { ContextMessageUpdate } from 'telegraf'
|
|
|
|
import { flatten, head } from 'lodash'
|
|
|
|
|
|
|
|
import { ContextTransformer } from './ContextTransformer'
|
|
|
|
import { TelegramCatch } from './decorators/TelegramCatch'
|
|
|
|
import { TelegramErrorHandler } from './interfaces/TelegramErrorHandler'
|
|
|
|
import { Handler } from './Handler'
|
|
|
|
import { Bot } from './Bot'
|
|
|
|
import { TelegramActionHandler } from './decorators/TelegramActionHandler'
|
|
|
|
import { TokenInjectionToken } from './TokenInjectionToken'
|
2019-02-28 11:38:07 +03:00
|
|
|
import { TelegramModuleOptionsFactory } from './TelegramModuleOptionsFactory'
|
2019-03-03 16:15:16 +03:00
|
|
|
import { InvalidConfigurationException } from './InvalidConfigurationException'
|
2019-02-28 11:29:26 +03:00
|
|
|
|
|
|
|
@Injectable()
|
|
|
|
export class TelegramBot {
|
2019-03-03 15:43:30 +03:00
|
|
|
private readonly sitePublicUrl?: string
|
2019-03-03 19:37:18 +03:00
|
|
|
private readonly bot: Bot
|
2019-02-28 11:29:26 +03:00
|
|
|
private ref: ModuleRef
|
|
|
|
|
|
|
|
public constructor(
|
|
|
|
@Inject(TokenInjectionToken) factory: TelegramModuleOptionsFactory,
|
|
|
|
) {
|
2019-03-03 15:43:30 +03:00
|
|
|
const { token, sitePublicUrl } = factory.createOptions()
|
2019-03-03 19:37:18 +03:00
|
|
|
|
2019-03-03 15:43:30 +03:00
|
|
|
this.sitePublicUrl = sitePublicUrl
|
2019-03-03 19:37:18 +03:00
|
|
|
this.bot = new Telegraf(token)
|
2019-02-28 11:29:26 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
public init(ref: ModuleRef) {
|
|
|
|
this.ref = ref
|
|
|
|
|
|
|
|
const handlers = this.createHandlers()
|
|
|
|
|
2019-03-03 19:37:18 +03:00
|
|
|
this.setupOnStart(handlers)
|
|
|
|
this.setupOnCommand(handlers)
|
2019-02-28 11:29:26 +03:00
|
|
|
}
|
|
|
|
|
2019-03-03 15:43:30 +03:00
|
|
|
public getMiddleware(path: string) {
|
|
|
|
if (!this.sitePublicUrl) {
|
|
|
|
throw new InvalidConfigurationException(
|
|
|
|
'sitePublicUrl',
|
|
|
|
'does not exist, but webook used',
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
this.bot.telegram
|
|
|
|
.setWebhook(`${this.sitePublicUrl}/${path}`)
|
|
|
|
.then(() => console.log('Webhook set success'))
|
|
|
|
|
|
|
|
return this.bot.webhookCallback(path)
|
|
|
|
}
|
|
|
|
|
|
|
|
public startPolling() {
|
2019-02-28 11:29:26 +03:00
|
|
|
this.bot.startPolling()
|
|
|
|
}
|
|
|
|
|
|
|
|
private createHandlers(): Handler[] {
|
|
|
|
return flatten(
|
|
|
|
Array.from((TelegramActionHandler.handlers || new Map()).entries()).map(
|
|
|
|
([handlerClass, classConfig]) => {
|
|
|
|
const handlerInstance = this.ref.get(handlerClass, { strict: false })
|
|
|
|
|
|
|
|
return Array.from(classConfig.entries()).map(
|
|
|
|
([methodName, methodCondig]) => ({
|
|
|
|
handle: handlerInstance[methodName].bind(handlerInstance),
|
|
|
|
config: methodCondig,
|
|
|
|
}),
|
|
|
|
)
|
|
|
|
},
|
|
|
|
),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2019-03-03 19:37:18 +03:00
|
|
|
private setupOnStart(handlers: Handler[]): void {
|
2019-02-28 11:29:26 +03:00
|
|
|
const onStart = handlers.filter(({ config }) => config.onStart)
|
|
|
|
|
|
|
|
if (onStart.length !== 1) {
|
|
|
|
throw new Error()
|
|
|
|
}
|
|
|
|
|
2019-03-03 19:37:18 +03:00
|
|
|
this.bot.start(this.adoptHandle(head(onStart)))
|
2019-02-28 11:29:26 +03:00
|
|
|
}
|
|
|
|
|
2019-03-03 19:37:18 +03:00
|
|
|
private setupOnCommand(handlers: Handler[]): void {
|
2019-02-28 11:29:26 +03:00
|
|
|
const commandHandlers = handlers.filter(({ config }) => config.command)
|
|
|
|
|
|
|
|
commandHandlers.forEach(handler => {
|
2019-03-03 19:37:18 +03:00
|
|
|
this.bot.command(handler.config.command, this.adoptHandle(handler))
|
2019-02-28 11:29:26 +03:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
private adoptHandle({ handle, config }: Handler) {
|
|
|
|
const errorHandler = this.createCatch()
|
|
|
|
|
|
|
|
return async (ctx: ContextMessageUpdate) => {
|
|
|
|
const args = await Promise.all(
|
|
|
|
(config.transformations || [])
|
|
|
|
.sort((a, b) => a.index - b.index)
|
|
|
|
.map(({ transform }) =>
|
|
|
|
this.ref
|
|
|
|
.get<ContextTransformer>(transform, { strict: false })
|
|
|
|
.transform(ctx),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
return handle(ctx, ...args).catch(errorHandler(ctx))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private createCatch() {
|
|
|
|
const handlers = Array.from(
|
|
|
|
(TelegramCatch.handlers || new Map()).entries(),
|
|
|
|
).map(([errorType, handlerType]) => {
|
|
|
|
const handler = this.ref.get<TelegramErrorHandler>(handlerType, {
|
|
|
|
strict: false,
|
|
|
|
})
|
|
|
|
|
|
|
|
return {
|
|
|
|
errorType,
|
|
|
|
handler,
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
return (ctx: ContextMessageUpdate) => (e: any) => {
|
|
|
|
for (const { errorType, handler } of handlers) {
|
|
|
|
if (e instanceof (errorType as any)) {
|
|
|
|
return handler.catch(ctx, e)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
throw e
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|