2023-06-10 04:45:11 +00:00
|
|
|
import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common';
|
2022-09-17 18:27:08 +00:00
|
|
|
import push from 'web-push';
|
2023-04-14 04:50:05 +00:00
|
|
|
import * as Redis from 'ioredis';
|
2022-09-17 18:27:08 +00:00
|
|
|
import { DI } from '@/di-symbols.js';
|
2022-09-20 20:33:11 +00:00
|
|
|
import type { Config } from '@/config.js';
|
2023-06-25 12:13:15 +00:00
|
|
|
import type { Packed } from '@/misc/json-schema.js';
|
2022-09-17 18:27:08 +00:00
|
|
|
import { getNoteSummary } from '@/misc/get-note-summary.js';
|
2023-04-11 05:20:16 +00:00
|
|
|
import type { SwSubscription, SwSubscriptionsRepository } from '@/models/index.js';
|
2022-12-04 01:16:03 +00:00
|
|
|
import { MetaService } from '@/core/MetaService.js';
|
2022-12-04 08:05:32 +00:00
|
|
|
import { bindThis } from '@/decorators.js';
|
2023-04-11 05:20:16 +00:00
|
|
|
import { RedisKVCache } from '@/misc/cache.js';
|
2022-09-17 18:27:08 +00:00
|
|
|
|
2022-12-18 10:50:02 +00:00
|
|
|
// Defined also packages/sw/types.ts#L13
|
2023-02-22 05:51:34 +00:00
|
|
|
type PushNotificationsTypes = {
|
2022-09-17 18:27:08 +00:00
|
|
|
'notification': Packed<'Notification'>;
|
2022-12-18 10:50:02 +00:00
|
|
|
'unreadAntennaNote': {
|
|
|
|
antenna: { id: string, name: string };
|
|
|
|
note: Packed<'Note'>;
|
|
|
|
};
|
2023-04-11 05:11:39 +00:00
|
|
|
'readAllNotifications': undefined;
|
2022-09-17 18:27:08 +00:00
|
|
|
};
|
|
|
|
|
2022-12-18 10:50:02 +00:00
|
|
|
// Reduce length because push message servers have character limits
|
2023-02-22 05:58:41 +00:00
|
|
|
function truncateBody<T extends keyof PushNotificationsTypes>(type: T, body: PushNotificationsTypes[T]): PushNotificationsTypes[T] {
|
2023-02-19 23:13:37 +00:00
|
|
|
if (typeof body !== 'object') return body;
|
2022-12-18 10:50:02 +00:00
|
|
|
|
|
|
|
return {
|
|
|
|
...body,
|
|
|
|
...(('note' in body && body.note) ? {
|
|
|
|
note: {
|
|
|
|
...body.note,
|
|
|
|
// textをgetNoteSummaryしたものに置き換える
|
|
|
|
text: getNoteSummary(('type' in body && body.type === 'renote') ? body.note.renote as Packed<'Note'> : body.note),
|
2023-07-07 22:08:16 +00:00
|
|
|
|
2022-12-18 10:50:02 +00:00
|
|
|
cw: undefined,
|
|
|
|
reply: undefined,
|
|
|
|
renote: undefined,
|
|
|
|
user: type === 'notification' ? undefined as any : body.note.user,
|
2023-02-15 04:06:06 +00:00
|
|
|
},
|
2022-12-18 10:50:02 +00:00
|
|
|
} : {}),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
@Injectable()
|
2023-06-10 04:45:11 +00:00
|
|
|
export class PushNotificationService implements OnApplicationShutdown {
|
2023-04-11 05:20:16 +00:00
|
|
|
private subscriptionsCache: RedisKVCache<SwSubscription[]>;
|
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
constructor(
|
|
|
|
@Inject(DI.config)
|
|
|
|
private config: Config,
|
|
|
|
|
2023-04-11 05:20:16 +00:00
|
|
|
@Inject(DI.redis)
|
|
|
|
private redisClient: Redis.Redis,
|
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
@Inject(DI.swSubscriptionsRepository)
|
|
|
|
private swSubscriptionsRepository: SwSubscriptionsRepository,
|
|
|
|
|
|
|
|
private metaService: MetaService,
|
|
|
|
) {
|
2023-04-11 05:20:16 +00:00
|
|
|
this.subscriptionsCache = new RedisKVCache<SwSubscription[]>(this.redisClient, 'userSwSubscriptions', {
|
|
|
|
lifetime: 1000 * 60 * 60 * 1, // 1h
|
|
|
|
memoryCacheLifetime: 1000 * 60 * 3, // 3m
|
|
|
|
fetcher: (key) => this.swSubscriptionsRepository.findBy({ userId: key }),
|
|
|
|
toRedisConverter: (value) => JSON.stringify(value),
|
|
|
|
fromRedisConverter: (value) => JSON.parse(value),
|
|
|
|
});
|
2022-09-17 18:27:08 +00:00
|
|
|
}
|
|
|
|
|
2022-12-06 00:17:37 +00:00
|
|
|
@bindThis
|
2023-02-22 05:51:34 +00:00
|
|
|
public async pushNotification<T extends keyof PushNotificationsTypes>(userId: string, type: T, body: PushNotificationsTypes[T]) {
|
2022-09-17 18:27:08 +00:00
|
|
|
const meta = await this.metaService.fetch();
|
2023-07-07 22:08:16 +00:00
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
if (!meta.enableServiceWorker || meta.swPublicKey == null || meta.swPrivateKey == null) return;
|
2023-07-07 22:08:16 +00:00
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
// アプリケーションの連絡先と、サーバーサイドの鍵ペアの情報を登録
|
|
|
|
push.setVapidDetails(this.config.url,
|
|
|
|
meta.swPublicKey,
|
|
|
|
meta.swPrivateKey);
|
2023-07-07 22:08:16 +00:00
|
|
|
|
2023-04-11 05:20:16 +00:00
|
|
|
const subscriptions = await this.subscriptionsCache.fetch(userId);
|
2023-07-07 22:08:16 +00:00
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
for (const subscription of subscriptions) {
|
2023-04-11 05:11:39 +00:00
|
|
|
if ([
|
|
|
|
'readAllNotifications',
|
|
|
|
].includes(type) && !subscription.sendReadMessage) continue;
|
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
const pushSubscription = {
|
|
|
|
endpoint: subscription.endpoint,
|
|
|
|
keys: {
|
|
|
|
auth: subscription.auth,
|
|
|
|
p256dh: subscription.publickey,
|
|
|
|
},
|
|
|
|
};
|
2022-12-18 10:50:02 +00:00
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
push.sendNotification(pushSubscription, JSON.stringify({
|
|
|
|
type,
|
2022-12-18 10:50:02 +00:00
|
|
|
body: (type === 'notification' || type === 'unreadAntennaNote') ? truncateBody(type, body) : body,
|
2022-09-17 18:27:08 +00:00
|
|
|
userId,
|
|
|
|
dateTime: (new Date()).getTime(),
|
|
|
|
}), {
|
|
|
|
proxy: this.config.proxy,
|
|
|
|
}).catch((err: any) => {
|
|
|
|
//swLogger.info(err.statusCode);
|
|
|
|
//swLogger.info(err.headers);
|
|
|
|
//swLogger.info(err.body);
|
2023-07-07 22:08:16 +00:00
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
if (err.statusCode === 410) {
|
|
|
|
this.swSubscriptionsRepository.delete({
|
|
|
|
userId: userId,
|
|
|
|
endpoint: subscription.endpoint,
|
|
|
|
auth: subscription.auth,
|
|
|
|
publickey: subscription.publickey,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2023-06-10 04:45:11 +00:00
|
|
|
|
|
|
|
@bindThis
|
|
|
|
public dispose(): void {
|
|
|
|
this.subscriptionsCache.dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
@bindThis
|
|
|
|
public onApplicationShutdown(signal?: string | undefined): void {
|
|
|
|
this.dispose();
|
|
|
|
}
|
2022-09-17 18:27:08 +00:00
|
|
|
}
|