2016-03-14 10:20:24 -04:00
|
|
|
|
import { md5Hash, newObjectId } from './cryptoUtils';
|
2016-06-26 20:50:40 -07:00
|
|
|
|
import { logger } from './logger';
|
2016-03-13 23:34:44 -04:00
|
|
|
|
|
2016-04-14 19:24:56 -04:00
|
|
|
|
const PUSH_STATUS_COLLECTION = '_PushStatus';
|
|
|
|
|
|
|
2016-03-26 11:02:26 -04:00
|
|
|
|
export function flatten(array) {
|
|
|
|
|
|
return array.reduce((memo, element) => {
|
|
|
|
|
|
if (Array.isArray(element)) {
|
|
|
|
|
|
memo = memo.concat(flatten(element));
|
|
|
|
|
|
} else {
|
|
|
|
|
|
memo = memo.concat(element);
|
|
|
|
|
|
}
|
|
|
|
|
|
return memo;
|
|
|
|
|
|
}, []);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2016-03-13 23:34:44 -04:00
|
|
|
|
export default function pushStatusHandler(config) {
|
|
|
|
|
|
|
|
|
|
|
|
let initialPromise;
|
|
|
|
|
|
let pushStatus;
|
2016-04-07 18:08:09 -04:00
|
|
|
|
let objectId = newObjectId();
|
2016-06-26 20:50:40 -07:00
|
|
|
|
let database = config.database;
|
2016-07-23 21:20:04 +02:00
|
|
|
|
let lastPromise;
|
2016-03-29 22:42:37 -04:00
|
|
|
|
let setInitial = function(body = {}, where, options = {source: 'rest'}) {
|
2016-03-14 10:20:24 -04:00
|
|
|
|
let now = new Date();
|
2016-03-29 22:42:37 -04:00
|
|
|
|
let data = body.data || {};
|
2016-04-12 17:43:46 -04:00
|
|
|
|
let payloadString = JSON.stringify(data);
|
2016-08-09 18:00:56 +02:00
|
|
|
|
let pushHash;
|
|
|
|
|
|
if (typeof data.alert === 'string') {
|
|
|
|
|
|
pushHash = md5Hash(data.alert);
|
|
|
|
|
|
} else if (typeof data.alert === 'object') {
|
|
|
|
|
|
pushHash = md5Hash(JSON.stringify(data.alert));
|
|
|
|
|
|
} else {
|
|
|
|
|
|
pushHash = 'd41d8cd98f00b204e9800998ecf8427e';
|
|
|
|
|
|
}
|
2016-03-13 23:34:44 -04:00
|
|
|
|
let object = {
|
2016-04-21 21:36:15 -04:00
|
|
|
|
objectId,
|
|
|
|
|
|
createdAt: now,
|
2016-03-14 10:20:24 -04:00
|
|
|
|
pushTime: now.toISOString(),
|
2016-03-13 23:34:44 -04:00
|
|
|
|
query: JSON.stringify(where),
|
2016-04-12 17:43:46 -04:00
|
|
|
|
payload: payloadString,
|
2016-03-13 23:34:44 -04:00
|
|
|
|
source: options.source,
|
|
|
|
|
|
title: options.title,
|
|
|
|
|
|
expiry: body.expiration_time,
|
|
|
|
|
|
status: "pending",
|
|
|
|
|
|
numSent: 0,
|
2016-08-09 18:00:56 +02:00
|
|
|
|
pushHash,
|
2016-03-14 10:20:24 -04:00
|
|
|
|
// lockdown!
|
2016-04-21 21:36:15 -04:00
|
|
|
|
ACL: {}
|
2016-03-13 23:34:44 -04:00
|
|
|
|
}
|
2016-04-14 19:24:56 -04:00
|
|
|
|
|
2016-07-23 21:20:04 +02:00
|
|
|
|
lastPromise = database.create(PUSH_STATUS_COLLECTION, object).then(() => {
|
2016-03-14 10:20:24 -04:00
|
|
|
|
pushStatus = {
|
2016-04-12 17:43:46 -04:00
|
|
|
|
objectId
|
2016-03-14 10:20:24 -04:00
|
|
|
|
};
|
2016-03-13 23:34:44 -04:00
|
|
|
|
return Promise.resolve(pushStatus);
|
2016-04-14 19:24:56 -04:00
|
|
|
|
});
|
2016-07-23 21:20:04 +02:00
|
|
|
|
return lastPromise;
|
2016-03-13 23:34:44 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
2016-03-29 22:42:37 -04:00
|
|
|
|
let setRunning = function(installations) {
|
|
|
|
|
|
logger.verbose('sending push to %d installations', installations.length);
|
2016-07-23 21:20:04 +02:00
|
|
|
|
lastPromise = lastPromise.then(() => {
|
|
|
|
|
|
return database.update(PUSH_STATUS_COLLECTION,
|
|
|
|
|
|
{status:"pending", objectId: objectId},
|
|
|
|
|
|
{status: "running", updatedAt: new Date() });
|
|
|
|
|
|
});
|
|
|
|
|
|
return lastPromise;
|
2016-03-13 23:34:44 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
let complete = function(results) {
|
|
|
|
|
|
let update = {
|
|
|
|
|
|
status: 'succeeded',
|
2016-04-21 21:36:15 -04:00
|
|
|
|
updatedAt: new Date(),
|
2016-03-13 23:34:44 -04:00
|
|
|
|
numSent: 0,
|
|
|
|
|
|
numFailed: 0,
|
|
|
|
|
|
};
|
|
|
|
|
|
if (Array.isArray(results)) {
|
2016-03-26 11:02:26 -04:00
|
|
|
|
results = flatten(results);
|
2016-03-13 23:34:44 -04:00
|
|
|
|
results.reduce((memo, result) => {
|
|
|
|
|
|
// Cannot handle that
|
2016-04-18 22:30:19 -04:00
|
|
|
|
if (!result || !result.device || !result.device.deviceType) {
|
2016-03-13 23:34:44 -04:00
|
|
|
|
return memo;
|
|
|
|
|
|
}
|
|
|
|
|
|
let deviceType = result.device.deviceType;
|
|
|
|
|
|
if (result.transmitted)
|
|
|
|
|
|
{
|
|
|
|
|
|
memo.numSent++;
|
|
|
|
|
|
memo.sentPerType = memo.sentPerType || {};
|
|
|
|
|
|
memo.sentPerType[deviceType] = memo.sentPerType[deviceType] || 0;
|
|
|
|
|
|
memo.sentPerType[deviceType]++;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
memo.numFailed++;
|
|
|
|
|
|
memo.failedPerType = memo.failedPerType || {};
|
|
|
|
|
|
memo.failedPerType[deviceType] = memo.failedPerType[deviceType] || 0;
|
|
|
|
|
|
memo.failedPerType[deviceType]++;
|
|
|
|
|
|
}
|
|
|
|
|
|
return memo;
|
|
|
|
|
|
}, update);
|
|
|
|
|
|
}
|
2016-03-29 22:42:37 -04:00
|
|
|
|
logger.verbose('sent push! %d success, %d failures', update.numSent, update.numFailed);
|
2016-07-23 21:20:04 +02:00
|
|
|
|
lastPromise = lastPromise.then(() => {
|
|
|
|
|
|
return database.update(PUSH_STATUS_COLLECTION, {status:"running", objectId }, update);
|
|
|
|
|
|
});
|
|
|
|
|
|
return lastPromise;
|
2016-03-13 23:34:44 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
2016-03-29 22:42:37 -04:00
|
|
|
|
let fail = function(err) {
|
|
|
|
|
|
let update = {
|
|
|
|
|
|
errorMessage: JSON.stringify(err),
|
2016-04-21 21:36:15 -04:00
|
|
|
|
status: 'failed',
|
|
|
|
|
|
updatedAt: new Date()
|
2016-03-29 22:42:37 -04:00
|
|
|
|
}
|
2016-05-19 18:52:44 +02:00
|
|
|
|
logger.info('warning: error while sending push', err);
|
2016-07-23 21:20:04 +02:00
|
|
|
|
lastPromise = lastPromise.then(() => {
|
|
|
|
|
|
return database.update(PUSH_STATUS_COLLECTION, { objectId }, update);
|
|
|
|
|
|
});
|
|
|
|
|
|
return lastPromise;
|
2016-03-29 22:42:37 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
2016-03-13 23:34:44 -04:00
|
|
|
|
return Object.freeze({
|
2016-04-07 18:08:09 -04:00
|
|
|
|
objectId,
|
2016-03-13 23:34:44 -04:00
|
|
|
|
setInitial,
|
|
|
|
|
|
setRunning,
|
2016-03-29 22:42:37 -04:00
|
|
|
|
complete,
|
|
|
|
|
|
fail
|
2016-03-13 23:34:44 -04:00
|
|
|
|
})
|
|
|
|
|
|
}
|