forked from CivicTechAtlanta/courtbot
-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathsendUnmatched.js
144 lines (133 loc) · 5.18 KB
/
sendUnmatched.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
/* eslint "no-console": "off" */
const db = require('./db.js');
const messages = require('./utils/messages');
const manager = require('./utils/db/manager');
const logger = require('./utils/logger');
const knex = manager.knex;
/**
* Retrieve array of requests that have sat too long.
*
* @return {Promise} Promise that resolves to an array of objects:
* [{phone: 'encrypted-phone', case_id: [id1, id2, ...]}, ...]
*/
function getExpiredRequests() {
/* We dont delete these all at once even though that's easier, becuase we only want to
delete if there's not a tillio (or other) error. */
return knex('requests')
.where('known_case', false)
.andWhere('active', true)
.and.whereRaw(`updated_at < CURRENT_DATE - interval '${process.env.QUEUE_TTL_DAYS} day'`)
.whereNotExists(function() { // should only be neccessary if there's an error in discoverNewCitations
this.select('*').from('hearings').whereRaw('hearings.case_id = requests.case_id');
})
.select('*')
}
/**
* Deletes given case_ids and sends unable-to-find message
* Perform both actions inside transaction so if we only update DB if twilio succeeds
* and don't send to Twilio if delete fails.
*
* @param {*} groupedRequest is an object with a phone and an array of case_ids.
*/
function notifyExpired(expiredRequest) {
const phone = db.decryptPhone(expiredRequest.phone);
return knex('requests')
.where('phone', expiredRequest.phone)
.and.where('case_id', expiredRequest.case_id )
.update('active', false)
.then(() => messages.send(phone, process.env.TWILIO_PHONE_NUMBER, messages.unableToFindCitationForTooLong(expiredRequest)))
.then(() => knex('notifications')
.insert({
case_id: expiredRequest.case_id,
phone:expiredRequest.phone,
type:'expired'
})
.then(() => expiredRequest )
)
.catch(err => {
/* The most likely error should be Twilio's error for users who have stopped service,
but other errors like bad numbers are possible. This adds an error to the notification
so end admin user can see if there were send errors on notifications */
return knex('notifications')
.insert({
case_id: expiredRequest.case_id,
phone:expiredRequest.phone,
type:'expired',
error: err.message
})
.then(() => {
logger.error(err)
expiredRequest.error = err
return expiredRequest;
})
})
}
/**
* Finds requests that have matched a real citation for the first time.
* These are identified with the 'know_case = false' flag
* @return {Promise} that resolves to case and request information
*/
function discoverNewCitations() {
return knex.select('*', knex.raw(`
CURRENT_DATE = date_trunc('day', date) as today,
date < CURRENT_TIMESTAMP as has_past`))
.from('requests')
.innerJoin('hearings', {'requests.case_id': 'hearings.case_id'})
.where('requests.known_case', false)
}
/**
* Inform subscriber that we found this case and will send reminders before future hearings.
* Perform both actions inside transaction so if we only update DB if twilio suceeds
* @param {*} request_case object from join of request and case table
*/
async function updateAndNotify(request_case) {
const phone = db.decryptPhone(request_case.phone);
return knex.update({
'known_case': true,
'updated_at': knex.fn.now()
})
.into('requests')
.where('phone', request_case.phone)
.andWhere('case_id', request_case.case_id )
.then(() => messages.send(phone, process.env.TWILIO_PHONE_NUMBER, messages.foundItWillRemind(true, request_case)))
.then(() => knex('notifications')
.insert({
case_id: request_case.case_id,
phone:request_case.phone,
type:'matched'
})
.then(() => request_case )
)
.catch(err => {
/* add error to notification so end admin user can see if there were send errors */
return knex('notifications')
.insert({
case_id: request_case.case_id,
phone:request_case.phone,
type:'matched',
error:err.message
})
.then(() => {
logger.error(err)
request_case.error = err
return request_case
})
})
}
/**
* Hook for processing all requests which have not yet been matched to a real case_id.
*
* @return {Promise} Promise to process all queued messages.
*/
async function sendUnmatched() {
const matched = await discoverNewCitations()
const matched_sent = await Promise.all(matched.map(r => updateAndNotify(r)))
const expired = await getExpiredRequests()
const expired_sent = await Promise.all(expired.map((r => notifyExpired(r))))
// returning these results to make it easier to log in one place
return {expired: expired_sent, matched: matched_sent }
}
module.exports = {
sendUnmatched,
getExpiredRequests
};