Skip to content

Commit

Permalink
Revert concurrent sessions (#347)
Browse files Browse the repository at this point in the history
This is causing data loss issues for users. Will need to revisit later.

#346
  • Loading branch information
wavded authored Feb 7, 2022
1 parent 1be0cdf commit 818e94b
Show file tree
Hide file tree
Showing 3 changed files with 30 additions and 113 deletions.
98 changes: 25 additions & 73 deletions lib/connect-redis.js
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@ module.exports = function (session) {
// All callbacks should have a noop if none provided for compatibility
// with the most Redis clients.
const noop = () => {}
const TOMBSTONE = "TOMBSTONE"

class RedisStore extends Store {
constructor(options = {}) {
Expand All @@ -28,13 +27,12 @@ module.exports = function (session) {
this.disableTouch = options.disableTouch || false
}

get(sid, cb = noop, showTombs = false) {
get(sid, cb = noop) {
let key = this.prefix + sid

this.client.get(key, (err, data) => {
if (err) return cb(err)
if (!data) return cb()
if (data === TOMBSTONE) return cb(null, showTombs ? data : undefined)

let result
try {
Expand All @@ -47,40 +45,28 @@ module.exports = function (session) {
}

set(sid, sess, cb = noop) {
this.get(
sid,
(err, oldSess) => {
if (oldSess === TOMBSTONE) {
return cb()
} else if (oldSess && oldSess.lastModified !== sess.lastModified) {
sess = mergeDeep(oldSess, sess)
}
let args = [this.prefix + sid]
let value
sess.lastModified = Date.now()
try {
value = this.serializer.stringify(sess)
} catch (er) {
return cb(er)
}
args.push(value)
args.push("EX", this._getTTL(sess))
let args = [this.prefix + sid]

let ttl = 1
if (!this.disableTTL) {
ttl = this._getTTL(sess)
args.push("EX", ttl)
}
let value
try {
value = this.serializer.stringify(sess)
} catch (er) {
return cb(er)
}
args.push(value)

if (ttl > 0) {
this.client.set(args, cb)
} else {
// If the resulting TTL is negative we can delete / destroy the key
this.destroy(sid, cb)
}
},
true
)
let ttl = 1
if (!this.disableTTL) {
ttl = this._getTTL(sess)
args.push("EX", ttl)
}

if (ttl > 0) {
this.client.set(args, cb)
} else {
// If the resulting TTL is negative we can delete / destroy the key
this.destroy(sid, cb)
}
}

touch(sid, sess, cb = noop) {
Expand All @@ -95,9 +81,7 @@ module.exports = function (session) {

destroy(sid, cb = noop) {
let key = this.prefix + sid
this.client.set([key, TOMBSTONE, "EX", 300], (err) => {
cb(err, 1)
})
this.client.del(key, cb)
}

clear(cb = noop) {
Expand All @@ -108,9 +92,9 @@ module.exports = function (session) {
}

length(cb = noop) {
this.all((err, result) => {
this._getAllKeys((err, keys) => {
if (err) return cb(err)
return cb(null, result.length)
return cb(null, keys.length)
})
}

Expand All @@ -137,7 +121,7 @@ module.exports = function (session) {
let result
try {
result = sessions.reduce((accum, data, index) => {
if (!data || data === TOMBSTONE) return accum
if (!data) return accum
data = this.serializer.parse(data)
data.id = keys[index].substr(prefixLen)
accum.push(data)
Expand Down Expand Up @@ -189,35 +173,3 @@ module.exports = function (session) {

return RedisStore
}

/**
* Simple object check.
* @param item
* @returns {boolean}
*/
function isObject(item) {
return item && typeof item === "object" && !Array.isArray(item)
}

/**
* Deep merge two objects.
* @param target
* @param ...sources
*/
function mergeDeep(target, ...sources) {
if (!sources.length) return target
const source = sources.shift()

if (isObject(target) && isObject(source)) {
for (const key in source) {
if (isObject(source[key])) {
if (!target[key]) Object.assign(target, { [key]: {} })
mergeDeep(target[key], source[key])
} else {
Object.assign(target, { [key]: source[key] })
}
}
}

return mergeDeep(target, ...sources)
}
1 change: 0 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@
"eslint-config-prettier": "^8.3.0",
"express-session": "^1.17.0",
"ioredis": "^4.17.1",
"mockdate": "^2.0.5",
"nyc": "^15.0.1",
"prettier": "^2.0.5",
"redis-mock": "^0.56.3",
Expand Down
44 changes: 5 additions & 39 deletions test/connect-redis-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,10 +5,8 @@ const redisV3 = require("redis-v3")
const redisV4 = require("redis-v4")
const ioRedis = require("ioredis")
const redisMock = require("redis-mock")
const MockDate = require("mockdate")

let RedisStore = require("../")(session)
MockDate.set("2000-11-22")

let p =
(ctx, method) =>
Expand Down Expand Up @@ -72,34 +70,11 @@ test("redis-mock client", async (t) => {
test("teardown", redisSrv.disconnect)

async function lifecycleTest(store, t) {
await p(store, "set")("123", { foo: "bar3" })
let res = await p(store, "get")("123")
t.same(res, { foo: "bar3", lastModified: 974851200000 }, "get value 1")
await p(store, "set")("123", {
foo: "bar3",
luke: "skywalker",
obi: "wan",
lastModified: 974851000000,
})
await p(store, "set")("123", {
luke: "skywalker",
lastModified: 974851000000,
})
res = await p(store, "get")("123")
t.same(
res,
{ foo: "bar3", luke: "skywalker", obi: "wan", lastModified: 974851200000 },
"get merged value"
)

res = await p(store, "clear")()
t.ok(res >= 1, "cleared key")

res = await p(store, "set")("123", { foo: "bar" })
let res = await p(store, "set")("123", { foo: "bar" })
t.equal(res, "OK", "set value")

res = await p(store, "get")("123")
t.same(res, { foo: "bar", lastModified: 974851200000 }, "get value")
t.same(res, { foo: "bar" }, "get value")

res = await p(store.client, "ttl")("sess:123")
t.ok(res >= 86399, "check one day ttl")
Expand Down Expand Up @@ -133,29 +108,20 @@ async function lifecycleTest(store, t) {
t.same(
res,
[
{ id: "123", foo: "bar", lastModified: 974851200000 },
{ id: "456", cookie: { expires }, lastModified: 974851200000 },
{ id: "123", foo: "bar" },
{ id: "456", cookie: { expires } },
],
"stored two keys data"
)

res = await p(store, "destroy")("456")
t.equal(res, 1, "destroyed one")

res = await p(store, "get")("456")
t.equal(res, undefined, "tombstoned one")

res = await p(store, "set")("456", { a: "new hope" })
t.equal(res, undefined, "tombstoned set")

res = await p(store, "get")("456")
t.equal(res, undefined, "tombstoned two")

res = await p(store, "length")()
t.equal(res, 1, "one key remains")

res = await p(store, "clear")()
t.equal(res, 2, "cleared remaining key")
t.equal(res, 1, "cleared remaining key")

res = await p(store, "length")()
t.equal(res, 0, "no key remains")
Expand Down

0 comments on commit 818e94b

Please sign in to comment.