mirror of
https://github.com/brendan-ch/project-inter-server.git
synced 2026-04-19 08:50:29 +00:00
Merge pull request #33 from brendan-ch/feat/flexible-timing-support-for-notifications
[INT-27] feat/flexible-timing-support-for-notifications
This commit is contained in:
@@ -66,6 +66,7 @@ type Query {
|
|||||||
system(id: ID): System
|
system(id: ID): System
|
||||||
|
|
||||||
isNotificationScheduled(input: NotificationInput!): Boolean
|
isNotificationScheduled(input: NotificationInput!): Boolean
|
||||||
|
secondsThresholdForNotification(input: NotificationInput!): Int
|
||||||
}
|
}
|
||||||
|
|
||||||
# Mutations
|
# Mutations
|
||||||
@@ -78,6 +79,7 @@ input NotificationInput {
|
|||||||
deviceId: ID!
|
deviceId: ID!
|
||||||
shuttleId: ID!
|
shuttleId: ID!
|
||||||
stopId: ID!
|
stopId: ID!
|
||||||
|
secondsThreshold: Int
|
||||||
}
|
}
|
||||||
|
|
||||||
type NotificationResponse {
|
type NotificationResponse {
|
||||||
@@ -90,4 +92,5 @@ type Notification {
|
|||||||
deviceId: ID!
|
deviceId: ID!
|
||||||
shuttleId: ID!
|
shuttleId: ID!
|
||||||
stopId: ID!
|
stopId: ID!
|
||||||
|
secondsThreshold: Int
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,14 +3,26 @@ import { TupleKey } from "../../types/TupleKey";
|
|||||||
import { IEta } from "../../entities/entities";
|
import { IEta } from "../../entities/entities";
|
||||||
import { AppleNotificationSender, NotificationAlertArguments } from "../senders/AppleNotificationSender";
|
import { AppleNotificationSender, NotificationAlertArguments } from "../senders/AppleNotificationSender";
|
||||||
|
|
||||||
export interface ScheduledNotificationData {
|
export interface NotificationLookupArguments {
|
||||||
deviceId: string;
|
deviceId: string;
|
||||||
shuttleId: string;
|
shuttleId: string;
|
||||||
stopId: string;
|
stopId: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface NotificationSchedulingArguments extends NotificationLookupArguments {
|
||||||
|
/**
|
||||||
|
* Value which specifies the ETA of the shuttle for when
|
||||||
|
* the notification should fire.
|
||||||
|
* For example, a secondsThreshold of 180 would mean that the notification
|
||||||
|
* fires when the ETA drops below 3 minutes.
|
||||||
|
*/
|
||||||
|
secondsThreshold: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
type DeviceIdSecondsThresholdAssociation = { [key: string]: number };
|
||||||
|
|
||||||
export class ETANotificationScheduler {
|
export class ETANotificationScheduler {
|
||||||
public readonly secondsThresholdForNotificationToFire = 180;
|
public static readonly defaultSecondsThresholdForNotificationToFire = 180;
|
||||||
|
|
||||||
constructor(private repository: GetterRepository,
|
constructor(private repository: GetterRepository,
|
||||||
private appleNotificationSender = new AppleNotificationSender()
|
private appleNotificationSender = new AppleNotificationSender()
|
||||||
@@ -26,11 +38,12 @@ export class ETANotificationScheduler {
|
|||||||
* An object of device ID arrays to deliver notifications to.
|
* An object of device ID arrays to deliver notifications to.
|
||||||
* The key should be a combination of the shuttle ID and
|
* The key should be a combination of the shuttle ID and
|
||||||
* stop ID, which can be generated using `TupleKey`.
|
* stop ID, which can be generated using `TupleKey`.
|
||||||
|
* The value is a dictionary of the device ID to the stored seconds threshold.
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
private deviceIdsToDeliverTo: { [key: string]: Set<string> } = {}
|
private deviceIdsToDeliverTo: { [key: string]: DeviceIdSecondsThresholdAssociation } = {}
|
||||||
|
|
||||||
private async sendEtaNotificationImmediately(notificationData: ScheduledNotificationData): Promise<boolean> {
|
private async sendEtaNotificationImmediately(notificationData: NotificationSchedulingArguments): Promise<boolean> {
|
||||||
const { deviceId, shuttleId, stopId } = notificationData;
|
const { deviceId, shuttleId, stopId } = notificationData;
|
||||||
|
|
||||||
const shuttle = await this.repository.getShuttleById(shuttleId);
|
const shuttle = await this.repository.getShuttleById(shuttleId);
|
||||||
@@ -61,33 +74,37 @@ export class ETANotificationScheduler {
|
|||||||
|
|
||||||
private async etaSubscriberCallback(eta: IEta) {
|
private async etaSubscriberCallback(eta: IEta) {
|
||||||
const tuple = new TupleKey(eta.shuttleId, eta.stopId);
|
const tuple = new TupleKey(eta.shuttleId, eta.stopId);
|
||||||
if (this.deviceIdsToDeliverTo[tuple.toString()] === undefined) {
|
const tupleKey = tuple.toString();
|
||||||
|
if (this.deviceIdsToDeliverTo[tupleKey] === undefined) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const deviceIdsToRemove = new Set<string>();
|
const deviceIdsToRemove = new Set<string>();
|
||||||
for (let deviceId of this.deviceIdsToDeliverTo[tuple.toString()].values()) {
|
for (let deviceId of Object.keys(this.deviceIdsToDeliverTo[tupleKey])) {
|
||||||
const deliveredSuccessfully = await this.sendEtaNotificationImmediatelyIfSecondsRemainingBelowThreshold(deviceId, eta);
|
const scheduledNotificationData: NotificationSchedulingArguments = {
|
||||||
|
deviceId,
|
||||||
|
secondsThreshold: this.deviceIdsToDeliverTo[tupleKey][deviceId],
|
||||||
|
shuttleId: eta.shuttleId,
|
||||||
|
stopId: eta.stopId,
|
||||||
|
}
|
||||||
|
|
||||||
|
const deliveredSuccessfully = await this.sendEtaNotificationImmediatelyIfSecondsRemainingBelowThreshold(scheduledNotificationData, eta.secondsRemaining);
|
||||||
if (deliveredSuccessfully) {
|
if (deliveredSuccessfully) {
|
||||||
deviceIdsToRemove.add(deviceId);
|
deviceIdsToRemove.add(deviceId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
deviceIdsToRemove.forEach((deviceId) => {
|
deviceIdsToRemove.forEach((deviceId) => {
|
||||||
this.deviceIdsToDeliverTo[tuple.toString()].delete(deviceId);
|
delete this.deviceIdsToDeliverTo[tupleKey][deviceId]
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private async sendEtaNotificationImmediatelyIfSecondsRemainingBelowThreshold(deviceId: string, eta: IEta) {
|
private async sendEtaNotificationImmediatelyIfSecondsRemainingBelowThreshold(notificationObject: NotificationSchedulingArguments, etaSecondsRemaining: number) {
|
||||||
if (eta.secondsRemaining > this.secondsThresholdForNotificationToFire) {
|
if (etaSecondsRemaining > notificationObject.secondsThreshold) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return await this.sendEtaNotificationImmediately({
|
return await this.sendEtaNotificationImmediately(notificationObject);
|
||||||
deviceId,
|
|
||||||
shuttleId: eta.shuttleId,
|
|
||||||
stopId: eta.stopId,
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -95,13 +112,16 @@ export class ETANotificationScheduler {
|
|||||||
* @param deviceId The device ID to send the notification to.
|
* @param deviceId The device ID to send the notification to.
|
||||||
* @param shuttleId Shuttle ID of ETA object to check.
|
* @param shuttleId Shuttle ID of ETA object to check.
|
||||||
* @param stopId Stop ID of ETA object to check.
|
* @param stopId Stop ID of ETA object to check.
|
||||||
|
* @param secondsThreshold Value which specifies the ETA of the shuttle for when
|
||||||
|
* the notification should fire.
|
||||||
*/
|
*/
|
||||||
public async scheduleNotification({ deviceId, shuttleId, stopId }: ScheduledNotificationData) {
|
public async scheduleNotification({ deviceId, shuttleId, stopId, secondsThreshold }: NotificationSchedulingArguments) {
|
||||||
const tuple = new TupleKey(shuttleId, stopId);
|
const tuple = new TupleKey(shuttleId, stopId);
|
||||||
if (this.deviceIdsToDeliverTo[tuple.toString()] === undefined) {
|
if (this.deviceIdsToDeliverTo[tuple.toString()] === undefined) {
|
||||||
this.deviceIdsToDeliverTo[tuple.toString()] = new Set();
|
this.deviceIdsToDeliverTo[tuple.toString()] = {};
|
||||||
}
|
}
|
||||||
this.deviceIdsToDeliverTo[tuple.toString()].add(deviceId);
|
|
||||||
|
this.deviceIdsToDeliverTo[tuple.toString()][deviceId] = secondsThreshold;
|
||||||
|
|
||||||
this.repository.unsubscribeFromEtaUpdates(this.etaSubscriberCallback);
|
this.repository.unsubscribeFromEtaUpdates(this.etaSubscriberCallback);
|
||||||
this.repository.subscribeToEtaUpdates(this.etaSubscriberCallback);
|
this.repository.subscribeToEtaUpdates(this.etaSubscriberCallback);
|
||||||
@@ -113,49 +133,52 @@ export class ETANotificationScheduler {
|
|||||||
* @param shuttleId Shuttle ID of the ETA object.
|
* @param shuttleId Shuttle ID of the ETA object.
|
||||||
* @param stopId Stop ID of the ETA object.
|
* @param stopId Stop ID of the ETA object.
|
||||||
*/
|
*/
|
||||||
public async cancelNotificationIfExists({ deviceId, shuttleId, stopId }: ScheduledNotificationData) {
|
public async cancelNotificationIfExists({ deviceId, shuttleId, stopId }: NotificationLookupArguments) {
|
||||||
const tupleKey = new TupleKey(shuttleId, stopId);
|
const tupleKey = new TupleKey(shuttleId, stopId);
|
||||||
if (
|
if (
|
||||||
this.deviceIdsToDeliverTo[tupleKey.toString()] === undefined
|
this.deviceIdsToDeliverTo[tupleKey.toString()] === undefined
|
||||||
|| !this.deviceIdsToDeliverTo[tupleKey.toString()].has(deviceId)
|
|| !(deviceId in this.deviceIdsToDeliverTo[tupleKey.toString()])
|
||||||
) {
|
) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.deviceIdsToDeliverTo[tupleKey.toString()].delete(deviceId);
|
delete this.deviceIdsToDeliverTo[tupleKey.toString()][deviceId];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check whether the notification is scheduled.
|
* Check whether the notification is scheduled.
|
||||||
* @param deviceId
|
|
||||||
* @param shuttleId
|
|
||||||
* @param stopId
|
|
||||||
*/
|
*/
|
||||||
public isNotificationScheduled({ deviceId, shuttleId, stopId }: ScheduledNotificationData): boolean {
|
public isNotificationScheduled(lookupArguments: NotificationLookupArguments): boolean {
|
||||||
|
return this.getSecondsThresholdForScheduledNotification(lookupArguments) != null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public getSecondsThresholdForScheduledNotification({ deviceId, shuttleId, stopId }: NotificationLookupArguments): number | null {
|
||||||
const tuple = new TupleKey(shuttleId, stopId);
|
const tuple = new TupleKey(shuttleId, stopId);
|
||||||
if (this.deviceIdsToDeliverTo[tuple.toString()] === undefined) {
|
if (this.deviceIdsToDeliverTo[tuple.toString()] === undefined) {
|
||||||
return false;
|
return null;
|
||||||
}
|
}
|
||||||
return this.deviceIdsToDeliverTo[tuple.toString()].has(deviceId);
|
return this.deviceIdsToDeliverTo[tuple.toString()][deviceId];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return all scheduled notification for the given device ID.
|
* Return all scheduled notification for the given device ID.
|
||||||
* @param deviceId
|
* @param deviceId
|
||||||
*/
|
*/
|
||||||
public async getAllScheduledNotificationsForDevice(deviceId: string): Promise<ScheduledNotificationData[]> {
|
public async getAllScheduledNotificationsForDevice(deviceId: string): Promise<NotificationLookupArguments[]> {
|
||||||
const scheduledNotifications: ScheduledNotificationData[] = [];
|
const scheduledNotifications: NotificationSchedulingArguments[] = [];
|
||||||
|
|
||||||
for (const key of Object.keys(this.deviceIdsToDeliverTo)) {
|
for (const key of Object.keys(this.deviceIdsToDeliverTo)) {
|
||||||
if (this.deviceIdsToDeliverTo[key].has(deviceId)) {
|
if (deviceId in this.deviceIdsToDeliverTo[key]) {
|
||||||
const tupleKey = TupleKey.fromExistingStringKey(key);
|
const tupleKey = TupleKey.fromExistingStringKey(key);
|
||||||
const shuttleId = tupleKey.tuple[0]
|
const shuttleId = tupleKey.tuple[0]
|
||||||
const stopId = tupleKey.tuple[1];
|
const stopId = tupleKey.tuple[1];
|
||||||
|
const secondsThreshold = this.deviceIdsToDeliverTo[key][deviceId];
|
||||||
|
|
||||||
scheduledNotifications.push({
|
scheduledNotifications.push({
|
||||||
shuttleId,
|
shuttleId,
|
||||||
stopId,
|
stopId,
|
||||||
deviceId,
|
deviceId,
|
||||||
|
secondsThreshold,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -96,7 +96,7 @@ export class AppleNotificationSender {
|
|||||||
req.setEncoding('utf8');
|
req.setEncoding('utf8');
|
||||||
|
|
||||||
await new Promise<void>((resolve, reject) => {
|
await new Promise<void>((resolve, reject) => {
|
||||||
req.on('response', (headers, flags) => {
|
req.on('response', (headers, _flags) => {
|
||||||
if (headers[":status"] !== 200) {
|
if (headers[":status"] !== 200) {
|
||||||
reject(`APNs request failed with status ${headers[":status"]}`);
|
reject(`APNs request failed with status ${headers[":status"]}`);
|
||||||
}
|
}
|
||||||
@@ -106,6 +106,7 @@ export class AppleNotificationSender {
|
|||||||
req.write(JSON.stringify({
|
req.write(JSON.stringify({
|
||||||
aps: {
|
aps: {
|
||||||
alert: notificationAlertArguments,
|
alert: notificationAlertArguments,
|
||||||
|
sound: "default"
|
||||||
}
|
}
|
||||||
}));
|
}));
|
||||||
req.end();
|
req.end();
|
||||||
@@ -130,12 +131,11 @@ export class AppleNotificationSender {
|
|||||||
const path = "/3/device/" + deviceId;
|
const path = "/3/device/" + deviceId;
|
||||||
const fullUrl = hostToUse + path;
|
const fullUrl = hostToUse + path;
|
||||||
|
|
||||||
const constructedObject = {
|
return {
|
||||||
fullUrl,
|
fullUrl,
|
||||||
host: hostToUse,
|
host: hostToUse,
|
||||||
path,
|
path,
|
||||||
}
|
};
|
||||||
return constructedObject;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,9 @@
|
|||||||
import { NotificationInput, NotificationResponse, Resolvers } from "../generated/graphql";
|
import { NotificationResponse, Resolvers } from "../generated/graphql";
|
||||||
import { ServerContext } from "../ServerContext";
|
import { ServerContext } from "../ServerContext";
|
||||||
|
import {
|
||||||
|
ETANotificationScheduler,
|
||||||
|
NotificationSchedulingArguments
|
||||||
|
} from "../notifications/schedulers/ETANotificationScheduler";
|
||||||
|
|
||||||
export const MutationResolvers: Resolvers<ServerContext> = {
|
export const MutationResolvers: Resolvers<ServerContext> = {
|
||||||
Mutation: {
|
Mutation: {
|
||||||
@@ -19,7 +23,14 @@ export const MutationResolvers: Resolvers<ServerContext> = {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
await context.notificationService.scheduleNotification(args.input);
|
const notificationData: NotificationSchedulingArguments = {
|
||||||
|
...args.input,
|
||||||
|
secondsThreshold: typeof args.input.secondsThreshold === 'number'
|
||||||
|
? args.input.secondsThreshold
|
||||||
|
: ETANotificationScheduler.defaultSecondsThresholdForNotificationToFire,
|
||||||
|
}
|
||||||
|
|
||||||
|
await context.notificationService.scheduleNotification(notificationData);
|
||||||
|
|
||||||
const response: NotificationResponse = {
|
const response: NotificationResponse = {
|
||||||
message: "Notification scheduled",
|
message: "Notification scheduled",
|
||||||
|
|||||||
@@ -3,10 +3,10 @@ import { Resolvers } from "../generated/graphql";
|
|||||||
|
|
||||||
export const QueryResolvers: Resolvers<ServerContext> = {
|
export const QueryResolvers: Resolvers<ServerContext> = {
|
||||||
Query: {
|
Query: {
|
||||||
systems: async (parent, args, contextValue, info) => {
|
systems: async (_parent, args, contextValue, _info) => {
|
||||||
return await contextValue.repository.getSystems();
|
return await contextValue.repository.getSystems();
|
||||||
},
|
},
|
||||||
system: async (parent, args, contextValue, info) => {
|
system: async (_parent, args, contextValue, _info) => {
|
||||||
if (!args.id) return null;
|
if (!args.id) return null;
|
||||||
const system = await contextValue.repository.getSystemById(args.id);
|
const system = await contextValue.repository.getSystemById(args.id);
|
||||||
if (system === null) return null;
|
if (system === null) return null;
|
||||||
@@ -19,6 +19,10 @@ export const QueryResolvers: Resolvers<ServerContext> = {
|
|||||||
isNotificationScheduled: async (_parent, args, contextValue, _info) => {
|
isNotificationScheduled: async (_parent, args, contextValue, _info) => {
|
||||||
const notificationData = args.input;
|
const notificationData = args.input;
|
||||||
return contextValue.notificationService.isNotificationScheduled(notificationData);
|
return contextValue.notificationService.isNotificationScheduled(notificationData);
|
||||||
}
|
},
|
||||||
|
secondsThresholdForNotification: async (_parent, args, contextValue, _info) => {
|
||||||
|
const notificationData = args.input;
|
||||||
|
return contextValue.notificationService.getSecondsThresholdForScheduledNotification(notificationData);
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -65,10 +65,12 @@ describe("ETANotificationScheduler", () => {
|
|||||||
deviceId: "1",
|
deviceId: "1",
|
||||||
shuttleId: eta.shuttleId,
|
shuttleId: eta.shuttleId,
|
||||||
stopId: eta.stopId,
|
stopId: eta.stopId,
|
||||||
|
secondsThreshold: 240,
|
||||||
}
|
}
|
||||||
const notificationData2 = {
|
const notificationData2 = {
|
||||||
...notificationData1,
|
...notificationData1,
|
||||||
deviceId: "2",
|
deviceId: "2",
|
||||||
|
secondsThreshold: 180,
|
||||||
}
|
}
|
||||||
return { eta, notificationData1, notificationData2 };
|
return { eta, notificationData1, notificationData2 };
|
||||||
}
|
}
|
||||||
@@ -79,7 +81,8 @@ describe("ETANotificationScheduler", () => {
|
|||||||
const notificationData = {
|
const notificationData = {
|
||||||
deviceId: "1",
|
deviceId: "1",
|
||||||
shuttleId: "1",
|
shuttleId: "1",
|
||||||
stopId: "1"
|
stopId: "1",
|
||||||
|
secondsThreshold: 120,
|
||||||
};
|
};
|
||||||
|
|
||||||
await notificationService.scheduleNotification(notificationData);
|
await notificationService.scheduleNotification(notificationData);
|
||||||
@@ -117,7 +120,7 @@ describe("ETANotificationScheduler", () => {
|
|||||||
const shuttle = await addMockShuttleToRepository(repository, "1");
|
const shuttle = await addMockShuttleToRepository(repository, "1");
|
||||||
const stop = await addMockStopToRepository(repository, "1");
|
const stop = await addMockStopToRepository(repository, "1");
|
||||||
const { eta, notificationData1 } = generateNotificationDataAndEta(shuttle, stop);
|
const { eta, notificationData1 } = generateNotificationDataAndEta(shuttle, stop);
|
||||||
eta.secondsRemaining = notificationService.secondsThresholdForNotificationToFire + 100;
|
notificationData1.secondsThreshold = eta.secondsRemaining - 10;
|
||||||
|
|
||||||
// Act
|
// Act
|
||||||
await notificationService.scheduleNotification(notificationData1);
|
await notificationService.scheduleNotification(notificationData1);
|
||||||
|
|||||||
@@ -53,6 +53,33 @@ describe("MutationResolvers", () => {
|
|||||||
const shuttle = await addMockShuttleToRepository(context.repository, system.id);
|
const shuttle = await addMockShuttleToRepository(context.repository, system.id);
|
||||||
const stop = await addMockStopToRepository(context.repository, system.id);
|
const stop = await addMockStopToRepository(context.repository, system.id);
|
||||||
|
|
||||||
|
const notificationInput = {
|
||||||
|
deviceId: "1",
|
||||||
|
shuttleId: shuttle.id,
|
||||||
|
stopId: stop.id,
|
||||||
|
secondsThreshold: 240,
|
||||||
|
};
|
||||||
|
const response = await getServerResponse(query, notificationInput);
|
||||||
|
|
||||||
|
assert(response.body.kind === "single");
|
||||||
|
expect(response.body.singleResult.errors).toBeUndefined();
|
||||||
|
|
||||||
|
const expectedNotificationData: any = {
|
||||||
|
...notificationInput,
|
||||||
|
}
|
||||||
|
delete expectedNotificationData.secondsThreshold;
|
||||||
|
const notificationResponse = response.body.singleResult.data?.scheduleNotification as any;
|
||||||
|
expect(notificationResponse?.success).toBe(true);
|
||||||
|
expect(notificationResponse?.data).toEqual(expectedNotificationData);
|
||||||
|
|
||||||
|
expect(context.notificationService.getSecondsThresholdForScheduledNotification(expectedNotificationData)).toBe(240);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("adds a notification with the default seconds threshold if none is provided", async () => {
|
||||||
|
const system = await addMockSystemToRepository(context.repository);
|
||||||
|
const shuttle = await addMockShuttleToRepository(context.repository, system.id);
|
||||||
|
const stop = await addMockStopToRepository(context.repository, system.id);
|
||||||
|
|
||||||
const notificationInput = {
|
const notificationInput = {
|
||||||
deviceId: "1",
|
deviceId: "1",
|
||||||
shuttleId: shuttle.id,
|
shuttleId: shuttle.id,
|
||||||
@@ -65,9 +92,8 @@ describe("MutationResolvers", () => {
|
|||||||
|
|
||||||
const notificationResponse = response.body.singleResult.data?.scheduleNotification as any;
|
const notificationResponse = response.body.singleResult.data?.scheduleNotification as any;
|
||||||
expect(notificationResponse?.success).toBe(true);
|
expect(notificationResponse?.success).toBe(true);
|
||||||
expect(notificationResponse?.data).toEqual(notificationInput);
|
|
||||||
|
|
||||||
expect(context.notificationService.isNotificationScheduled(notificationInput)).toBe(true);
|
expect(context.notificationService.getSecondsThresholdForScheduledNotification(notificationInput)).toBe(180);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("fails if the shuttle ID doesn't exist", async () => {
|
it("fails if the shuttle ID doesn't exist", async () => {
|
||||||
@@ -118,23 +144,29 @@ describe("MutationResolvers", () => {
|
|||||||
const shuttle = await addMockShuttleToRepository(context.repository, system.id);
|
const shuttle = await addMockShuttleToRepository(context.repository, system.id);
|
||||||
const stop = await addMockStopToRepository(context.repository, system.id);
|
const stop = await addMockStopToRepository(context.repository, system.id);
|
||||||
|
|
||||||
const notificationInput = {
|
const notificationInput: any = {
|
||||||
deviceId: "1",
|
deviceId: "1",
|
||||||
shuttleId: shuttle.id,
|
shuttleId: shuttle.id,
|
||||||
stopId: stop.id,
|
stopId: stop.id,
|
||||||
|
secondsThreshold: 180,
|
||||||
}
|
}
|
||||||
await context.notificationService.scheduleNotification(notificationInput);
|
await context.notificationService.scheduleNotification(notificationInput);
|
||||||
|
|
||||||
const response = await getServerResponse(query, notificationInput);
|
const notificationLookup = {
|
||||||
|
...notificationInput
|
||||||
|
}
|
||||||
|
delete notificationLookup.secondsThreshold;
|
||||||
|
|
||||||
|
const response = await getServerResponse(query, notificationLookup);
|
||||||
|
|
||||||
assert(response.body.kind === "single");
|
assert(response.body.kind === "single");
|
||||||
expect(response.body.singleResult.errors).toBeUndefined();
|
expect(response.body.singleResult.errors).toBeUndefined();
|
||||||
|
|
||||||
const notificationResponse = response.body.singleResult.data?.cancelNotification as any;
|
const notificationResponse = response.body.singleResult.data?.cancelNotification as any;
|
||||||
expect(notificationResponse.success).toBe(true);
|
expect(notificationResponse.success).toBe(true);
|
||||||
expect(notificationResponse.data).toEqual(notificationInput);
|
expect(notificationResponse.data).toEqual(notificationLookup);
|
||||||
|
|
||||||
expect(context.notificationService.isNotificationScheduled(notificationInput)).toBe(false);
|
expect(context.notificationService.isNotificationScheduled(notificationLookup)).toBe(false);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("fails if the notification doesn't exist", async () => {
|
it("fails if the notification doesn't exist", async () => {
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import { describe, expect, it } from "@jest/globals";
|
|||||||
import { generateMockSystems } from "../testHelpers/mockDataGenerators";
|
import { generateMockSystems } from "../testHelpers/mockDataGenerators";
|
||||||
import { setupTestServerContext, setupTestServerHolder } from "../testHelpers/apolloTestServerHelpers";
|
import { setupTestServerContext, setupTestServerHolder } from "../testHelpers/apolloTestServerHelpers";
|
||||||
import assert = require("node:assert");
|
import assert = require("node:assert");
|
||||||
import { ScheduledNotificationData } from "../../src/notifications/schedulers/ETANotificationScheduler";
|
import { NotificationSchedulingArguments } from "../../src/notifications/schedulers/ETANotificationScheduler";
|
||||||
import { addMockShuttleToRepository, addMockStopToRepository } from "../testHelpers/repositorySetupHelpers";
|
import { addMockShuttleToRepository, addMockStopToRepository } from "../testHelpers/repositorySetupHelpers";
|
||||||
|
|
||||||
// See Apollo documentation for integration test guide
|
// See Apollo documentation for integration test guide
|
||||||
@@ -96,32 +96,37 @@ describe("QueryResolvers", () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("isNotificationScheduled", () => {
|
describe("isNotificationScheduled and secondsThresholdForNotification", () => {
|
||||||
const query = `
|
const query = `
|
||||||
query IsNotificationScheduled($input: NotificationInput!) {
|
query IsNotificationScheduled($input: NotificationInput!) {
|
||||||
isNotificationScheduled(input: $input)
|
isNotificationScheduled(input: $input)
|
||||||
|
secondsThresholdForNotification(input: $input)
|
||||||
}
|
}
|
||||||
`
|
`;
|
||||||
|
|
||||||
it("returns true if the notification is scheduled", async () => {
|
it("returns correct data if the notification is scheduled", async () => {
|
||||||
// Arrange
|
// Arrange
|
||||||
const shuttle = await addMockShuttleToRepository(context.repository, "1");
|
const shuttle = await addMockShuttleToRepository(context.repository, "1");
|
||||||
const stop = await addMockStopToRepository(context.repository, "1")
|
const stop = await addMockStopToRepository(context.repository, "1")
|
||||||
|
|
||||||
const notification: ScheduledNotificationData = {
|
const notification: NotificationSchedulingArguments = {
|
||||||
shuttleId: shuttle.id,
|
shuttleId: shuttle.id,
|
||||||
stopId: stop.id,
|
stopId: stop.id,
|
||||||
deviceId: "1",
|
deviceId: "1",
|
||||||
|
secondsThreshold: 240,
|
||||||
};
|
};
|
||||||
await context.notificationService.scheduleNotification(notification);
|
await context.notificationService.scheduleNotification(notification);
|
||||||
|
|
||||||
|
const notificationLookup: any = {
|
||||||
|
...notification,
|
||||||
|
}
|
||||||
|
delete notificationLookup.secondsThreshold;
|
||||||
|
|
||||||
// Act
|
// Act
|
||||||
const response = await holder.testServer.executeOperation({
|
const response = await holder.testServer.executeOperation({
|
||||||
query,
|
query,
|
||||||
variables: {
|
variables: {
|
||||||
input: {
|
input: notificationLookup,
|
||||||
...notification,
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
contextValue: context,
|
contextValue: context,
|
||||||
@@ -130,10 +135,11 @@ describe("QueryResolvers", () => {
|
|||||||
// Assert
|
// Assert
|
||||||
assert(response.body.kind === "single");
|
assert(response.body.kind === "single");
|
||||||
expect(response.body.singleResult.errors).toBeUndefined();
|
expect(response.body.singleResult.errors).toBeUndefined();
|
||||||
|
expect(response.body.singleResult.data?.secondsThresholdForNotification).toEqual(240);
|
||||||
expect(response.body.singleResult.data?.isNotificationScheduled).toBe(true);
|
expect(response.body.singleResult.data?.isNotificationScheduled).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("returns false if the notification isn't scheduled", async () => {
|
it("returns false/null data if the notification isn't scheduled", async () => {
|
||||||
// Act
|
// Act
|
||||||
const response = await holder.testServer.executeOperation({
|
const response = await holder.testServer.executeOperation({
|
||||||
query,
|
query,
|
||||||
@@ -152,6 +158,7 @@ describe("QueryResolvers", () => {
|
|||||||
assert(response.body.kind === "single");
|
assert(response.body.kind === "single");
|
||||||
expect(response.body.singleResult.errors).toBeUndefined();
|
expect(response.body.singleResult.errors).toBeUndefined();
|
||||||
expect(response.body.singleResult.data?.isNotificationScheduled).toBe(false);
|
expect(response.body.singleResult.data?.isNotificationScheduled).toBe(false);
|
||||||
|
expect(response.body.singleResult.data?.secondsThresholdForNotification).toBe(null);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
Reference in New Issue
Block a user