-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathindex.js
89 lines (79 loc) · 2.34 KB
/
index.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
'use strict'
const NODE_VERSION_REQUIREMENT = 16
const nodeVersion = Number(process.version.match(/^v(\d+)/)[1])
if (nodeVersion < NODE_VERSION_REQUIREMENT) {
throw new Error(`node '${process.version}' not supported, needs 'v14.x' or greater`)
}
const EventEmitter = require('events')
class SqliteStore extends EventEmitter {
/**
* @param {Object} sqlite3db database instance of better-sqlite3.
* @param {string} table table name where session data will be stored, defaults to `session`.
*/
constructor (sqlite3db, table = 'session') {
try {
sqlite3db.exec(`
create table ${table} (
sid TEXT PRIMARY KEY NOT NULL,
expires DATETIME NOT NULL,
session TEXT NOT NULL
)`
)
} catch (err) {
if (err.toString() !== 'SqliteError: table session already exists') {
throw err
}
}
super()
this.setSession = sqlite3db.prepare(`INSERT INTO ${table} (sid, expires, session) VALUES (?, ?, ?) ON CONFLICT (sid) DO UPDATE SET session = excluded.session, expires = excluded.expires`)
this.getSession = sqlite3db.prepare(`SELECT sid, expires, session FROM ${table} WHERE sid = ?`)
this.destroySession = sqlite3db.prepare(`DELETE FROM ${table} WHERE sid = ?`)
}
}
SqliteStore.prototype.set = function set (sessionId, session, callback) {
try {
this.setSession.run(
sessionId,
session.cookie._expires.toISOString(),
JSON.stringify(session)
)
callback(null)
} catch (err) {
if (err.code === 'SQLITE_CONSTRAINT_PRIMARYKEY') {
callback(null)
} else {
callback(err)
}
}
}
SqliteStore.prototype.get = function get (sessionId, callback) {
try {
const results = []
for (const row of this.getSession.iterate(sessionId)) {
results.push(row)
}
let session = null
if (results.length === 1) {
const found = JSON.parse(results[0].session)
if (found.cookie.expires) {
if (new Date() < new Date(found.cookie.expires)) {
session = found
}
} else {
session = found
}
}
callback(null, session)
} catch (err) {
callback(err)
}
}
SqliteStore.prototype.destroy = function destroy (sessionId, callback) {
try {
this.destroySession.run(sessionId)
callback(null)
} catch (err) {
callback(err)
}
}
module.exports = SqliteStore