@sjcrh/proteinpaint-server 2.99.1-0 → 2.99.1-1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@sjcrh/proteinpaint-server",
3
- "version": "2.99.1-0",
3
+ "version": "2.99.1-1",
4
4
  "type": "module",
5
5
  "description": "a genomics visualization tool for exploring a cohort's genotype and phenotype data",
6
6
  "main": "src/app.js",
@@ -61,16 +61,20 @@
61
61
  "@sjcrh/augen": "2.87.0",
62
62
  "@sjcrh/proteinpaint-rust": "2.99.0",
63
63
  "@sjcrh/proteinpaint-shared": "2.99.0",
64
- "@sjcrh/proteinpaint-types": "2.99.1-0",
64
+ "@sjcrh/proteinpaint-types": "2.99.1-1",
65
+ "@types/express": "^5.0.0",
66
+ "@types/express-session": "^1.18.1",
65
67
  "better-sqlite3": "^9.4.1",
66
68
  "body-parser": "^1.15.2",
67
69
  "canvas": "~2.11.2",
68
70
  "compression": "^1.6.2",
71
+ "connect-redis": "^6.1.3",
69
72
  "cookie-parser": "^1.4.5",
70
73
  "d3": "^7.6.1",
71
74
  "deep-object-diff": "^1.1.0",
72
75
  "express": "^4.17.1",
73
76
  "express-basic-auth": "^1.1.5",
77
+ "express-session": "^1.18.1",
74
78
  "got": "^14.2.0",
75
79
  "image-size": "^0.5.5",
76
80
  "jsonwebtoken": "^9.0.0",
@@ -81,6 +85,7 @@
81
85
  "minimatch": "^3.1.2",
82
86
  "node-fetch": "^2.6.1",
83
87
  "partjson": "^0.58.2",
88
+ "redis": "^4.7.0",
84
89
  "tiny-async-pool": "^1.2.0",
85
90
  "tough-cookie": "^4.1.4"
86
91
  },
