201 lines
5.8 KiB
TypeScript
201 lines
5.8 KiB
TypeScript
import EventEmitter from "events";
|
|
import {Job} from "./Job/Job";
|
|
import Logger from "./Logger";
|
|
import {PduProcessor} from "./PDUProcessor/PduProcessor";
|
|
|
|
const NanoTimer = require("nanotimer");
|
|
const smpp = require("smpp");
|
|
|
|
export abstract class SmppSession {
|
|
readonly EVENT: any = {
|
|
STATUS_CHANGED: "STATUS_CHANGED",
|
|
STATE_CHANGED: "STATE_CHANGED",
|
|
ANY_PDU: "ANY_PDU",
|
|
MESSAGE_SEND_COUNTER_UPDATE_EVENT: "MESSAGE_SEND_COUNTER_UPDATE_EVENT",
|
|
};
|
|
abstract STATUSES: string[];
|
|
abstract pduProcessors: PduProcessor[];
|
|
readonly UPDATE_WS: string = "UPDATE_WS";
|
|
readonly eventEmitter: EventEmitter = new EventEmitter();
|
|
readonly logger: Logger = new Logger(`SmppSession`);
|
|
readonly sendTimer: any = new NanoTimer();
|
|
readonly counterUpdateTimer: any = new NanoTimer();
|
|
readonly MESSAGE_SEND_UPDATE_DELAY: number = Number(process.env.MESSAGE_SEND_UPDATE_DELAY) || 500;
|
|
|
|
protected constructor() {
|
|
this.eventEmitter.on(this.EVENT.STATE_CHANGED, () => this.updateWs(this.EVENT.STATE_CHANGED));
|
|
this.eventEmitter.on(this.EVENT.STATUS_CHANGED, () => this.updateWs(this.EVENT.STATUS_CHANGED));
|
|
this.eventEmitter.on(this.EVENT.ANY_PDU, (pdu: any) => this.updateWs(this.EVENT.ANY_PDU, [pdu]));
|
|
this.eventEmitter.on(this.EVENT.MESSAGE_SEND_COUNTER_UPDATE_EVENT, (count: number) => this.updateWs(this.EVENT.MESSAGE_SEND_COUNTER_UPDATE_EVENT, [count]));
|
|
}
|
|
|
|
abstract _username: string;
|
|
|
|
get username(): string {
|
|
return this._username;
|
|
}
|
|
|
|
set username(username: string) {
|
|
this._username = username;
|
|
this.eventEmitter.emit(this.EVENT.STATE_CHANGED, this.serialize());
|
|
}
|
|
|
|
abstract _password: string;
|
|
|
|
get password(): string {
|
|
return this._password;
|
|
}
|
|
|
|
set password(password: string) {
|
|
this._password = password;
|
|
this.eventEmitter.emit(this.EVENT.STATE_CHANGED, this.serialize());
|
|
}
|
|
|
|
abstract _id: number;
|
|
|
|
get id(): number {
|
|
return this._id;
|
|
}
|
|
|
|
abstract _status: string;
|
|
|
|
get status(): string {
|
|
return this._status;
|
|
}
|
|
|
|
set status(status: string) {
|
|
this._status = status;
|
|
this.eventEmitter.emit(this.EVENT.STATUS_CHANGED, this.status);
|
|
}
|
|
|
|
abstract _defaultSingleJob: Job;
|
|
|
|
get defaultSingleJob(): Job {
|
|
return this._defaultSingleJob;
|
|
}
|
|
|
|
set defaultSingleJob(job: Job) {
|
|
this._defaultSingleJob = job;
|
|
job.on(Job.STATE_CHANGED, this.eventJobUpdated);
|
|
this.eventEmitter.emit(this.EVENT.STATE_CHANGED, this.serialize());
|
|
}
|
|
|
|
abstract _defaultMultipleJob: Job;
|
|
|
|
get defaultMultipleJob(): Job {
|
|
return this._defaultMultipleJob;
|
|
}
|
|
|
|
set defaultMultipleJob(job: Job) {
|
|
this._defaultMultipleJob = job;
|
|
job.on(Job.STATE_CHANGED, this.eventJobUpdated);
|
|
this.eventEmitter.emit(this.EVENT.STATE_CHANGED, this.serialize());
|
|
}
|
|
|
|
setStatus(statusIndex: number) {
|
|
this._status = this.STATUSES[statusIndex];
|
|
this.eventEmitter.emit(this.EVENT.STATUS_CHANGED, this.status);
|
|
}
|
|
|
|
abstract sendPdu(pdu: object, force?: boolean): Promise<object>;
|
|
|
|
sendSingle(job: Job): Promise<object> {
|
|
return this.sendPdu(job.pdu);
|
|
}
|
|
|
|
sendSingleDefault(): Promise<object> {
|
|
return this.sendSingle(this.defaultSingleJob);
|
|
}
|
|
|
|
abstract sendMultiple(job: Job): Promise<void>;
|
|
|
|
sendMultipleDefault(): Promise<void> {
|
|
return this.sendMultiple(this.defaultMultipleJob);
|
|
}
|
|
|
|
cancelSendInterval(): void {
|
|
this.sendTimer.clearInterval();
|
|
this.counterUpdateTimer.clearInterval();
|
|
this.setStatus(this.STATUSES.length - 2);
|
|
}
|
|
|
|
abstract close(): Promise<void>;
|
|
|
|
abstract serialize(): object;
|
|
|
|
on(event: string, callback: (...args: any[]) => void): void {
|
|
this.eventEmitter.on(event, callback);
|
|
}
|
|
|
|
updateWs(event: string, args?: any[]): void {
|
|
this.logger.log1(`Update WS: ${event}`);
|
|
let message: {
|
|
type: string,
|
|
data?: string
|
|
} = {
|
|
type: event,
|
|
};
|
|
switch (event) {
|
|
case this.EVENT.STATE_CHANGED:
|
|
message.data = JSON.stringify(this.serialize());
|
|
break;
|
|
case this.EVENT.STATUS_CHANGED:
|
|
message.data = JSON.stringify(this.status);
|
|
break;
|
|
case this.EVENT.ANY_PDU:
|
|
message.data = JSON.stringify(args![0]);
|
|
break;
|
|
case this.EVENT.MESSAGE_SEND_COUNTER_UPDATE_EVENT:
|
|
message.data = JSON.stringify(args![0]);
|
|
break;
|
|
}
|
|
this.eventEmitter.emit(this.UPDATE_WS, message);
|
|
}
|
|
|
|
eventJobUpdated(): void {
|
|
this.eventEmitter.emit(this.EVENT.STATE_CHANGED, this.serialize());
|
|
}
|
|
|
|
addPduProcessor(pduProcessor: PduProcessor): void {
|
|
if (this.pduProcessors.indexOf(pduProcessor) === -1) {
|
|
this.pduProcessors.push(pduProcessor);
|
|
this.logger.log1(`Adding PDU processor: ${pduProcessor.constructor.name}-${this.id}, now active: ${this.pduProcessors.length} processors`);
|
|
this.eventEmitter.emit(this.EVENT.STATE_CHANGED, this.serialize());
|
|
} else {
|
|
this.logger.log1(`PDU processor: ${pduProcessor.constructor.name}-${this.id} already attached to session`);
|
|
}
|
|
}
|
|
|
|
removePduProcessor(pduProcessor: PduProcessor): void {
|
|
this.pduProcessors = this.pduProcessors.splice(this.pduProcessors.indexOf(pduProcessor), 1);
|
|
this.logger.log1(`Removing PDU processor: ${pduProcessor.constructor.name}-${this.id}, now active: ${this.pduProcessors.length} processors`);
|
|
this.eventEmitter.emit(this.EVENT.STATE_CHANGED, this.serialize());
|
|
}
|
|
|
|
getPduProcessors(): PduProcessor[] {
|
|
return this.pduProcessors;
|
|
}
|
|
|
|
serializePduProcessors(): object {
|
|
this.logger.log1(`Serializing ${this.pduProcessors.length} clients`)
|
|
return this.pduProcessors.map((processor: PduProcessor) => {
|
|
return processor.serialize();
|
|
});
|
|
}
|
|
|
|
eventAnyPdu(session: any, pdu: any): Promise<any> {
|
|
this.eventEmitter.emit(this.EVENT.ANY_PDU, pdu);
|
|
let successful: number = 0;
|
|
this.pduProcessors.forEach((pduProcessor: PduProcessor) => {
|
|
pduProcessor.processPdu(session, pdu).then((result: any) => {
|
|
successful++;
|
|
}, (error: any) => {
|
|
});
|
|
});
|
|
if (successful === 0) {
|
|
return Promise.resolve("No PDU processor was able to process the PDU");
|
|
} else {
|
|
return Promise.resolve();
|
|
}
|
|
}
|
|
} |