2020-02-18 00:41:32 +01:00
|
|
|
import rndstr from 'rndstr';
|
2022-02-27 03:07:39 +01:00
|
|
|
import { Note } from '@/models/entities/note.js';
|
|
|
|
import { User } from '@/models/entities/user.js';
|
|
|
|
import { PromoReads, PromoNotes, Notes, Users } from '@/models/index.js';
|
2020-02-18 00:41:32 +01:00
|
|
|
|
2020-02-18 11:05:11 +01:00
|
|
|
export async function injectPromo(timeline: Note[], user?: User | null) {
|
2020-02-18 00:41:32 +01:00
|
|
|
if (timeline.length < 5) return;
|
|
|
|
|
|
|
|
// TODO: readやexpireフィルタはクエリ側でやる
|
|
|
|
|
2022-03-26 07:34:00 +01:00
|
|
|
const reads = user ? await PromoReads.findBy({
|
2021-12-09 15:58:30 +01:00
|
|
|
userId: user.id,
|
2020-02-18 11:05:11 +01:00
|
|
|
}) : [];
|
2020-02-18 00:41:32 +01:00
|
|
|
|
|
|
|
let promos = await PromoNotes.find();
|
|
|
|
|
|
|
|
promos = promos.filter(n => n.expiresAt.getTime() > Date.now());
|
|
|
|
promos = promos.filter(n => !reads.map(r => r.noteId).includes(n.noteId));
|
|
|
|
|
|
|
|
if (promos.length === 0) return;
|
|
|
|
|
2020-02-18 11:05:11 +01:00
|
|
|
// Pick random promo
|
2020-02-18 00:41:32 +01:00
|
|
|
const promo = promos[Math.floor(Math.random() * promos.length)];
|
|
|
|
|
2022-03-26 07:34:00 +01:00
|
|
|
const note = await Notes.findOneByOrFail({ id: promo.noteId });
|
2020-02-18 00:41:32 +01:00
|
|
|
|
|
|
|
// Join
|
2022-03-26 07:34:00 +01:00
|
|
|
note.user = await Users.findOneByOrFail({ id: note.userId });
|
2020-02-18 00:41:32 +01:00
|
|
|
|
2020-02-18 11:05:11 +01:00
|
|
|
(note as any)._prId_ = rndstr('a-z0-9', 8);
|
2020-02-18 00:41:32 +01:00
|
|
|
|
|
|
|
// Inject promo
|
|
|
|
timeline.splice(3, 0, note);
|
|
|
|
}
|