forked from codeforanchorage/courtbot
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathdb.js
149 lines (135 loc) · 4.21 KB
/
db.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
145
146
147
148
149
require('dotenv').config();
const crypto = require('crypto');
const manager = require('./utils/db/manager');
const knex = manager.knex;
const log = require('./utils/logger')
/**
* encrypts the phone number
*
* param {string} phone number to encrypt
* returns {string} encrypted phone number
*/
function encryptPhone(phone) {
// Be careful when refactoring this function, the decipher object needs to be created
// each time a reminder is sent because the decipher.final() method destroys the object
// Reference: https://nodejs.org/api/crypto.html#crypto_decipher_final_output_encoding
const cipher = crypto.createCipher('aes256', process.env.PHONE_ENCRYPTION_KEY);
return cipher.update(phone, 'utf8', 'hex') + cipher.final('hex');
}
/**
* decrypts the phone number
*
* param {string} phone number to decrypt
* returns {string} decrypted phone number
*/
function decryptPhone(phone) {
// Be careful when refactoring this function, the decipher object needs to be created
// each time a reminder is sent because the decipher.final() method destroys the object
// Reference: https://nodejs.org/api/crypto.html#crypto_decipher_final_output_encoding
const decipher = crypto.createDecipher('aes256', process.env.PHONE_ENCRYPTION_KEY);
return decipher.update(phone, 'hex', 'utf8') + decipher.final('utf8');
}
/**
* Given a case id return the hearing(s)
* @param {string} case_id
* return
*/
function findCitation(case_id) {
return knex('hearings').where('case_id', case_id )
.select('*', knex.raw(`
CURRENT_DATE = date_trunc('day', date) as today,
date < CURRENT_TIMESTAMP as has_past
`))
}
/**
*
* @param {*} case_id
* @param {*} phone
*/
function findRequest(case_id, phone) {
return knex('requests').where('case_id', case_id )
.andWhere('phone', encryptPhone(phone) )
.andWhere('active', true)
.select('*')
}
/**
* Find request's case_ids based on phone
* @param {string} phone
* @returns {Promise} resolves to an array of case_ids
*/
function requestsFor(phone) {
return knex('requests')
.where('phone', encryptPhone(phone))
.select('case_id')
}
/**
* Deletes requests associated with phone number
* @param {string} phone
* @returns {Promise} resolves deleted case ids
*/
function deactivateRequestsFor(phone){
return knex('requests')
.where('phone', encryptPhone(phone))
.update('active', false)
.returning('case_id')
}
/**
* Adds the given request. Requests have a unique constraint on (case_id, phone)
* adding a duplicate will renew the updated_at date, which in the case of unmatched
* requests will start the clock on them again
* @param {*} data
* @returns {Promise} no resolve value
*/
function addRequest(data) {
data.phone = encryptPhone(data.phone)
return knex.raw(`
INSERT INTO requests
(case_id, phone, known_case)
VALUES(:case_id ,:phone, :known_case)
ON CONFLICT (case_id, phone) DO UPDATE SET updated_at = NOW(), active = true`,
{
case_id: data.case_id,
phone: data.phone,
known_case: data.known_case
}
)
}
/**
* Marks requests associated with the case_id and phone number as inactive
* @param {string} case_id
* @param {string} unencrypted phone number
*/
function deactivateRequest(case_id, phone) {
const enc_phone = encryptPhone(phone)
return knex('requests')
.where('phone', enc_phone)
.andWhere('case_id', case_id)
.update('active', false)
}
/**
* Find hearings based on case_id or partial name search
* @param {string} str
* @returns {Promise} array of rows from hearings table
*/
function fuzzySearch(str) {
const parts = str.trim().toUpperCase().split(' ');
// Search for Names
let query = knex('hearings').where('defendant', 'ilike', `%${parts[0]}%`);
if (parts.length > 1) query = query.andWhere('defendant', 'ilike', `%${parts[1]}%`);
// Search for Citations
query = query.orWhere('case_id',parts[0]);
// Limit to ten results
query = query.limit(10);
return query;
}
module.exports = {
addRequest,
decryptPhone,
encryptPhone,
findCitation,
fuzzySearch,
deactivateRequestsFor,
deactivateRequest,
requestsFor,
findRequest,
};