@@ -0,0 +1,51 @@
1
+ import { clearWSImagesSessionsPayload } from "@sjcrh/proteinpaint-types/routes/clearwsisessions.js";
2
+ import SessionManager from "#src/wsisessions/SessionManager.js";
3
+ import ky from "ky";
4
+ import serverconfig from "#src/serverconfig.js";
5
+ const api = {
6
+ endpoint: "clearwsisession",
7
+ methods: {
8
+ delete: {
9
+ ...clearWSImagesSessionsPayload,
10
+ init
11
+ }
12
+ }
13
+ };
14
+ function init() {
15
+ return async (req, res) => {
16
+ try {
17
+ if (serverconfig.redis) {
18
+ const sessionsString = req.query.sessions;
19
+ const sessionsArray = JSON.parse(sessionsString);
20
+ const sessions = new Map(sessionsArray);
21
+ const sessionManager = SessionManager.getInstance(serverconfig.redis.url);
22
+ for (const [key, value] of sessions.entries()) {
23
+ const sessionData = await sessionManager.getSession(key);
24
+ if (sessionData) {
25
+ const userSessionIds = sessionData.userSessionIds;
26
+ if (!userSessionIds.some((sessionId) => sessionId === value)) {
27
+ break;
28
+ }
29
+ const newSessions = userSessionIds.filter((sessionId) => sessionId !== value);
30
+ if (newSessions.length === 0) {
31
+ await ky.put(`${serverconfig.tileServerURL}/tileserver/reset/${sessionData.imageSessionId}`);
32
+ await sessionManager.deleteSession(key);
33
+ } else {
34
+ sessionData.userSessionIds = newSessions;
35
+ await sessionManager.setSession(key, sessionData);
36
+ }
37
+ }
38
+ }
39
+ res.send({ message: "Sessions cleared" });
40
+ } else {
41
+ res.status(404).send("Redis not configured");
42
+ }
43
+ } catch (e) {
44
+ console.log(e);
45
+ res.status(404).send("Error clearing sessions");
46
+ }
47
+ };
48
+ }
49
+ export {
50
+ api
51
+ };
@@ -23,6 +23,17 @@ function init({ genomes }) {
23
23
  if (!ds)
24
24
  throw "invalid dataset name";
25
25
  const sampleId = query.sample_id;
26
+ if (ds.queries.WSImages.sources) {
27
+ const images2 = [];
28
+ if (ds.queries.WSImages.sources) {
29
+ images2.push({
30
+ filename: sampleId + "_fsspec.json",
31
+ metadata: ""
32
+ });
33
+ }
34
+ res.send({ sampleWSImages: images2 });
35
+ return;
36
+ }
26
37
  const images = await ds.queries.WSImages.getWSImages({ sampleId });
27
38
  res.send({ sampleWSImages: images });
28
39
  } catch (e) {
@@ -19,7 +19,7 @@ function init() {
19
19
  try {
20
20
  const { sampleId, TileGroup, z, x, y } = req.params;
21
21
  const url = `${serverconfig.tileServerURL}/tileserver/layer/slide/${sampleId}/zoomify/${TileGroup}/${z}-${x}-${y}@1x.jpg`;
22
- const response = await ky.get(url);
22
+ const response = await ky.get(url, { timeout: 12e4 });
23
23
  const buffer = await response.arrayBuffer();
24
24
  res.status(response.status).send(Buffer.from(buffer));
25
25
  } catch (error) {
@@ -5,6 +5,8 @@ import serverconfig from "#src/serverconfig.js";
5
5
  import { CookieJar } from "tough-cookie";
6
6
  import { promisify } from "util";
7
7
  import { wsImagesPayload } from "#types/checkers";
8
+ import SessionManager, { SessionData } from "../src/wsisessions/SessionManager.ts";
9
+ import crypto from "crypto";
8
10
  const routePath = "wsimages";
9
11
  const api = {
10
12
  endpoint: `${routePath}`,
@@ -25,93 +27,116 @@ function init({ genomes }) {
25
27
  const query = req.query;
26
28
  const g = genomes[query.genome];
27
29
  if (!g)
28
- throw "invalid genome name";
30
+ throw new Error("Invalid genome name");
29
31
  const ds = g.datasets[query.dslabel];
30
32
  if (!ds)
31
- throw "invalid dataset name";
33
+ throw new Error("Invalid dataset name");
32
34
  const sampleId = query.sampleId;
33
35
  if (!sampleId)
34
- throw "invalid sampleId";
36
+ throw new Error("Invalid sampleId");
35
37
  const wsimage = query.wsimage;
36
38
  if (!wsimage)
37
- throw "invalid wsimage";
39
+ throw new Error("Invalid wsimage");
38
40
  const cookieJar = new CookieJar();
39
41
  const setCookie = promisify(cookieJar.setCookie.bind(cookieJar));
40
42
  const getCookieString = promisify(cookieJar.getCookieString.bind(cookieJar));
41
- await ky.get(`${serverconfig.tileServerURL}/tileserver/session_id`, {
42
- hooks: {
43
- beforeRequest: [
44
- async (request) => {
45
- const cookie = await getCookieString(request.url);
46
- request.headers.set("Cookie", cookie);
47
- }
48
- ],
49
- afterResponse: [
50
- async (request, options, response) => {
51
- const setCookieHeader = response.headers.get("set-cookie");
52
- if (setCookieHeader) {
53
- await setCookie(setCookieHeader, request.url);
54
- }
55
- }
56
- ]
57
- }
58
- });
59
- const cookieString = await getCookieString(`${serverconfig.tileServerURL}/tileserver/session_id`);
60
- const sessionId = cookieString.match(/session_id=([^;]*)/)?.[1];
61
- const sampleWsiTileServer = path.join(
62
- `${serverconfig.tileServerMount}/${ds.queries.WSImages.imageBySampleFolder}/${sampleId}`,
63
- wsimage
64
- );
65
- const data = qs.stringify({ slide_path: sampleWsiTileServer });
66
- await ky.put(`${serverconfig.tileServerURL}/tileserver/slide`, {
67
- body: data,
68
- headers: {
69
- "Content-Type": "application/x-www-form-urlencoded",
70
- Cookie: `session_id=${sessionId}`
71
- // Include the session_id in the headers
72
- },
73
- hooks: {
74
- beforeRequest: [
75
- async (request) => {
76
- const cookie = await getCookieString(request.url);
77
- request.headers.set("Cookie", cookie);
78
- }
79
- ],
80
- afterResponse: [
81
- async (request, options, response) => {
82
- const setCookieHeader = response.headers.get("set-cookie");
83
- if (setCookieHeader) {
84
- await setCookie(setCookieHeader, request.url);
85
- }
86
- }
87
- ]
88
- }
89
- });
90
- const getWsiImageResponse = await ky.get(`${serverconfig.tileServerURL}/tileserver/slide`, {
91
- hooks: {
92
- beforeRequest: [
93
- async (request) => {
94
- const cookie = await getCookieString(request.url);
95
- request.headers.set("Cookie", cookie);
96
- }
97
- ]
98
- }
99
- }).json();
43
+ let sessionManager;
44
+ let sessionData;
45
+ let userSessionId = void 0;
46
+ if (serverconfig.redis) {
47
+ sessionManager = SessionManager.getInstance(serverconfig.redis.url);
48
+ userSessionId = crypto.createHash("sha256").update(crypto.randomBytes(32).toString("hex")).digest("hex");
49
+ sessionData = await sessionManager.getSession(wsimage);
50
+ }
51
+ const sessionId = sessionData ? sessionData.imageSessionId : await getSessionId(cookieJar, getCookieString, setCookie, wsimage, ds, sampleId);
52
+ if (serverconfig.redis && sessionManager) {
53
+ await manageUserSession(sessionManager, sessionData, wsimage, userSessionId, sessionId);
54
+ }
55
+ const getWsiImageResponse = await getWsiImageDimensions(sessionId, getCookieString);
100
56
  const payload = {
101
57
  status: "ok",
102
- sessionId,
58
+ wsiSessionId: sessionId,
59
+ browserImageInstanceId: serverconfig.redis ? userSessionId : void 0,
103
60
  slide_dimensions: getWsiImageResponse.slide_dimensions
104
61
  };
105
62
  res.status(200).json(payload);
106
63
  } catch (e) {
107
- console.log(e);
108
- res.send({
64
+ console.error(e);
65
+ res.status(500).send({
109
66
  status: "error",
110
- error: e.error || e
67
+ error: e.message || e
111
68
  });
112
69
  }
113
70
  };
114
71
  }
72
+ async function getSessionId(cookieJar, getCookieString, setCookie, wsimage, ds, sampleId) {
73
+ await ky.get(`${serverconfig.tileServerURL}/tileserver/session_id`, {
74
+ timeout: 5e4,
75
+ hooks: getHooks(cookieJar, getCookieString, setCookie)
76
+ });
77
+ const cookieString = await getCookieString(`${serverconfig.tileServerURL}/tileserver/session_id`);
78
+ const sessionId = cookieString.match(/session_id=([^;]*)/)?.[1];
79
+ if (!sessionId)
80
+ throw new Error("session_id not found");
81
+ const sampleWsiTileServer = path.join(
82
+ `${serverconfig.tileServerMount}/${ds.queries.WSImages.imageBySampleFolder}/${sampleId}`,
83
+ wsimage
84
+ );
85
+ const data = qs.stringify({ slide_path: sampleWsiTileServer });
86
+ await ky.put(`${serverconfig.tileServerURL}/tileserver/slide`, {
87
+ body: data,
88
+ timeout: 5e4,
89
+ headers: {
90
+ "Content-Type": "application/x-www-form-urlencoded",
91
+ Cookie: `session_id=${sessionId}`
92
+ },
93
+ hooks: getHooks(cookieJar, getCookieString, setCookie)
94
+ });
95
+ return sessionId;
96
+ }
97
+ async function manageUserSession(sessionManager, sessionData, wsimage, userId, sessionId) {
98
+ if (!sessionData) {
99
+ await sessionManager.setSession(wsimage, new SessionData(sessionId, [userId]));
100
+ } else if (!sessionData.userSessionIds || !sessionData.userSessionIds.includes(userId)) {
101
+ sessionData.userSessionIds = sessionData.userSessionIds || [];
102
+ sessionData.userSessionIds.push(userId);
103
+ await sessionManager.setSession(wsimage, sessionData);
104
+ }
105
+ }
106
+ async function getWsiImageDimensions(sessionId, getCookieString) {
107
+ return await ky.get(`${serverconfig.tileServerURL}/tileserver/slide`, {
108
+ timeout: 12e4,
109
+ hooks: {
110
+ beforeRequest: [
111
+ async (request) => {
112
+ let cookie = await getCookieString(request.url);
113
+ if (!cookie) {
114
+ cookie = `session_id=${sessionId}`;
115
+ }
116
+ request.headers.set("Cookie", cookie);
117
+ }
118
+ ]
119
+ }
120
+ }).json();
121
+ }
122
+ function getHooks(cookieJar, getCookieString, setCookie) {
123
+ return {
124
+ beforeRequest: [
125
+ async (request) => {
126
+ const cookie = await getCookieString(request.url);
127
+ request.headers.set("Cookie", cookie);
128
+ }
129
+ ],
130
+ afterResponse: [
131
+ async (request, options, response) => {
132
+ const setCookieHeader = response.headers.get("set-cookie");
133
+ if (setCookieHeader) {
134
+ await setCookie(setCookieHeader, request.url);
135
+ }
136
+ }
137
+ ]
138
+ };
139
+ }
115
140
  export {
116
141
  api
117
142
  };