fchat-rising/chat/ads/ad-manager.ts

317 lines
9.2 KiB
TypeScript
Raw Permalink Normal View History

2020-04-11 14:48:09 +00:00
import throat from 'throat';
import * as _ from 'lodash';
2020-06-29 19:30:08 +00:00
import log from 'electron-log'; //tslint:disable-line:match-default-export-name
2020-03-15 16:23:39 +00:00
import core from '../core';
import { Conversation } from '../interfaces';
2019-07-06 16:49:19 +00:00
import Timer = NodeJS.Timer;
2020-04-11 14:48:09 +00:00
import ChannelConversation = Conversation.ChannelConversation;
2019-06-07 19:31:42 +00:00
const adManagerThroat = throat(1);
2020-04-11 14:48:09 +00:00
export interface RecoverableAd {
channel: string;
index: number;
nextPostDue: Date | undefined,
firstPost: Date | undefined,
expireDue: Date | undefined;
}
2019-06-07 19:31:42 +00:00
export class AdManager {
static readonly POSTING_PERIOD = 3 * 60 * 60 * 1000;
2019-06-07 21:43:22 +00:00
static readonly START_VARIANCE = 3 * 60 * 1000;
static readonly POST_VARIANCE = 8 * 60 * 1000;
static readonly POST_DELAY = 1.5 * 60 * 1000;
2019-06-07 19:31:42 +00:00
static readonly POST_MANUAL_THRESHOLD = 5 * 1000; // don't post anything within 5 seconds of other posts
2019-06-07 19:31:42 +00:00
private conversation: Conversation;
private adIndex = 0;
private active = false;
private nextPostDue?: Date;
private expireDue?: Date;
private firstPost?: Date;
2019-07-06 16:49:19 +00:00
private interval?: Timer;
2020-12-28 23:07:10 +00:00
private adMap: number[] = [];
2019-06-07 19:31:42 +00:00
constructor(conversation: Conversation) {
this.conversation = conversation;
}
isActive(): boolean {
return this.active;
}
2020-12-31 01:59:56 +00:00
skipAd(): void {
this.adIndex += 1;
}
// tslint:disable-next-line
private async delay(ms: number): Promise<void> {
return new Promise((resolve) => setTimeout(resolve, ms));
}
// This makes sure there is a 5s delay between channel posts
private async sendAdToChannel(msg: string, conv: Conversation.ChannelConversation): Promise<void> {
2020-06-29 19:30:08 +00:00
const initTime = Date.now();
await adManagerThroat(
async() => {
2020-06-29 19:30:08 +00:00
const throatTime = Date.now();
const delta = Date.now() - core.cache.getLastPost().getTime();
if ((delta > 0) && (delta < AdManager.POST_MANUAL_THRESHOLD)) {
await this.delay(delta);
}
2020-06-29 19:30:08 +00:00
const delayTime = Date.now();
log.debug(
'adManager.sendAdToChannel',
{
character: core.characters.ownCharacter?.name,
channel: conv.channel.name,
throatDelta: throatTime - initTime,
delayDelta: delayTime - throatTime,
totalWait: delayTime - initTime,
msg
}
);
2020-06-29 19:30:08 +00:00
await conv.sendAd(msg);
}
);
}
2023-09-03 01:55:06 +00:00
private determineNextAdDelayMs(chanConv: Conversation.ChannelConversation): number {
2023-12-03 00:41:17 +00:00
const match = chanConv.channel.description.toLowerCase().match(/\[\s*ads:\s*([0-9.]+)\s*(m|mins?|minutes?|h|hrs?|hours?|s|secs?|seconds?)\.?\s*]/);
2023-09-03 01:55:06 +00:00
if (!match) {
return AdManager.POST_DELAY;
}
const n = _.toNumber(match[1]);
let mul = 1000; // seconds
if (match[2].substr(0, 1) === 'h') {
mul = 60 * 60 * 1000; // hours
} else if (match[2].substr(0, 1) === 'm') {
mul = 60 * 1000; // minutes
}
2023-12-03 00:41:17 +00:00
return Math.max((n * mul) - Math.max(Date.now() - chanConv.nextAd, 0), AdManager.POST_DELAY);
2023-09-03 01:55:06 +00:00
}
2019-06-07 19:31:42 +00:00
private async sendNextPost(): Promise<void> {
const msg = this.getNextAd();
if ((!msg) || ((this.expireDue) && (this.expireDue.getTime() < Date.now()))) {
this.stop();
return;
}
const chanConv = (<Conversation.ChannelConversation>this.conversation);
await this.sendAdToChannel(msg, chanConv);
2019-06-07 19:31:42 +00:00
// post next ad every 12 - 22 minutes
const nextInMs = Math.max(0, (chanConv.nextAd - Date.now())) +
2023-09-03 01:55:06 +00:00
this.determineNextAdDelayMs(chanConv) +
2019-06-07 19:31:42 +00:00
Math.random() * AdManager.POST_VARIANCE;
this.adIndex = this.adIndex + 1;
2022-09-03 17:46:40 +00:00
this.nextPostDue = new Date(Math.max(
Date.now() + nextInMs,
2022-09-03 17:54:54 +00:00
(chanConv.settings.adSettings.lastAdTimestamp || 0) + (core.connection.vars.lfrp_flood * 1000)
2022-09-03 17:46:40 +00:00
));
2019-06-07 19:31:42 +00:00
2019-07-06 16:49:19 +00:00
// tslint:disable-next-line: no-unnecessary-type-assertion
2019-06-07 19:31:42 +00:00
this.interval = setTimeout(
async() => {
await this.sendNextPost();
},
nextInMs
2019-07-06 16:49:19 +00:00
) as Timer;
2019-06-07 19:31:42 +00:00
}
2020-12-28 23:07:10 +00:00
generateAdMap(): number[] {
const ads = this.getAds();
const idx = _.range(ads.length);
return this.shouldUseRandomOrder() ? _.shuffle(idx) : idx;
}
shouldUseRandomOrder(): boolean {
return !!this.conversation.settings.adSettings.randomOrder;
}
2019-06-07 19:31:42 +00:00
getAds(): string[] {
return this.conversation.settings.adSettings.ads;
}
getNextAd(): string | undefined {
const ads = this.getAds();
if (ads.length === 0)
return;
2020-12-28 23:07:10 +00:00
if (ads.length !== this.adMap.length) {
log.debug('adManager.regenerate.on-the-fly', ads.length, this.adMap.length);
this.adMap = this.generateAdMap();
}
return ads[this.adMap[this.adIndex % this.adMap.length] % ads.length];
2019-06-07 19:31:42 +00:00
}
getNextPostDue(): Date | undefined {
return this.nextPostDue;
}
getExpireDue(): Date | undefined {
return this.expireDue;
}
getFirstPost(): Date | undefined {
return this.firstPost;
}
2022-12-25 05:44:55 +00:00
start(timeoutMs = AdManager.POSTING_PERIOD): void {
2019-06-07 19:31:42 +00:00
const chanConv = (<Conversation.ChannelConversation>this.conversation);
2022-09-03 18:47:11 +00:00
const initialWait = Math.max(
Math.random() * AdManager.START_VARIANCE,
(chanConv.nextAd - Date.now()) * 1.1,
2022-09-03 18:49:47 +00:00
((this.conversation.settings.adSettings.lastAdTimestamp || 0) + (core.connection.vars.lfrp_flood * 1000)) - Date.now()
2022-09-03 18:47:11 +00:00
);
2019-06-07 19:31:42 +00:00
this.adIndex = 0;
this.active = true;
2022-09-03 17:46:40 +00:00
this.nextPostDue = new Date(Math.max(
Date.now() + initialWait,
2022-09-03 17:54:54 +00:00
(this.conversation.settings.adSettings.lastAdTimestamp || 0) + (core.connection.vars.lfrp_flood * 1000)
2022-09-03 17:46:40 +00:00
));
2022-12-25 05:44:55 +00:00
this.expireDue = new Date(Date.now() + timeoutMs);
2020-12-28 23:07:10 +00:00
this.adMap = this.generateAdMap();
2019-06-07 19:31:42 +00:00
2019-07-06 16:49:19 +00:00
// tslint:disable-next-line: no-unnecessary-type-assertion
2019-06-07 19:31:42 +00:00
this.interval = setTimeout(
async() => {
this.firstPost = new Date();
await this.sendNextPost();
},
initialWait
2019-07-06 16:49:19 +00:00
) as Timer;
2019-06-07 19:31:42 +00:00
}
2020-04-11 14:48:09 +00:00
protected forceTimeout(waitTime: number): void {
if (this.interval) {
clearTimeout(this.interval);
}
// tslint:disable-next-line: no-unnecessary-type-assertion
this.interval = setTimeout(
async() => {
await this.sendNextPost();
},
waitTime
) as Timer;
}
2019-06-07 19:31:42 +00:00
stop(): void {
if (this.interval)
clearTimeout(this.interval);
delete this.interval;
delete this.nextPostDue;
delete this.expireDue;
delete this.firstPost;
this.active = false;
this.adIndex = 0;
// const message = new EventMessage(`Advertisements on channel [channel]${this.conversation.name}[/channel] have expired.`);
// addEventMessage(message);
}
renew(): void {
if (!this.active)
return;
this.expireDue = new Date(Date.now() + 3 * 60 * 60 * 1000);
}
2020-04-11 14:48:09 +00:00
2020-04-19 17:06:50 +00:00
protected static recoverableCharacter = '';
2020-04-11 14:48:09 +00:00
protected static recoverableAds: RecoverableAd[] = [];
2020-04-19 17:06:50 +00:00
static onConnectionClosed(): void {
2021-02-09 02:52:03 +00:00
AdManager.recoverableCharacter = _.get(core, 'characters.ownCharacter.name', '');
2020-04-11 14:48:09 +00:00
AdManager.recoverableAds = _.map(
_.filter(core.conversations.channelConversations, (c) => ((c.adManager) && (c.adManager.isActive()))),
(chanConv): RecoverableAd => {
const adManager = chanConv.adManager;
return {
channel : chanConv.name,
index : adManager.adIndex,
nextPostDue : adManager.nextPostDue,
firstPost : adManager.firstPost,
expireDue : adManager.expireDue
};
}
);
_.each(
_.filter(core.conversations.channelConversations, (c) => ((c.adManager) && (c.adManager.isActive()))),
(c) => c.adManager.stop()
);
}
2020-04-19 17:06:50 +00:00
static onNewChannelAvailable(channel: ChannelConversation): void {
2020-04-11 14:48:09 +00:00
if (AdManager.recoverableCharacter !== core.characters.ownCharacter.name) {
AdManager.recoverableAds = [];
AdManager.recoverableCharacter = '';
return;
}
const ra = _.find(AdManager.recoverableAds, (r) => (r.channel === channel.name));
if (!ra) {
return;
}
const adManager = channel.adManager;
adManager.stop();
adManager.start();
adManager.adIndex = ra.index;
adManager.firstPost = ra.firstPost;
2022-09-03 19:26:12 +00:00
adManager.nextPostDue = adManager.nextPostDue || ra.nextPostDue || new Date();
2020-04-11 14:48:09 +00:00
adManager.expireDue = ra.expireDue;
adManager.forceTimeout(
2022-09-03 19:26:12 +00:00
Math.max(0, adManager.nextPostDue.getTime() - Date.now())
2020-04-11 14:48:09 +00:00
);
AdManager.recoverableAds = _.filter(AdManager.recoverableAds, (r) => (r.channel !== ra.channel));
}
2019-06-07 19:31:42 +00:00
}
2020-04-11 14:48:09 +00:00