@xwiki/cristal-electron-authentication-nextcloud-main 0.20.0 → 0.21.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/CHANGELOG.md CHANGED
@@ -1,5 +1,24 @@
1
1
  # @xwiki/cristal-electron-authentication-nextcloud-main
2
2
 
3
+ ## 0.21.1
4
+
5
+ ### Patch Changes
6
+
7
+ - Cristal 0.21.1 Release
8
+ - Updated dependencies
9
+ - @xwiki/cristal-authentication-api@0.21.1
10
+
11
+ ## 0.21.0
12
+
13
+ ### Minor Changes
14
+
15
+ - Cristal 0.21 Release
16
+
17
+ ### Patch Changes
18
+
19
+ - Updated dependencies
20
+ - @xwiki/cristal-authentication-api@0.21.0
21
+
3
22
  ## 0.20.0
4
23
 
5
24
  ### Minor Changes
package/dist/index.es.js CHANGED
@@ -1,6 +1,7 @@
1
- import M from "electron-store";
2
- import D from "axios";
3
- const k = "tokenType", T = "accessToken", m = "refreshToken", x = "expiryDate", v = "userId", L = {
1
+ import L from "electron-store";
2
+ import b from "axios";
3
+ import { ipcMain as c, shell as S, BrowserWindow as F } from "electron";
4
+ const w = "tokenType", T = "accessToken", x = "refreshToken", m = "expiryDate", _ = "userId", K = {
4
5
  tokenType: {
5
6
  type: "string"
6
7
  },
@@ -16,242 +17,198 @@ const k = "tokenType", T = "accessToken", m = "refreshToken", x = "expiryDate",
16
17
  userId: {
17
18
  type: "string"
18
19
  }
19
- }, R = new M({
20
+ }, $ = new L({
20
21
  name: "authentication-nextcloud-main",
21
- schema: L
22
+ schema: K
22
23
  // TODO: add encryption key
23
24
  });
24
- function i(e, n, a) {
25
- R.set(`${e}-${a}`, n);
25
+ function i(e, t, s) {
26
+ $.set(`${e}-${s}`, t);
26
27
  }
27
- function u(e, n) {
28
- return R.get(`${e}-${n}`);
28
+ function u(e, t) {
29
+ return $.get(`${e}-${t}`);
29
30
  }
30
- function l(e, n) {
31
- R.delete(`${e}-${n}`);
31
+ function l(e, t) {
32
+ $.delete(`${e}-${t}`);
32
33
  }
33
- function _(e, n) {
34
- i(k, e, n);
34
+ function y(e, t) {
35
+ i(w, e, t);
35
36
  }
36
- function f(e, n) {
37
- i(T, e, n);
37
+ function f(e, t) {
38
+ i(T, e, t);
38
39
  }
39
- function j(e, n) {
40
- i(m, e, n);
40
+ function U(e, t) {
41
+ i(x, e, t);
41
42
  }
42
- function S(e, n) {
43
- i(x, e, n);
43
+ function D(e, t) {
44
+ i(m, e, t);
44
45
  }
45
- function w(e, n) {
46
- i(v, e, n);
46
+ function g(e, t) {
47
+ i(_, e, t);
47
48
  }
48
- function P(e) {
49
- return u(k, e);
49
+ function v(e) {
50
+ return u(w, e);
50
51
  }
51
- function $(e) {
52
+ function I(e) {
52
53
  return u(T, e);
53
54
  }
54
- function U(e) {
55
- return u(m, e);
56
- }
57
- function F(e) {
55
+ function j(e) {
58
56
  return u(x, e);
59
57
  }
60
58
  function A(e) {
61
- return u(v, e);
59
+ return u(m, e);
62
60
  }
63
61
  function C(e) {
64
- l(k, e);
62
+ return u(_, e);
65
63
  }
66
- function K(e) {
67
- l(T, e);
64
+ function E(e) {
65
+ l(w, e);
68
66
  }
69
- function q(e) {
70
- l(m, e);
67
+ function z(e) {
68
+ l(T, e);
71
69
  }
72
70
  function B(e) {
73
71
  l(x, e);
74
72
  }
75
- function N(e) {
76
- l(v, e);
77
- }
78
- function z(e) {
79
- if (Object.prototype.hasOwnProperty.call(e, "__esModule")) return e;
80
- var n = e.default;
81
- if (typeof n == "function") {
82
- var a = function t() {
83
- var o = !1;
84
- try {
85
- o = this instanceof t;
86
- } catch {
87
- }
88
- return o ? Reflect.construct(n, arguments, this.constructor) : n.apply(this, arguments);
89
- };
90
- a.prototype = n.prototype;
91
- } else a = {};
92
- return Object.defineProperty(a, "__esModule", { value: !0 }), Object.keys(e).forEach(function(t) {
93
- var o = Object.getOwnPropertyDescriptor(e, t);
94
- Object.defineProperty(a, t, o.get ? o : {
95
- enumerable: !0,
96
- get: function() {
97
- return e[t];
98
- }
99
- });
100
- }), a;
73
+ function O(e) {
74
+ l(m, e);
101
75
  }
102
- const V = {}, W = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
103
- __proto__: null,
104
- default: V
105
- }, Symbol.toStringTag, { value: "Module" })), E = /* @__PURE__ */ z(W);
106
- var g, I;
107
- function H() {
108
- if (I) return g;
109
- I = 1;
110
- const e = E, n = E, a = n.join(__dirname, "path.txt");
111
- function t() {
112
- let o;
113
- if (e.existsSync(a) && (o = e.readFileSync(a, "utf-8")), process.env.ELECTRON_OVERRIDE_DIST_PATH)
114
- return n.join(process.env.ELECTRON_OVERRIDE_DIST_PATH, o || "electron");
115
- if (o)
116
- return n.join(__dirname, "dist", o);
117
- throw new Error("Electron failed to install correctly, please delete node_modules/electron and try installing again");
118
- }
119
- return g = t(), g;
76
+ function W(e) {
77
+ l(_, e);
120
78
  }
121
- var s = H();
122
- const b = "http://callback/";
123
- async function G(e, n, a) {
124
- const t = new URL(`${a}/token`);
125
- t.searchParams.set("base_url", n), t.searchParams.set("code", e), t.searchParams.set("redirect_uri", b);
126
- const o = await D.get(t.toString());
127
- _(o.data.token_type, "oauth2"), f(o.data.access_token, "oauth2"), j(o.data.refresh_token, "oauth2"), S(Date.now() + (o.data.expires_in - 10) * 1e3, "oauth2"), w(o.data.user_id, "oauth2");
79
+ const R = "http://callback/";
80
+ async function q(e, t, s) {
81
+ const n = new URL(`${s}/token`);
82
+ n.searchParams.set("base_url", t), n.searchParams.set("code", e), n.searchParams.set("redirect_uri", R);
83
+ const o = await b.get(n.toString());
84
+ y(o.data.token_type, "oauth2"), f(o.data.access_token, "oauth2"), U(o.data.refresh_token, "oauth2"), D(Date.now() + (o.data.expires_in - 10) * 1e3, "oauth2"), g(o.data.user_id, "oauth2");
128
85
  }
129
- function J(e, n, a, t) {
86
+ function M(e, t, s, n) {
130
87
  e.webContents.session.webRequest.onBeforeRequest(
131
88
  {
132
- urls: [`${b}*`]
89
+ urls: [`${R}*`]
133
90
  },
134
- async ({ url: o }, r) => {
135
- if (o.startsWith(t))
136
- r({ cancel: !1 });
91
+ async ({ url: o }, a) => {
92
+ if (o.startsWith(n))
93
+ a({ cancel: !1 });
137
94
  else {
138
- const c = new URL(o);
139
- await G(
140
- c.searchParams.get("code"),
141
- a,
142
- t
143
- ), p.show(), n(p), e?.close();
95
+ const r = new URL(o);
96
+ await q(
97
+ r.searchParams.get("code"),
98
+ s,
99
+ n
100
+ ), d.show(), t(d), e?.close();
144
101
  }
145
102
  }
146
103
  );
147
104
  }
148
- async function Q(e) {
149
- const n = new s.BrowserWindow({
105
+ async function N(e) {
106
+ const t = new F({
150
107
  width: 800,
151
108
  height: 600,
152
109
  webPreferences: {
153
110
  nodeIntegration: !1
154
111
  }
155
112
  });
156
- return n.setMenu(null), await n.loadURL(e), n;
113
+ return t.setMenu(null), await t.loadURL(e), t;
157
114
  }
158
- let O, p;
159
- function Z(e, n) {
160
- s.ipcMain.handle(
115
+ let P, d;
116
+ function J(e, t) {
117
+ c.handle(
161
118
  "authentication:nextcloud:loginOauth2",
162
- async (a, {
163
- baseUrl: t,
119
+ async (s, {
120
+ baseUrl: n,
164
121
  authenticationBaseUrl: o
165
122
  }) => {
166
- const r = new URL(`${o}/authorize`);
167
- r.searchParams.set("base_url", t), r.searchParams.set("redirect_uri", b), p = e, O = await Q(r.toString()), J(O, n, t, o), p.hide();
123
+ const a = new URL(`${o}/authorize`);
124
+ a.searchParams.set("base_url", n), a.searchParams.set("redirect_uri", R), d = e, P = await N(a.toString()), M(P, t, n, o), d.hide();
168
125
  }
169
- ), s.ipcMain.handle(
126
+ ), c.handle(
170
127
  "authentication:nextcloud:loginBasic",
171
- async (a, {
172
- username: t,
128
+ async (s, {
129
+ username: n,
173
130
  password: o
174
131
  }) => {
175
- f(btoa(`${t}:${o}`), "basic"), _("Basic", "basic"), w(t, "basic"), n(e);
132
+ f(btoa(`${n}:${o}`), "basic"), y("Basic", "basic"), g(n, "basic"), t(e);
176
133
  }
177
- ), s.ipcMain.handle(
134
+ ), c.handle(
178
135
  "authentication:nextcloud:loginFlow",
179
- async (a, {
180
- baseUrl: t
136
+ async (s, {
137
+ baseUrl: n
181
138
  }) => {
182
- const o = `${t}/index.php/login/v2`, c = await (await fetch(o, { method: "POST" })).json();
183
- s.shell.openExternal(c.login);
184
- const d = setInterval(async () => {
185
- const h = await fetch(c.poll.endpoint, {
139
+ const o = `${n}/index.php/login/v2`, r = await (await fetch(o, { method: "POST" })).json();
140
+ S.openExternal(r.login);
141
+ const p = setInterval(async () => {
142
+ const h = await fetch(r.poll.endpoint, {
186
143
  method: "POST",
187
144
  body: new URLSearchParams({
188
- token: c.poll.token
145
+ token: r.poll.token
189
146
  }),
190
147
  headers: {
191
148
  "Content-Type": "application/x-www-form-urlencoded"
192
149
  }
193
150
  });
194
151
  if (h.ok) {
195
- const y = await h.json();
152
+ const k = await h.json();
196
153
  f(
197
- btoa(`${y.loginName}:${y.appPassword}`),
154
+ btoa(`${k.loginName}:${k.appPassword}`),
198
155
  "login-flow"
199
- ), _("Basic", "login-flow"), w(y.loginName, "login-flow"), clearInterval(d), n(e);
156
+ ), y("Basic", "login-flow"), g(k.loginName, "login-flow"), clearInterval(p), t(e);
200
157
  }
201
158
  }, 3e3);
202
159
  }
203
- ), s.ipcMain.handle(
160
+ ), c.handle(
204
161
  "authentication:nextcloud:isLoggedIn",
205
- async (a, { mode: t }) => {
206
- const o = P(t), r = $(t);
207
- return o && r;
162
+ async (s, { mode: n }) => {
163
+ const o = v(n), a = I(n);
164
+ return o && a;
208
165
  }
209
- ), s.ipcMain.handle(
166
+ ), c.handle(
210
167
  "authentication:nextcloud:userDetails",
211
- async (a, { baseUrl: t, mode: o }) => {
212
- const r = A(o);
168
+ async (s, { baseUrl: n, mode: o }) => {
169
+ const a = C(o);
213
170
  return {
214
- profile: `${t}/u/${r}`,
215
- username: r,
216
- name: r,
171
+ profile: `${n}/u/${a}`,
172
+ username: a,
173
+ name: a,
217
174
  // TODO: Find a way to get the display name (CRISTAL-589).
218
- avatar: `${t}/avatar/${r}/64`
175
+ avatar: `${n}/avatar/${a}/64`
219
176
  // We want the 64x64 avatar.
220
177
  };
221
178
  }
222
- ), s.ipcMain.handle(
179
+ ), c.handle(
223
180
  "authentication:nextcloud:authorizationValue",
224
- async (a, { mode: t }) => ({
225
- tokenType: P(t),
226
- accessToken: $(t)
181
+ async (s, { mode: n }) => ({
182
+ tokenType: v(n),
183
+ accessToken: I(n)
227
184
  })
228
- ), s.ipcMain.handle(
185
+ ), c.handle(
229
186
  "authentication:nextcloud:logout",
230
- async (a, { mode: t }) => {
231
- K(t), C(t), q(t), B(t), N(t);
187
+ async (s, { mode: n }) => {
188
+ z(n), E(n), B(n), O(n), W(n);
232
189
  }
233
- ), s.ipcMain.handle(
190
+ ), c.handle(
234
191
  "authentication:nextcloud:refreshToken",
235
- async (a, {
236
- baseUrl: t,
192
+ async (s, {
193
+ baseUrl: n,
237
194
  authenticationBaseUrl: o
238
195
  }) => {
239
- if (Date.now() > F("oauth2")) {
240
- const r = new URL(`${o}/refresh`);
241
- r.searchParams.set("base_url", t), r.searchParams.set("refresh_token", U("oauth2"));
196
+ if (Date.now() > A("oauth2")) {
197
+ const a = new URL(`${o}/refresh`);
198
+ a.searchParams.set("base_url", n), a.searchParams.set("refresh_token", j("oauth2"));
242
199
  const {
243
200
  data: {
244
- access_token: c,
245
- refresh_token: d,
201
+ access_token: r,
202
+ refresh_token: p,
246
203
  expires_in: h
247
204
  }
248
- } = await D.get(r.toString());
249
- f(c, "oauth2"), j(d, "oauth2"), S(Date.now() + (h - 10) * 1e3, "oauth2");
205
+ } = await b.get(a.toString());
206
+ f(r, "oauth2"), U(p, "oauth2"), D(Date.now() + (h - 10) * 1e3, "oauth2");
250
207
  }
251
208
  }
252
209
  );
253
210
  }
254
211
  export {
255
- Z as load
212
+ J as load
256
213
  };
257
214
  //# sourceMappingURL=index.es.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.es.js","sources":["../src/storage.ts","../__vite-browser-external","../../../../../node_modules/.pnpm/electron@37.1.0/node_modules/electron/index.js","../src/index.ts"],"sourcesContent":["/*\n * See the LICENSE file distributed with this work for additional\n * information regarding copyright ownership.\n *\n * This is free software; you can redistribute it and/or modify it\n * under the terms of the GNU Lesser General Public License as\n * published by the Free Software Foundation; either version 2.1 of\n * the License, or (at your option) any later version.\n *\n * This software is distributed in the hope that it will be useful,\n * but WITHOUT ANY WARRANTY; without even the implied warranty of\n * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU\n * Lesser General Public License for more details.\n *\n * You should have received a copy of the GNU Lesser General Public\n * License along with this software; if not, write to the Free\n * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA\n * 02110-1301 USA, or see the FSF site: http://www.fsf.org.\n */\n\nimport Store from \"electron-store\";\n\nconst tokenTypeKey = \"tokenType\";\n\nconst accessTokenKey = \"accessToken\";\n\nconst refreshTokenKey = \"refreshToken\";\n\nconst expiryDateKey = \"expiryDate\";\n\nconst userIdKey = \"userId\";\n\nconst schema = {\n tokenType: {\n type: \"string\",\n },\n\n accessToken: {\n type: \"string\",\n },\n\n refreshToken: {\n type: \"string\",\n },\n\n expiryDate: {\n type: \"number\",\n },\n\n userId: {\n type: \"string\",\n },\n};\n\nconst storeInstance: Store = new Store({\n name: \"authentication-nextcloud-main\",\n schema,\n // TODO: add encryption key\n});\n\nfunction set<T>(key: string, value: T, mode: string) {\n // @ts-expect-error type resolution failing because of electron-store library bug\n storeInstance.set(`${key}-${mode}`, value);\n}\n\nfunction get<T>(key: string, mode: string): T {\n // @ts-expect-error type resolution failing because of electron-store library bug\n return storeInstance.get(`${key}-${mode}`) as T;\n}\n\nfunction rm(key: string, mode: string) {\n // @ts-expect-error type resolution failing because of electron-store library bug\n storeInstance.delete(`${key}-${mode}`);\n}\n\nfunction setTokenType(value: string, mode: string): void {\n set(tokenTypeKey, value, mode);\n}\n\nfunction setAccessToken(value: string, mode: string): void {\n set(accessTokenKey, value, mode);\n}\n\nfunction setRefreshToken(value: string, mode: string): void {\n set(refreshTokenKey, value, mode);\n}\n\nfunction setExpiryDate(value: number, mode: string): void {\n set(expiryDateKey, value, mode);\n}\n\nfunction setUserId(value: string, mode: string): void {\n set(userIdKey, value, mode);\n}\n\nfunction getTokenType(mode: string): string {\n return get(tokenTypeKey, mode);\n}\n\nfunction getAccessToken(mode: string): string {\n return get(accessTokenKey, mode);\n}\n\nfunction getRefreshToken(mode: string): string {\n return get(refreshTokenKey, mode);\n}\n\nfunction getExpiryDate(mode: string): number {\n return get(expiryDateKey, mode);\n}\n\nfunction getUserId(mode: string): string {\n return get(userIdKey, mode);\n}\n\nfunction deleteTokenType(mode: string): void {\n rm(tokenTypeKey, mode);\n}\n\nfunction deleteAccessToken(mode: string): void {\n rm(accessTokenKey, mode);\n}\n\nfunction deleteRefreshToken(mode: string): void {\n rm(refreshTokenKey, mode);\n}\n\nfunction deleteExpiryDate(mode: string): void {\n rm(expiryDateKey, mode);\n}\n\nfunction deleteUserId(mode: string): void {\n rm(userIdKey, mode);\n}\n\nexport {\n deleteAccessToken,\n deleteExpiryDate,\n deleteRefreshToken,\n deleteTokenType,\n deleteUserId,\n getAccessToken,\n getExpiryDate,\n getRefreshToken,\n getTokenType,\n getUserId,\n setAccessToken,\n setExpiryDate,\n setRefreshToken,\n setTokenType,\n setUserId,\n};\n","export default {}","const fs = require('fs');\nconst path = require('path');\n\nconst pathFile = path.join(__dirname, 'path.txt');\n\nfunction getElectronPath () {\n let executablePath;\n if (fs.existsSync(pathFile)) {\n executablePath = fs.readFileSync(pathFile, 'utf-8');\n }\n if (process.env.ELECTRON_OVERRIDE_DIST_PATH) {\n return path.join(process.env.ELECTRON_OVERRIDE_DIST_PATH, executablePath || 'electron');\n }\n if (executablePath) {\n return path.join(__dirname, 'dist', executablePath);\n } else {\n throw new Error('Electron failed to install correctly, please delete node_modules/electron and try installing again');\n }\n}\n\nmodule.exports = getElectronPath();\n","/*\n * See the LICENSE file distributed with this work for additional\n * information regarding copyright ownership.\n *\n * This is free software; you can redistribute it and/or modify it\n * under the terms of the GNU Lesser General Public License as\n * published by the Free Software Foundation; either version 2.1 of\n * the License, or (at your option) any later version.\n *\n * This software is distributed in the hope that it will be useful,\n * but WITHOUT ANY WARRANTY; without even the implied warranty of\n * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU\n * Lesser General Public License for more details.\n *\n * You should have received a copy of the GNU Lesser General Public\n * License along with this software; if not, write to the Free\n * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA\n * 02110-1301 USA, or see the FSF site: http://www.fsf.org.\n */\n\nimport {\n deleteAccessToken,\n deleteExpiryDate,\n deleteRefreshToken,\n deleteTokenType,\n deleteUserId,\n getAccessToken,\n getExpiryDate,\n getRefreshToken,\n getTokenType,\n getUserId,\n setAccessToken,\n setExpiryDate,\n setRefreshToken,\n setTokenType,\n setUserId,\n} from \"./storage.js\";\nimport { UserDetails } from \"@xwiki/cristal-authentication-api\";\nimport axios from \"axios\";\nimport { BrowserWindow, ipcMain, shell } from \"electron\";\n\nconst callbackUrl = \"http://callback/\";\n\nasync function getTokenFromCallbackCode(\n code: string,\n baseUrl: string,\n authenticationBaseUrl: string,\n) {\n const tokenUrl = new URL(`${authenticationBaseUrl}/token`);\n tokenUrl.searchParams.set(\"base_url\", baseUrl);\n tokenUrl.searchParams.set(\"code\", code);\n tokenUrl.searchParams.set(\"redirect_uri\", callbackUrl);\n const response = await axios.get(tokenUrl.toString());\n setTokenType(response.data.token_type, \"oauth2\");\n setAccessToken(response.data.access_token, \"oauth2\");\n setRefreshToken(response.data.refresh_token, \"oauth2\");\n // We apply a safety margin of 10s to the expiration date.\n setExpiryDate(Date.now() + (response.data.expires_in - 10) * 1000, \"oauth2\");\n setUserId(response.data.user_id, \"oauth2\");\n}\n\nfunction initAuth(\n win: BrowserWindow,\n reload: (win: BrowserWindow) => void,\n baseUrl: string,\n authenticationBaseUrl: string,\n) {\n win.webContents.session.webRequest.onBeforeRequest(\n {\n urls: [`${callbackUrl}*`],\n },\n async ({ url }, callback) => {\n if (url.startsWith(authenticationBaseUrl)) {\n // Allow for the redirects from the oidc server to be performed without being blocked.\n callback({ cancel: false });\n } else {\n const parsedURL = new URL(url);\n await getTokenFromCallbackCode(\n parsedURL.searchParams.get(\"code\")!,\n baseUrl,\n authenticationBaseUrl,\n );\n mainWin.show();\n reload(mainWin);\n win?.close();\n }\n },\n );\n}\n\nasync function createWindow(url: string) {\n const win = new BrowserWindow({\n width: 800,\n height: 600,\n webPreferences: {\n nodeIntegration: false,\n },\n });\n\n win.setMenu(null);\n\n await win.loadURL(url);\n\n return win;\n}\n\nlet authWin: BrowserWindow;\nlet mainWin: BrowserWindow;\n\nexport function load(\n browserWindow: BrowserWindow,\n reload: (win: BrowserWindow) => void,\n): void {\n ipcMain.handle(\n \"authentication:nextcloud:loginOauth2\",\n async (\n _event,\n {\n baseUrl,\n authenticationBaseUrl,\n }: {\n baseUrl: string;\n authenticationBaseUrl: string;\n },\n ): Promise<void> => {\n const authorizationUrl = new URL(`${authenticationBaseUrl}/authorize`);\n authorizationUrl.searchParams.set(\"base_url\", baseUrl);\n authorizationUrl.searchParams.set(\"redirect_uri\", callbackUrl);\n // Save the window asking for login (i.e., the main window), before creating\n // a new windows for the oidc web page. Then, hide the main window for the\n // duration of the authentication process.\n mainWin = browserWindow;\n authWin = await createWindow(authorizationUrl.toString());\n\n initAuth(authWin, reload, baseUrl, authenticationBaseUrl);\n mainWin.hide();\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:loginBasic\",\n async (\n _event,\n {\n username,\n password,\n }: {\n username: string;\n password: string;\n },\n ): Promise<void> => {\n setAccessToken(btoa(`${username}:${password}`), \"basic\");\n setTokenType(\"Basic\", \"basic\");\n setUserId(username, \"basic\");\n // We reload the content on successful login.\n reload(browserWindow);\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:loginFlow\",\n async (\n _event,\n {\n baseUrl,\n }: {\n baseUrl: string;\n },\n ): Promise<void> => {\n const loginFlowUrl = `${baseUrl}/index.php/login/v2`;\n\n const loginFlowResponse = await fetch(loginFlowUrl, { method: \"POST\" });\n const jsonLoginFlowResponse: {\n poll: { token: string; endpoint: string };\n login: string;\n } = await loginFlowResponse.json();\n\n shell.openExternal(jsonLoginFlowResponse.login);\n\n // This interval handles polling Nextcloud for the access token.\n // It will return a 404 error until the login process has succeeded.\n const intervalId = setInterval(async () => {\n const response = await fetch(jsonLoginFlowResponse.poll.endpoint, {\n method: \"POST\",\n body: new URLSearchParams({\n token: jsonLoginFlowResponse.poll.token,\n }),\n headers: {\n \"Content-Type\": \"application/x-www-form-urlencoded\",\n },\n });\n if (response.ok) {\n const jsonResponse: {\n loginName: string;\n appPassword: string;\n } = await response.json();\n setAccessToken(\n btoa(`${jsonResponse.loginName}:${jsonResponse.appPassword}`),\n \"login-flow\",\n );\n setTokenType(\"Basic\", \"login-flow\");\n setUserId(jsonResponse.loginName, \"login-flow\");\n clearInterval(intervalId);\n // We reload the content on successful login.\n reload(browserWindow);\n }\n }, 3000);\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:isLoggedIn\",\n async (_event, { mode }: { mode: string }) => {\n const tokenType = getTokenType(mode);\n const accessTokenKey = getAccessToken(mode);\n return tokenType && accessTokenKey;\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:userDetails\",\n async (\n _event,\n { baseUrl, mode }: { baseUrl: string; mode: string },\n ): Promise<UserDetails> => {\n const userId = getUserId(mode);\n return {\n profile: `${baseUrl}/u/${userId}`,\n username: userId,\n name: userId!, // TODO: Find a way to get the display name (CRISTAL-589).\n avatar: `${baseUrl}/avatar/${userId}/64`, // We want the 64x64 avatar.\n };\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:authorizationValue\",\n async (_event, { mode }: { mode: string }) => {\n return {\n tokenType: getTokenType(mode),\n accessToken: getAccessToken(mode),\n };\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:logout\",\n async (_event, { mode }: { mode: string }): Promise<void> => {\n deleteAccessToken(mode);\n deleteTokenType(mode);\n deleteRefreshToken(mode);\n deleteExpiryDate(mode);\n deleteUserId(mode);\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:refreshToken\",\n async (\n _event,\n {\n baseUrl,\n authenticationBaseUrl,\n }: { baseUrl: string; authenticationBaseUrl: string },\n ): Promise<void> => {\n if (Date.now() > getExpiryDate(\"oauth2\")) {\n const refreshUrl = new URL(`${authenticationBaseUrl}/refresh`);\n refreshUrl.searchParams.set(\"base_url\", baseUrl);\n refreshUrl.searchParams.set(\"refresh_token\", getRefreshToken(\"oauth2\"));\n\n const {\n data: {\n access_token: accessToken,\n refresh_token: refreshToken,\n expires_in: expiresIn,\n },\n } = await axios.get(refreshUrl.toString());\n\n setAccessToken(accessToken, \"oauth2\");\n setRefreshToken(refreshToken, \"oauth2\");\n // We apply a safety margin of 10s to the expiration date.\n setExpiryDate(Date.now() + (expiresIn - 10) * 1000, \"oauth2\");\n }\n },\n );\n}\n"],"names":["tokenTypeKey","accessTokenKey","refreshTokenKey","expiryDateKey","userIdKey","schema","storeInstance","Store","set","key","value","mode","get","rm","setTokenType","setAccessToken","setRefreshToken","setExpiryDate","setUserId","getTokenType","getAccessToken","getRefreshToken","getExpiryDate","getUserId","deleteTokenType","deleteAccessToken","deleteRefreshToken","deleteExpiryDate","deleteUserId","__viteBrowserExternal","fs","require$$0","path","require$$1","pathFile","getElectronPath","executablePath","electron","callbackUrl","getTokenFromCallbackCode","code","baseUrl","authenticationBaseUrl","tokenUrl","response","axios","initAuth","win","reload","url","callback","parsedURL","mainWin","createWindow","BrowserWindow","authWin","load","browserWindow","ipcMain","_event","authorizationUrl","username","password","loginFlowUrl","jsonLoginFlowResponse","shell","intervalId","jsonResponse","tokenType","userId","refreshUrl","accessToken","refreshToken","expiresIn"],"mappings":"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","x_google_ignoreList":[2]}
1
+ {"version":3,"file":"index.es.js","sources":["../src/storage.ts","../src/index.ts"],"sourcesContent":["/**\n * See the LICENSE file distributed with this work for additional\n * information regarding copyright ownership.\n *\n * This is free software; you can redistribute it and/or modify it\n * under the terms of the GNU Lesser General Public License as\n * published by the Free Software Foundation; either version 2.1 of\n * the License, or (at your option) any later version.\n *\n * This software is distributed in the hope that it will be useful,\n * but WITHOUT ANY WARRANTY; without even the implied warranty of\n * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU\n * Lesser General Public License for more details.\n *\n * You should have received a copy of the GNU Lesser General Public\n * License along with this software; if not, write to the Free\n * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA\n * 02110-1301 USA, or see the FSF site: http://www.fsf.org.\n */\n\nimport Store from \"electron-store\";\n\nconst tokenTypeKey = \"tokenType\";\n\nconst accessTokenKey = \"accessToken\";\n\nconst refreshTokenKey = \"refreshToken\";\n\nconst expiryDateKey = \"expiryDate\";\n\nconst userIdKey = \"userId\";\n\nconst schema = {\n tokenType: {\n type: \"string\",\n },\n\n accessToken: {\n type: \"string\",\n },\n\n refreshToken: {\n type: \"string\",\n },\n\n expiryDate: {\n type: \"number\",\n },\n\n userId: {\n type: \"string\",\n },\n};\n\nconst storeInstance: Store = new Store({\n name: \"authentication-nextcloud-main\",\n schema,\n // TODO: add encryption key\n});\n\nfunction set<T>(key: string, value: T, mode: string) {\n // @ts-expect-error type resolution failing because of electron-store library bug\n storeInstance.set(`${key}-${mode}`, value);\n}\n\nfunction get<T>(key: string, mode: string): T {\n // @ts-expect-error type resolution failing because of electron-store library bug\n return storeInstance.get(`${key}-${mode}`) as T;\n}\n\nfunction rm(key: string, mode: string) {\n // @ts-expect-error type resolution failing because of electron-store library bug\n storeInstance.delete(`${key}-${mode}`);\n}\n\nfunction setTokenType(value: string, mode: string): void {\n set(tokenTypeKey, value, mode);\n}\n\nfunction setAccessToken(value: string, mode: string): void {\n set(accessTokenKey, value, mode);\n}\n\nfunction setRefreshToken(value: string, mode: string): void {\n set(refreshTokenKey, value, mode);\n}\n\nfunction setExpiryDate(value: number, mode: string): void {\n set(expiryDateKey, value, mode);\n}\n\nfunction setUserId(value: string, mode: string): void {\n set(userIdKey, value, mode);\n}\n\nfunction getTokenType(mode: string): string {\n return get(tokenTypeKey, mode);\n}\n\nfunction getAccessToken(mode: string): string {\n return get(accessTokenKey, mode);\n}\n\nfunction getRefreshToken(mode: string): string {\n return get(refreshTokenKey, mode);\n}\n\nfunction getExpiryDate(mode: string): number {\n return get(expiryDateKey, mode);\n}\n\nfunction getUserId(mode: string): string {\n return get(userIdKey, mode);\n}\n\nfunction deleteTokenType(mode: string): void {\n rm(tokenTypeKey, mode);\n}\n\nfunction deleteAccessToken(mode: string): void {\n rm(accessTokenKey, mode);\n}\n\nfunction deleteRefreshToken(mode: string): void {\n rm(refreshTokenKey, mode);\n}\n\nfunction deleteExpiryDate(mode: string): void {\n rm(expiryDateKey, mode);\n}\n\nfunction deleteUserId(mode: string): void {\n rm(userIdKey, mode);\n}\n\nexport {\n deleteAccessToken,\n deleteExpiryDate,\n deleteRefreshToken,\n deleteTokenType,\n deleteUserId,\n getAccessToken,\n getExpiryDate,\n getRefreshToken,\n getTokenType,\n getUserId,\n setAccessToken,\n setExpiryDate,\n setRefreshToken,\n setTokenType,\n setUserId,\n};\n","/**\n * See the LICENSE file distributed with this work for additional\n * information regarding copyright ownership.\n *\n * This is free software; you can redistribute it and/or modify it\n * under the terms of the GNU Lesser General Public License as\n * published by the Free Software Foundation; either version 2.1 of\n * the License, or (at your option) any later version.\n *\n * This software is distributed in the hope that it will be useful,\n * but WITHOUT ANY WARRANTY; without even the implied warranty of\n * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU\n * Lesser General Public License for more details.\n *\n * You should have received a copy of the GNU Lesser General Public\n * License along with this software; if not, write to the Free\n * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA\n * 02110-1301 USA, or see the FSF site: http://www.fsf.org.\n */\n\nimport {\n deleteAccessToken,\n deleteExpiryDate,\n deleteRefreshToken,\n deleteTokenType,\n deleteUserId,\n getAccessToken,\n getExpiryDate,\n getRefreshToken,\n getTokenType,\n getUserId,\n setAccessToken,\n setExpiryDate,\n setRefreshToken,\n setTokenType,\n setUserId,\n} from \"./storage.js\";\nimport { UserDetails } from \"@xwiki/cristal-authentication-api\";\nimport axios from \"axios\";\nimport { BrowserWindow, ipcMain, shell } from \"electron\";\n\nconst callbackUrl = \"http://callback/\";\n\nasync function getTokenFromCallbackCode(\n code: string,\n baseUrl: string,\n authenticationBaseUrl: string,\n) {\n const tokenUrl = new URL(`${authenticationBaseUrl}/token`);\n tokenUrl.searchParams.set(\"base_url\", baseUrl);\n tokenUrl.searchParams.set(\"code\", code);\n tokenUrl.searchParams.set(\"redirect_uri\", callbackUrl);\n const response = await axios.get(tokenUrl.toString());\n setTokenType(response.data.token_type, \"oauth2\");\n setAccessToken(response.data.access_token, \"oauth2\");\n setRefreshToken(response.data.refresh_token, \"oauth2\");\n // We apply a safety margin of 10s to the expiration date.\n setExpiryDate(Date.now() + (response.data.expires_in - 10) * 1000, \"oauth2\");\n setUserId(response.data.user_id, \"oauth2\");\n}\n\nfunction initAuth(\n win: BrowserWindow,\n reload: (win: BrowserWindow) => void,\n baseUrl: string,\n authenticationBaseUrl: string,\n) {\n win.webContents.session.webRequest.onBeforeRequest(\n {\n urls: [`${callbackUrl}*`],\n },\n async ({ url }, callback) => {\n if (url.startsWith(authenticationBaseUrl)) {\n // Allow for the redirects from the oidc server to be performed without being blocked.\n callback({ cancel: false });\n } else {\n const parsedURL = new URL(url);\n await getTokenFromCallbackCode(\n parsedURL.searchParams.get(\"code\")!,\n baseUrl,\n authenticationBaseUrl,\n );\n mainWin.show();\n reload(mainWin);\n win?.close();\n }\n },\n );\n}\n\nasync function createWindow(url: string) {\n const win = new BrowserWindow({\n width: 800,\n height: 600,\n webPreferences: {\n nodeIntegration: false,\n },\n });\n\n win.setMenu(null);\n\n await win.loadURL(url);\n\n return win;\n}\n\nlet authWin: BrowserWindow;\nlet mainWin: BrowserWindow;\n\nexport function load(\n browserWindow: BrowserWindow,\n reload: (win: BrowserWindow) => void,\n): void {\n ipcMain.handle(\n \"authentication:nextcloud:loginOauth2\",\n async (\n _event,\n {\n baseUrl,\n authenticationBaseUrl,\n }: {\n baseUrl: string;\n authenticationBaseUrl: string;\n },\n ): Promise<void> => {\n const authorizationUrl = new URL(`${authenticationBaseUrl}/authorize`);\n authorizationUrl.searchParams.set(\"base_url\", baseUrl);\n authorizationUrl.searchParams.set(\"redirect_uri\", callbackUrl);\n // Save the window asking for login (i.e., the main window), before creating\n // a new windows for the oidc web page. Then, hide the main window for the\n // duration of the authentication process.\n mainWin = browserWindow;\n authWin = await createWindow(authorizationUrl.toString());\n\n initAuth(authWin, reload, baseUrl, authenticationBaseUrl);\n mainWin.hide();\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:loginBasic\",\n async (\n _event,\n {\n username,\n password,\n }: {\n username: string;\n password: string;\n },\n ): Promise<void> => {\n setAccessToken(btoa(`${username}:${password}`), \"basic\");\n setTokenType(\"Basic\", \"basic\");\n setUserId(username, \"basic\");\n // We reload the content on successful login.\n reload(browserWindow);\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:loginFlow\",\n async (\n _event,\n {\n baseUrl,\n }: {\n baseUrl: string;\n },\n ): Promise<void> => {\n const loginFlowUrl = `${baseUrl}/index.php/login/v2`;\n\n const loginFlowResponse = await fetch(loginFlowUrl, { method: \"POST\" });\n const jsonLoginFlowResponse: {\n poll: { token: string; endpoint: string };\n login: string;\n } = await loginFlowResponse.json();\n\n shell.openExternal(jsonLoginFlowResponse.login);\n\n // This interval handles polling Nextcloud for the access token.\n // It will return a 404 error until the login process has succeeded.\n const intervalId = setInterval(async () => {\n const response = await fetch(jsonLoginFlowResponse.poll.endpoint, {\n method: \"POST\",\n body: new URLSearchParams({\n token: jsonLoginFlowResponse.poll.token,\n }),\n headers: {\n \"Content-Type\": \"application/x-www-form-urlencoded\",\n },\n });\n if (response.ok) {\n const jsonResponse: {\n loginName: string;\n appPassword: string;\n } = await response.json();\n setAccessToken(\n btoa(`${jsonResponse.loginName}:${jsonResponse.appPassword}`),\n \"login-flow\",\n );\n setTokenType(\"Basic\", \"login-flow\");\n setUserId(jsonResponse.loginName, \"login-flow\");\n clearInterval(intervalId);\n // We reload the content on successful login.\n reload(browserWindow);\n }\n }, 3000);\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:isLoggedIn\",\n async (_event, { mode }: { mode: string }) => {\n const tokenType = getTokenType(mode);\n const accessTokenKey = getAccessToken(mode);\n return tokenType && accessTokenKey;\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:userDetails\",\n async (\n _event,\n { baseUrl, mode }: { baseUrl: string; mode: string },\n ): Promise<UserDetails> => {\n const userId = getUserId(mode);\n return {\n profile: `${baseUrl}/u/${userId}`,\n username: userId,\n name: userId!, // TODO: Find a way to get the display name (CRISTAL-589).\n avatar: `${baseUrl}/avatar/${userId}/64`, // We want the 64x64 avatar.\n };\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:authorizationValue\",\n async (_event, { mode }: { mode: string }) => {\n return {\n tokenType: getTokenType(mode),\n accessToken: getAccessToken(mode),\n };\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:logout\",\n async (_event, { mode }: { mode: string }): Promise<void> => {\n deleteAccessToken(mode);\n deleteTokenType(mode);\n deleteRefreshToken(mode);\n deleteExpiryDate(mode);\n deleteUserId(mode);\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:refreshToken\",\n async (\n _event,\n {\n baseUrl,\n authenticationBaseUrl,\n }: { baseUrl: string; authenticationBaseUrl: string },\n ): Promise<void> => {\n if (Date.now() > getExpiryDate(\"oauth2\")) {\n const refreshUrl = new URL(`${authenticationBaseUrl}/refresh`);\n refreshUrl.searchParams.set(\"base_url\", baseUrl);\n refreshUrl.searchParams.set(\"refresh_token\", getRefreshToken(\"oauth2\"));\n\n const {\n data: {\n access_token: accessToken,\n refresh_token: refreshToken,\n expires_in: expiresIn,\n },\n } = await axios.get(refreshUrl.toString());\n\n setAccessToken(accessToken, \"oauth2\");\n setRefreshToken(refreshToken, \"oauth2\");\n // We apply a safety margin of 10s to the expiration date.\n setExpiryDate(Date.now() + (expiresIn - 10) * 1000, \"oauth2\");\n }\n },\n );\n}\n"],"names":["tokenTypeKey","accessTokenKey","refreshTokenKey","expiryDateKey","userIdKey","schema","storeInstance","Store","set","key","value","mode","get","rm","setTokenType","setAccessToken","setRefreshToken","setExpiryDate","setUserId","getTokenType","getAccessToken","getRefreshToken","getExpiryDate","getUserId","deleteTokenType","deleteAccessToken","deleteRefreshToken","deleteExpiryDate","deleteUserId","callbackUrl","getTokenFromCallbackCode","code","baseUrl","authenticationBaseUrl","tokenUrl","response","axios","initAuth","win","reload","url","callback","parsedURL","mainWin","createWindow","BrowserWindow","authWin","load","browserWindow","ipcMain","_event","authorizationUrl","username","password","loginFlowUrl","jsonLoginFlowResponse","shell","intervalId","jsonResponse","tokenType","userId","refreshUrl","accessToken","refreshToken","expiresIn"],"mappings":"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"}
package/dist/index.umd.js CHANGED
@@ -1,2 +1,2 @@
1
- (function(i,u){typeof exports=="object"&&typeof module<"u"?u(exports,require("electron-store"),require("axios")):typeof define=="function"&&define.amd?define(["exports","electron-store","axios"],u):(i=typeof globalThis<"u"?globalThis:i||self,u(i["cristal_authentication-nextcloud-main"]={},i.Store,i.axios))})(this,function(i,u,$){"use strict";const _="tokenType",g="accessToken",w="refreshToken",T="expiryDate",k="userId",U={tokenType:{type:"string"},accessToken:{type:"string"},refreshToken:{type:"string"},expiryDate:{type:"number"},userId:{type:"string"}},x=new u({name:"authentication-nextcloud-main",schema:U});function l(e,n,a){x.set(`${e}-${a}`,n)}function f(e,n){return x.get(`${e}-${n}`)}function h(e,n){x.delete(`${e}-${n}`)}function v(e,n){l(_,e,n)}function d(e,n){l(g,e,n)}function I(e,n){l(w,e,n)}function O(e,n){l(T,e,n)}function m(e,n){l(k,e,n)}function j(e){return f(_,e)}function D(e){return f(g,e)}function F(e){return f(w,e)}function q(e){return f(T,e)}function A(e){return f(k,e)}function B(e){h(_,e)}function C(e){h(g,e)}function K(e){h(w,e)}function N(e){h(T,e)}function z(e){h(k,e)}function V(e){if(Object.prototype.hasOwnProperty.call(e,"__esModule"))return e;var n=e.default;if(typeof n=="function"){var a=function t(){var o=!1;try{o=this instanceof t}catch{}return o?Reflect.construct(n,arguments,this.constructor):n.apply(this,arguments)};a.prototype=n.prototype}else a={};return Object.defineProperty(a,"__esModule",{value:!0}),Object.keys(e).forEach(function(t){var o=Object.getOwnPropertyDescriptor(e,t);Object.defineProperty(a,t,o.get?o:{enumerable:!0,get:function(){return e[t]}})}),a}const S=V(Object.freeze(Object.defineProperty({__proto__:null,default:{}},Symbol.toStringTag,{value:"Module"})));var R,M;function W(){if(M)return R;M=1;const e=S,n=S,a=n.join(__dirname,"path.txt");function t(){let o;if(e.existsSync(a)&&(o=e.readFileSync(a,"utf-8")),process.env.ELECTRON_OVERRIDE_DIST_PATH)return n.join(process.env.ELECTRON_OVERRIDE_DIST_PATH,o||"electron");if(o)return n.join(__dirname,"dist",o);throw new Error("Electron failed to install correctly, please delete node_modules/electron and try installing again")}return R=t(),R}var s=W();const b="http://callback/";async function H(e,n,a){const t=new URL(`${a}/token`);t.searchParams.set("base_url",n),t.searchParams.set("code",e),t.searchParams.set("redirect_uri",b);const o=await $.get(t.toString());v(o.data.token_type,"oauth2"),d(o.data.access_token,"oauth2"),I(o.data.refresh_token,"oauth2"),O(Date.now()+(o.data.expires_in-10)*1e3,"oauth2"),m(o.data.user_id,"oauth2")}function G(e,n,a,t){e.webContents.session.webRequest.onBeforeRequest({urls:[`${b}*`]},async({url:o},r)=>{if(o.startsWith(t))r({cancel:!1});else{const c=new URL(o);await H(c.searchParams.get("code"),a,t),p.show(),n(p),e?.close()}})}async function J(e){const n=new s.BrowserWindow({width:800,height:600,webPreferences:{nodeIntegration:!1}});return n.setMenu(null),await n.loadURL(e),n}let L,p;function Q(e,n){s.ipcMain.handle("authentication:nextcloud:loginOauth2",async(a,{baseUrl:t,authenticationBaseUrl:o})=>{const r=new URL(`${o}/authorize`);r.searchParams.set("base_url",t),r.searchParams.set("redirect_uri",b),p=e,L=await J(r.toString()),G(L,n,t,o),p.hide()}),s.ipcMain.handle("authentication:nextcloud:loginBasic",async(a,{username:t,password:o})=>{d(btoa(`${t}:${o}`),"basic"),v("Basic","basic"),m(t,"basic"),n(e)}),s.ipcMain.handle("authentication:nextcloud:loginFlow",async(a,{baseUrl:t})=>{const o=`${t}/index.php/login/v2`,c=await(await fetch(o,{method:"POST"})).json();s.shell.openExternal(c.login);const E=setInterval(async()=>{const y=await fetch(c.poll.endpoint,{method:"POST",body:new URLSearchParams({token:c.poll.token}),headers:{"Content-Type":"application/x-www-form-urlencoded"}});if(y.ok){const P=await y.json();d(btoa(`${P.loginName}:${P.appPassword}`),"login-flow"),v("Basic","login-flow"),m(P.loginName,"login-flow"),clearInterval(E),n(e)}},3e3)}),s.ipcMain.handle("authentication:nextcloud:isLoggedIn",async(a,{mode:t})=>{const o=j(t),r=D(t);return o&&r}),s.ipcMain.handle("authentication:nextcloud:userDetails",async(a,{baseUrl:t,mode:o})=>{const r=A(o);return{profile:`${t}/u/${r}`,username:r,name:r,avatar:`${t}/avatar/${r}/64`}}),s.ipcMain.handle("authentication:nextcloud:authorizationValue",async(a,{mode:t})=>({tokenType:j(t),accessToken:D(t)})),s.ipcMain.handle("authentication:nextcloud:logout",async(a,{mode:t})=>{C(t),B(t),K(t),N(t),z(t)}),s.ipcMain.handle("authentication:nextcloud:refreshToken",async(a,{baseUrl:t,authenticationBaseUrl:o})=>{if(Date.now()>q("oauth2")){const r=new URL(`${o}/refresh`);r.searchParams.set("base_url",t),r.searchParams.set("refresh_token",F("oauth2"));const{data:{access_token:c,refresh_token:E,expires_in:y}}=await $.get(r.toString());d(c,"oauth2"),I(E,"oauth2"),O(Date.now()+(y-10)*1e3,"oauth2")}})}i.load=Q,Object.defineProperty(i,Symbol.toStringTag,{value:"Module"})});
1
+ (function(c,u){typeof exports=="object"&&typeof module<"u"?u(exports,require("electron-store"),require("axios"),require("electron")):typeof define=="function"&&define.amd?define(["exports","electron-store","axios","electron"],u):(c=typeof globalThis<"u"?globalThis:c||self,u(c["cristal_authentication-nextcloud-main"]={},c.Store,c.axios,c.electron))})(this,function(c,u,P,i){"use strict";const k="tokenType",g="accessToken",w="refreshToken",T="expiryDate",x="userId",S={tokenType:{type:"string"},accessToken:{type:"string"},refreshToken:{type:"string"},expiryDate:{type:"number"},userId:{type:"string"}},_=new u({name:"authentication-nextcloud-main",schema:S});function l(e,t,s){_.set(`${e}-${s}`,t)}function h(e,t){return _.get(`${e}-${t}`)}function d(e,t){_.delete(`${e}-${t}`)}function m(e,t){l(k,e,t)}function f(e,t){l(g,e,t)}function b(e,t){l(w,e,t)}function U(e,t){l(T,e,t)}function $(e,t){l(x,e,t)}function D(e){return h(k,e)}function M(e){return h(g,e)}function j(e){return h(w,e)}function F(e){return h(T,e)}function K(e){return h(x,e)}function q(e){d(k,e)}function A(e){d(g,e)}function C(e){d(w,e)}function E(e){d(T,e)}function O(e){d(x,e)}const R="http://callback/";async function z(e,t,s){const n=new URL(`${s}/token`);n.searchParams.set("base_url",t),n.searchParams.set("code",e),n.searchParams.set("redirect_uri",R);const o=await P.get(n.toString());m(o.data.token_type,"oauth2"),f(o.data.access_token,"oauth2"),b(o.data.refresh_token,"oauth2"),U(Date.now()+(o.data.expires_in-10)*1e3,"oauth2"),$(o.data.user_id,"oauth2")}function B(e,t,s,n){e.webContents.session.webRequest.onBeforeRequest({urls:[`${R}*`]},async({url:o},a)=>{if(o.startsWith(n))a({cancel:!1});else{const r=new URL(o);await z(r.searchParams.get("code"),s,n),p.show(),t(p),e?.close()}})}async function W(e){const t=new i.BrowserWindow({width:800,height:600,webPreferences:{nodeIntegration:!1}});return t.setMenu(null),await t.loadURL(e),t}let L,p;function N(e,t){i.ipcMain.handle("authentication:nextcloud:loginOauth2",async(s,{baseUrl:n,authenticationBaseUrl:o})=>{const a=new URL(`${o}/authorize`);a.searchParams.set("base_url",n),a.searchParams.set("redirect_uri",R),p=e,L=await W(a.toString()),B(L,t,n,o),p.hide()}),i.ipcMain.handle("authentication:nextcloud:loginBasic",async(s,{username:n,password:o})=>{f(btoa(`${n}:${o}`),"basic"),m("Basic","basic"),$(n,"basic"),t(e)}),i.ipcMain.handle("authentication:nextcloud:loginFlow",async(s,{baseUrl:n})=>{const o=`${n}/index.php/login/v2`,r=await(await fetch(o,{method:"POST"})).json();i.shell.openExternal(r.login);const v=setInterval(async()=>{const y=await fetch(r.poll.endpoint,{method:"POST",body:new URLSearchParams({token:r.poll.token}),headers:{"Content-Type":"application/x-www-form-urlencoded"}});if(y.ok){const I=await y.json();f(btoa(`${I.loginName}:${I.appPassword}`),"login-flow"),m("Basic","login-flow"),$(I.loginName,"login-flow"),clearInterval(v),t(e)}},3e3)}),i.ipcMain.handle("authentication:nextcloud:isLoggedIn",async(s,{mode:n})=>{const o=D(n),a=M(n);return o&&a}),i.ipcMain.handle("authentication:nextcloud:userDetails",async(s,{baseUrl:n,mode:o})=>{const a=K(o);return{profile:`${n}/u/${a}`,username:a,name:a,avatar:`${n}/avatar/${a}/64`}}),i.ipcMain.handle("authentication:nextcloud:authorizationValue",async(s,{mode:n})=>({tokenType:D(n),accessToken:M(n)})),i.ipcMain.handle("authentication:nextcloud:logout",async(s,{mode:n})=>{A(n),q(n),C(n),E(n),O(n)}),i.ipcMain.handle("authentication:nextcloud:refreshToken",async(s,{baseUrl:n,authenticationBaseUrl:o})=>{if(Date.now()>F("oauth2")){const a=new URL(`${o}/refresh`);a.searchParams.set("base_url",n),a.searchParams.set("refresh_token",j("oauth2"));const{data:{access_token:r,refresh_token:v,expires_in:y}}=await P.get(a.toString());f(r,"oauth2"),b(v,"oauth2"),U(Date.now()+(y-10)*1e3,"oauth2")}})}c.load=N,Object.defineProperty(c,Symbol.toStringTag,{value:"Module"})});
2
2
  //# sourceMappingURL=index.umd.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.umd.js","sources":["../src/storage.ts","../__vite-browser-external","../../../../../node_modules/.pnpm/electron@37.1.0/node_modules/electron/index.js","../src/index.ts"],"sourcesContent":["/*\n * See the LICENSE file distributed with this work for additional\n * information regarding copyright ownership.\n *\n * This is free software; you can redistribute it and/or modify it\n * under the terms of the GNU Lesser General Public License as\n * published by the Free Software Foundation; either version 2.1 of\n * the License, or (at your option) any later version.\n *\n * This software is distributed in the hope that it will be useful,\n * but WITHOUT ANY WARRANTY; without even the implied warranty of\n * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU\n * Lesser General Public License for more details.\n *\n * You should have received a copy of the GNU Lesser General Public\n * License along with this software; if not, write to the Free\n * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA\n * 02110-1301 USA, or see the FSF site: http://www.fsf.org.\n */\n\nimport Store from \"electron-store\";\n\nconst tokenTypeKey = \"tokenType\";\n\nconst accessTokenKey = \"accessToken\";\n\nconst refreshTokenKey = \"refreshToken\";\n\nconst expiryDateKey = \"expiryDate\";\n\nconst userIdKey = \"userId\";\n\nconst schema = {\n tokenType: {\n type: \"string\",\n },\n\n accessToken: {\n type: \"string\",\n },\n\n refreshToken: {\n type: \"string\",\n },\n\n expiryDate: {\n type: \"number\",\n },\n\n userId: {\n type: \"string\",\n },\n};\n\nconst storeInstance: Store = new Store({\n name: \"authentication-nextcloud-main\",\n schema,\n // TODO: add encryption key\n});\n\nfunction set<T>(key: string, value: T, mode: string) {\n // @ts-expect-error type resolution failing because of electron-store library bug\n storeInstance.set(`${key}-${mode}`, value);\n}\n\nfunction get<T>(key: string, mode: string): T {\n // @ts-expect-error type resolution failing because of electron-store library bug\n return storeInstance.get(`${key}-${mode}`) as T;\n}\n\nfunction rm(key: string, mode: string) {\n // @ts-expect-error type resolution failing because of electron-store library bug\n storeInstance.delete(`${key}-${mode}`);\n}\n\nfunction setTokenType(value: string, mode: string): void {\n set(tokenTypeKey, value, mode);\n}\n\nfunction setAccessToken(value: string, mode: string): void {\n set(accessTokenKey, value, mode);\n}\n\nfunction setRefreshToken(value: string, mode: string): void {\n set(refreshTokenKey, value, mode);\n}\n\nfunction setExpiryDate(value: number, mode: string): void {\n set(expiryDateKey, value, mode);\n}\n\nfunction setUserId(value: string, mode: string): void {\n set(userIdKey, value, mode);\n}\n\nfunction getTokenType(mode: string): string {\n return get(tokenTypeKey, mode);\n}\n\nfunction getAccessToken(mode: string): string {\n return get(accessTokenKey, mode);\n}\n\nfunction getRefreshToken(mode: string): string {\n return get(refreshTokenKey, mode);\n}\n\nfunction getExpiryDate(mode: string): number {\n return get(expiryDateKey, mode);\n}\n\nfunction getUserId(mode: string): string {\n return get(userIdKey, mode);\n}\n\nfunction deleteTokenType(mode: string): void {\n rm(tokenTypeKey, mode);\n}\n\nfunction deleteAccessToken(mode: string): void {\n rm(accessTokenKey, mode);\n}\n\nfunction deleteRefreshToken(mode: string): void {\n rm(refreshTokenKey, mode);\n}\n\nfunction deleteExpiryDate(mode: string): void {\n rm(expiryDateKey, mode);\n}\n\nfunction deleteUserId(mode: string): void {\n rm(userIdKey, mode);\n}\n\nexport {\n deleteAccessToken,\n deleteExpiryDate,\n deleteRefreshToken,\n deleteTokenType,\n deleteUserId,\n getAccessToken,\n getExpiryDate,\n getRefreshToken,\n getTokenType,\n getUserId,\n setAccessToken,\n setExpiryDate,\n setRefreshToken,\n setTokenType,\n setUserId,\n};\n","export default {}","const fs = require('fs');\nconst path = require('path');\n\nconst pathFile = path.join(__dirname, 'path.txt');\n\nfunction getElectronPath () {\n let executablePath;\n if (fs.existsSync(pathFile)) {\n executablePath = fs.readFileSync(pathFile, 'utf-8');\n }\n if (process.env.ELECTRON_OVERRIDE_DIST_PATH) {\n return path.join(process.env.ELECTRON_OVERRIDE_DIST_PATH, executablePath || 'electron');\n }\n if (executablePath) {\n return path.join(__dirname, 'dist', executablePath);\n } else {\n throw new Error('Electron failed to install correctly, please delete node_modules/electron and try installing again');\n }\n}\n\nmodule.exports = getElectronPath();\n","/*\n * See the LICENSE file distributed with this work for additional\n * information regarding copyright ownership.\n *\n * This is free software; you can redistribute it and/or modify it\n * under the terms of the GNU Lesser General Public License as\n * published by the Free Software Foundation; either version 2.1 of\n * the License, or (at your option) any later version.\n *\n * This software is distributed in the hope that it will be useful,\n * but WITHOUT ANY WARRANTY; without even the implied warranty of\n * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU\n * Lesser General Public License for more details.\n *\n * You should have received a copy of the GNU Lesser General Public\n * License along with this software; if not, write to the Free\n * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA\n * 02110-1301 USA, or see the FSF site: http://www.fsf.org.\n */\n\nimport {\n deleteAccessToken,\n deleteExpiryDate,\n deleteRefreshToken,\n deleteTokenType,\n deleteUserId,\n getAccessToken,\n getExpiryDate,\n getRefreshToken,\n getTokenType,\n getUserId,\n setAccessToken,\n setExpiryDate,\n setRefreshToken,\n setTokenType,\n setUserId,\n} from \"./storage.js\";\nimport { UserDetails } from \"@xwiki/cristal-authentication-api\";\nimport axios from \"axios\";\nimport { BrowserWindow, ipcMain, shell } from \"electron\";\n\nconst callbackUrl = \"http://callback/\";\n\nasync function getTokenFromCallbackCode(\n code: string,\n baseUrl: string,\n authenticationBaseUrl: string,\n) {\n const tokenUrl = new URL(`${authenticationBaseUrl}/token`);\n tokenUrl.searchParams.set(\"base_url\", baseUrl);\n tokenUrl.searchParams.set(\"code\", code);\n tokenUrl.searchParams.set(\"redirect_uri\", callbackUrl);\n const response = await axios.get(tokenUrl.toString());\n setTokenType(response.data.token_type, \"oauth2\");\n setAccessToken(response.data.access_token, \"oauth2\");\n setRefreshToken(response.data.refresh_token, \"oauth2\");\n // We apply a safety margin of 10s to the expiration date.\n setExpiryDate(Date.now() + (response.data.expires_in - 10) * 1000, \"oauth2\");\n setUserId(response.data.user_id, \"oauth2\");\n}\n\nfunction initAuth(\n win: BrowserWindow,\n reload: (win: BrowserWindow) => void,\n baseUrl: string,\n authenticationBaseUrl: string,\n) {\n win.webContents.session.webRequest.onBeforeRequest(\n {\n urls: [`${callbackUrl}*`],\n },\n async ({ url }, callback) => {\n if (url.startsWith(authenticationBaseUrl)) {\n // Allow for the redirects from the oidc server to be performed without being blocked.\n callback({ cancel: false });\n } else {\n const parsedURL = new URL(url);\n await getTokenFromCallbackCode(\n parsedURL.searchParams.get(\"code\")!,\n baseUrl,\n authenticationBaseUrl,\n );\n mainWin.show();\n reload(mainWin);\n win?.close();\n }\n },\n );\n}\n\nasync function createWindow(url: string) {\n const win = new BrowserWindow({\n width: 800,\n height: 600,\n webPreferences: {\n nodeIntegration: false,\n },\n });\n\n win.setMenu(null);\n\n await win.loadURL(url);\n\n return win;\n}\n\nlet authWin: BrowserWindow;\nlet mainWin: BrowserWindow;\n\nexport function load(\n browserWindow: BrowserWindow,\n reload: (win: BrowserWindow) => void,\n): void {\n ipcMain.handle(\n \"authentication:nextcloud:loginOauth2\",\n async (\n _event,\n {\n baseUrl,\n authenticationBaseUrl,\n }: {\n baseUrl: string;\n authenticationBaseUrl: string;\n },\n ): Promise<void> => {\n const authorizationUrl = new URL(`${authenticationBaseUrl}/authorize`);\n authorizationUrl.searchParams.set(\"base_url\", baseUrl);\n authorizationUrl.searchParams.set(\"redirect_uri\", callbackUrl);\n // Save the window asking for login (i.e., the main window), before creating\n // a new windows for the oidc web page. Then, hide the main window for the\n // duration of the authentication process.\n mainWin = browserWindow;\n authWin = await createWindow(authorizationUrl.toString());\n\n initAuth(authWin, reload, baseUrl, authenticationBaseUrl);\n mainWin.hide();\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:loginBasic\",\n async (\n _event,\n {\n username,\n password,\n }: {\n username: string;\n password: string;\n },\n ): Promise<void> => {\n setAccessToken(btoa(`${username}:${password}`), \"basic\");\n setTokenType(\"Basic\", \"basic\");\n setUserId(username, \"basic\");\n // We reload the content on successful login.\n reload(browserWindow);\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:loginFlow\",\n async (\n _event,\n {\n baseUrl,\n }: {\n baseUrl: string;\n },\n ): Promise<void> => {\n const loginFlowUrl = `${baseUrl}/index.php/login/v2`;\n\n const loginFlowResponse = await fetch(loginFlowUrl, { method: \"POST\" });\n const jsonLoginFlowResponse: {\n poll: { token: string; endpoint: string };\n login: string;\n } = await loginFlowResponse.json();\n\n shell.openExternal(jsonLoginFlowResponse.login);\n\n // This interval handles polling Nextcloud for the access token.\n // It will return a 404 error until the login process has succeeded.\n const intervalId = setInterval(async () => {\n const response = await fetch(jsonLoginFlowResponse.poll.endpoint, {\n method: \"POST\",\n body: new URLSearchParams({\n token: jsonLoginFlowResponse.poll.token,\n }),\n headers: {\n \"Content-Type\": \"application/x-www-form-urlencoded\",\n },\n });\n if (response.ok) {\n const jsonResponse: {\n loginName: string;\n appPassword: string;\n } = await response.json();\n setAccessToken(\n btoa(`${jsonResponse.loginName}:${jsonResponse.appPassword}`),\n \"login-flow\",\n );\n setTokenType(\"Basic\", \"login-flow\");\n setUserId(jsonResponse.loginName, \"login-flow\");\n clearInterval(intervalId);\n // We reload the content on successful login.\n reload(browserWindow);\n }\n }, 3000);\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:isLoggedIn\",\n async (_event, { mode }: { mode: string }) => {\n const tokenType = getTokenType(mode);\n const accessTokenKey = getAccessToken(mode);\n return tokenType && accessTokenKey;\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:userDetails\",\n async (\n _event,\n { baseUrl, mode }: { baseUrl: string; mode: string },\n ): Promise<UserDetails> => {\n const userId = getUserId(mode);\n return {\n profile: `${baseUrl}/u/${userId}`,\n username: userId,\n name: userId!, // TODO: Find a way to get the display name (CRISTAL-589).\n avatar: `${baseUrl}/avatar/${userId}/64`, // We want the 64x64 avatar.\n };\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:authorizationValue\",\n async (_event, { mode }: { mode: string }) => {\n return {\n tokenType: getTokenType(mode),\n accessToken: getAccessToken(mode),\n };\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:logout\",\n async (_event, { mode }: { mode: string }): Promise<void> => {\n deleteAccessToken(mode);\n deleteTokenType(mode);\n deleteRefreshToken(mode);\n deleteExpiryDate(mode);\n deleteUserId(mode);\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:refreshToken\",\n async (\n _event,\n {\n baseUrl,\n authenticationBaseUrl,\n }: { baseUrl: string; authenticationBaseUrl: string },\n ): Promise<void> => {\n if (Date.now() > getExpiryDate(\"oauth2\")) {\n const refreshUrl = new URL(`${authenticationBaseUrl}/refresh`);\n refreshUrl.searchParams.set(\"base_url\", baseUrl);\n refreshUrl.searchParams.set(\"refresh_token\", getRefreshToken(\"oauth2\"));\n\n const {\n data: {\n access_token: accessToken,\n refresh_token: refreshToken,\n expires_in: expiresIn,\n },\n } = await axios.get(refreshUrl.toString());\n\n setAccessToken(accessToken, \"oauth2\");\n setRefreshToken(refreshToken, \"oauth2\");\n // We apply a safety margin of 10s to the expiration date.\n setExpiryDate(Date.now() + (expiresIn - 10) * 1000, \"oauth2\");\n }\n },\n );\n}\n"],"names":["tokenTypeKey","accessTokenKey","refreshTokenKey","expiryDateKey","userIdKey","schema","storeInstance","Store","set","key","value","mode","get","rm","setTokenType","setAccessToken","setRefreshToken","setExpiryDate","setUserId","getTokenType","getAccessToken","getRefreshToken","getExpiryDate","getUserId","deleteTokenType","deleteAccessToken","deleteRefreshToken","deleteExpiryDate","deleteUserId","fs","require$$0","path","require$$1","pathFile","getElectronPath","executablePath","electron","callbackUrl","getTokenFromCallbackCode","code","baseUrl","authenticationBaseUrl","tokenUrl","response","axios","initAuth","win","reload","url","callback","parsedURL","mainWin","createWindow","BrowserWindow","authWin","load","browserWindow","ipcMain","_event","authorizationUrl","username","password","loginFlowUrl","jsonLoginFlowResponse","shell","intervalId","jsonResponse","tokenType","userId","refreshUrl","accessToken","refreshToken","expiresIn"],"mappings":"wVAsBA,MAAMA,EAAe,YAEfC,EAAiB,cAEjBC,EAAkB,eAElBC,EAAgB,aAEhBC,EAAY,SAEZC,EAAS,CACb,UAAW,CACT,KAAM,QACR,EAEA,YAAa,CACX,KAAM,QACR,EAEA,aAAc,CACZ,KAAM,QACR,EAEA,WAAY,CACV,KAAM,QACR,EAEA,OAAQ,CACN,KAAM,QAAA,CAEV,EAEMC,EAAuB,IAAIC,EAAM,CACrC,KAAM,gCACN,OAAAF,CAEF,CAAC,EAED,SAASG,EAAOC,EAAaC,EAAUC,EAAc,CAEnDL,EAAc,IAAI,GAAGG,CAAG,IAAIE,CAAI,GAAID,CAAK,CAC3C,CAEA,SAASE,EAAOH,EAAaE,EAAiB,CAE5C,OAAOL,EAAc,IAAI,GAAGG,CAAG,IAAIE,CAAI,EAAE,CAC3C,CAEA,SAASE,EAAGJ,EAAaE,EAAc,CAErCL,EAAc,OAAO,GAAGG,CAAG,IAAIE,CAAI,EAAE,CACvC,CAEA,SAASG,EAAaJ,EAAeC,EAAoB,CACnDH,EAAAR,EAAcU,EAAOC,CAAI,CAC/B,CAEA,SAASI,EAAeL,EAAeC,EAAoB,CACrDH,EAAAP,EAAgBS,EAAOC,CAAI,CACjC,CAEA,SAASK,EAAgBN,EAAeC,EAAoB,CACtDH,EAAAN,EAAiBQ,EAAOC,CAAI,CAClC,CAEA,SAASM,EAAcP,EAAeC,EAAoB,CACpDH,EAAAL,EAAeO,EAAOC,CAAI,CAChC,CAEA,SAASO,EAAUR,EAAeC,EAAoB,CAChDH,EAAAJ,EAAWM,EAAOC,CAAI,CAC5B,CAEA,SAASQ,EAAaR,EAAsB,CACnC,OAAAC,EAAIZ,EAAcW,CAAI,CAC/B,CAEA,SAASS,EAAeT,EAAsB,CACrC,OAAAC,EAAIX,EAAgBU,CAAI,CACjC,CAEA,SAASU,EAAgBV,EAAsB,CACtC,OAAAC,EAAIV,EAAiBS,CAAI,CAClC,CAEA,SAASW,EAAcX,EAAsB,CACpC,OAAAC,EAAIT,EAAeQ,CAAI,CAChC,CAEA,SAASY,EAAUZ,EAAsB,CAChC,OAAAC,EAAIR,EAAWO,CAAI,CAC5B,CAEA,SAASa,EAAgBb,EAAoB,CAC3CE,EAAGb,EAAcW,CAAI,CACvB,CAEA,SAASc,EAAkBd,EAAoB,CAC7CE,EAAGZ,EAAgBU,CAAI,CACzB,CAEA,SAASe,EAAmBf,EAAoB,CAC9CE,EAAGX,EAAiBS,CAAI,CAC1B,CAEA,SAASgB,EAAiBhB,EAAoB,CAC5CE,EAAGV,EAAeQ,CAAI,CACxB,CAEA,SAASiB,EAAajB,EAAoB,CACxCE,EAAGT,EAAWO,CAAI,CACpB,ukBCrIe,CAAA,iFCAf,MAAMkB,EAAKC,EACLC,EAAOC,EAEPC,EAAWF,EAAK,KAAK,UAAW,UAAU,EAEhD,SAASG,GAAmB,CAC1B,IAAIC,EAIJ,GAHIN,EAAG,WAAWI,CAAQ,IACxBE,EAAiBN,EAAG,aAAaI,EAAU,OAAO,GAEhD,QAAQ,IAAI,4BACd,OAAOF,EAAK,KAAK,QAAQ,IAAI,4BAA6BI,GAAkB,UAAU,EAExF,GAAIA,EACF,OAAOJ,EAAK,KAAK,UAAW,OAAQI,CAAc,EAElD,MAAM,IAAI,MAAM,oGAAoG,CAExH,CAEc,OAAAC,EAAGF,EAAiB,cCqBlC,MAAMG,EAAc,mBAEpB,eAAeC,EACbC,EACAC,EACAC,EACA,CACA,MAAMC,EAAW,IAAI,IAAI,GAAGD,CAAqB,QAAQ,EAChDC,EAAA,aAAa,IAAI,WAAYF,CAAO,EACpCE,EAAA,aAAa,IAAI,OAAQH,CAAI,EAC7BG,EAAA,aAAa,IAAI,eAAgBL,CAAW,EACrD,MAAMM,EAAW,MAAMC,EAAM,IAAIF,EAAS,UAAU,EACvC5B,EAAA6B,EAAS,KAAK,WAAY,QAAQ,EAChC5B,EAAA4B,EAAS,KAAK,aAAc,QAAQ,EACnC3B,EAAA2B,EAAS,KAAK,cAAe,QAAQ,EAEvC1B,EAAA,KAAK,OAAS0B,EAAS,KAAK,WAAa,IAAM,IAAM,QAAQ,EACjEzB,EAAAyB,EAAS,KAAK,QAAS,QAAQ,CAC3C,CAEA,SAASE,EACPC,EACAC,EACAP,EACAC,EACA,CACIK,EAAA,YAAY,QAAQ,WAAW,gBACjC,CACE,KAAM,CAAC,GAAGT,CAAW,GAAG,CAC1B,EACA,MAAO,CAAE,IAAAW,CAAI,EAAGC,IAAa,CACvB,GAAAD,EAAI,WAAWP,CAAqB,EAE7BQ,EAAA,CAAE,OAAQ,GAAO,MACrB,CACC,MAAAC,EAAY,IAAI,IAAIF,CAAG,EACvB,MAAAV,EACJY,EAAU,aAAa,IAAI,MAAM,EACjCV,EACAC,CACF,EACAU,EAAQ,KAAK,EACbJ,EAAOI,CAAO,EACdL,GAAK,MAAM,CAAA,CACb,CAEJ,CACF,CAEA,eAAeM,EAAaJ,EAAa,CACjC,MAAAF,EAAM,IAAIO,gBAAc,CAC5B,MAAO,IACP,OAAQ,IACR,eAAgB,CACd,gBAAiB,EAAA,CACnB,CACD,EAED,OAAAP,EAAI,QAAQ,IAAI,EAEV,MAAAA,EAAI,QAAQE,CAAG,EAEdF,CACT,CAEA,IAAIQ,EACAH,EAEY,SAAAI,EACdC,EACAT,EACM,CACEU,EAAAA,QAAA,OACN,uCACA,MACEC,EACA,CACE,QAAAlB,EACA,sBAAAC,CAAA,IAKgB,CAClB,MAAMkB,EAAmB,IAAI,IAAI,GAAGlB,CAAqB,YAAY,EACpDkB,EAAA,aAAa,IAAI,WAAYnB,CAAO,EACpCmB,EAAA,aAAa,IAAI,eAAgBtB,CAAW,EAInDc,EAAAK,EACVF,EAAU,MAAMF,EAAaO,EAAiB,SAAA,CAAU,EAE/Cd,EAAAS,EAASP,EAAQP,EAASC,CAAqB,EACxDU,EAAQ,KAAK,CAAA,CAEjB,EAEQM,EAAAA,QAAA,OACN,sCACA,MACEC,EACA,CACE,SAAAE,EACA,SAAAC,CAAA,IAKgB,CAClB9C,EAAe,KAAK,GAAG6C,CAAQ,IAAIC,CAAQ,EAAE,EAAG,OAAO,EACvD/C,EAAa,QAAS,OAAO,EAC7BI,EAAU0C,EAAU,OAAO,EAE3Bb,EAAOS,CAAa,CAAA,CAExB,EAEQC,EAAAA,QAAA,OACN,qCACA,MACEC,EACA,CACE,QAAAlB,CAAA,IAIgB,CACZ,MAAAsB,EAAe,GAAGtB,CAAO,sBAGzBuB,EAGF,MAJsB,MAAM,MAAMD,EAAc,CAAE,OAAQ,OAAQ,GAI1C,KAAK,EAE3BE,QAAA,aAAaD,EAAsB,KAAK,EAIxC,MAAAE,EAAa,YAAY,SAAY,CACzC,MAAMtB,EAAW,MAAM,MAAMoB,EAAsB,KAAK,SAAU,CAChE,OAAQ,OACR,KAAM,IAAI,gBAAgB,CACxB,MAAOA,EAAsB,KAAK,KAAA,CACnC,EACD,QAAS,CACP,eAAgB,mCAAA,CAClB,CACD,EACD,GAAIpB,EAAS,GAAI,CACT,MAAAuB,EAGF,MAAMvB,EAAS,KAAK,EACxB5B,EACE,KAAK,GAAGmD,EAAa,SAAS,IAAIA,EAAa,WAAW,EAAE,EAC5D,YACF,EACApD,EAAa,QAAS,YAAY,EACxBI,EAAAgD,EAAa,UAAW,YAAY,EAC9C,cAAcD,CAAU,EAExBlB,EAAOS,CAAa,CAAA,GAErB,GAAI,CAAA,CAEX,EAEQC,EAAAA,QAAA,OACN,sCACA,MAAOC,EAAQ,CAAE,KAAA/C,KAA6B,CACtC,MAAAwD,EAAYhD,EAAaR,CAAI,EAC7BV,EAAiBmB,EAAeT,CAAI,EAC1C,OAAOwD,GAAalE,CAAA,CAExB,EAEQwD,EAAAA,QAAA,OACN,uCACA,MACEC,EACA,CAAE,QAAAlB,EAAS,KAAA7B,KACc,CACnB,MAAAyD,EAAS7C,EAAUZ,CAAI,EACtB,MAAA,CACL,QAAS,GAAG6B,CAAO,MAAM4B,CAAM,GAC/B,SAAUA,EACV,KAAMA,EACN,OAAQ,GAAG5B,CAAO,WAAW4B,CAAM,KACrC,CAAA,CAEJ,EAEQX,EAAAA,QAAA,OACN,8CACA,MAAOC,EAAQ,CAAE,KAAA/C,MACR,CACL,UAAWQ,EAAaR,CAAI,EAC5B,YAAaS,EAAeT,CAAI,CAClC,EAEJ,EAEQ8C,EAAAA,QAAA,OACN,kCACA,MAAOC,EAAQ,CAAE,KAAA/C,KAA4C,CAC3Dc,EAAkBd,CAAI,EACtBa,EAAgBb,CAAI,EACpBe,EAAmBf,CAAI,EACvBgB,EAAiBhB,CAAI,EACrBiB,EAAajB,CAAI,CAAA,CAErB,EAEQ8C,EAAAA,QAAA,OACN,wCACA,MACEC,EACA,CACE,QAAAlB,EACA,sBAAAC,CAAA,IAEgB,CAClB,GAAI,KAAK,IAAA,EAAQnB,EAAc,QAAQ,EAAG,CACxC,MAAM+C,EAAa,IAAI,IAAI,GAAG5B,CAAqB,UAAU,EAClD4B,EAAA,aAAa,IAAI,WAAY7B,CAAO,EAC/C6B,EAAW,aAAa,IAAI,gBAAiBhD,EAAgB,QAAQ,CAAC,EAEhE,KAAA,CACJ,KAAM,CACJ,aAAciD,EACd,cAAeC,EACf,WAAYC,CAAA,GAEZ,MAAM5B,EAAM,IAAIyB,EAAW,UAAU,EAEzCtD,EAAeuD,EAAa,QAAQ,EACpCtD,EAAgBuD,EAAc,QAAQ,EAEtCtD,EAAc,KAAK,IAAI,GAAKuD,EAAY,IAAM,IAAM,QAAQ,CAAA,CAC9D,CAEJ,CACF","x_google_ignoreList":[2]}
1
+ {"version":3,"file":"index.umd.js","sources":["../src/storage.ts","../src/index.ts"],"sourcesContent":["/**\n * See the LICENSE file distributed with this work for additional\n * information regarding copyright ownership.\n *\n * This is free software; you can redistribute it and/or modify it\n * under the terms of the GNU Lesser General Public License as\n * published by the Free Software Foundation; either version 2.1 of\n * the License, or (at your option) any later version.\n *\n * This software is distributed in the hope that it will be useful,\n * but WITHOUT ANY WARRANTY; without even the implied warranty of\n * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU\n * Lesser General Public License for more details.\n *\n * You should have received a copy of the GNU Lesser General Public\n * License along with this software; if not, write to the Free\n * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA\n * 02110-1301 USA, or see the FSF site: http://www.fsf.org.\n */\n\nimport Store from \"electron-store\";\n\nconst tokenTypeKey = \"tokenType\";\n\nconst accessTokenKey = \"accessToken\";\n\nconst refreshTokenKey = \"refreshToken\";\n\nconst expiryDateKey = \"expiryDate\";\n\nconst userIdKey = \"userId\";\n\nconst schema = {\n tokenType: {\n type: \"string\",\n },\n\n accessToken: {\n type: \"string\",\n },\n\n refreshToken: {\n type: \"string\",\n },\n\n expiryDate: {\n type: \"number\",\n },\n\n userId: {\n type: \"string\",\n },\n};\n\nconst storeInstance: Store = new Store({\n name: \"authentication-nextcloud-main\",\n schema,\n // TODO: add encryption key\n});\n\nfunction set<T>(key: string, value: T, mode: string) {\n // @ts-expect-error type resolution failing because of electron-store library bug\n storeInstance.set(`${key}-${mode}`, value);\n}\n\nfunction get<T>(key: string, mode: string): T {\n // @ts-expect-error type resolution failing because of electron-store library bug\n return storeInstance.get(`${key}-${mode}`) as T;\n}\n\nfunction rm(key: string, mode: string) {\n // @ts-expect-error type resolution failing because of electron-store library bug\n storeInstance.delete(`${key}-${mode}`);\n}\n\nfunction setTokenType(value: string, mode: string): void {\n set(tokenTypeKey, value, mode);\n}\n\nfunction setAccessToken(value: string, mode: string): void {\n set(accessTokenKey, value, mode);\n}\n\nfunction setRefreshToken(value: string, mode: string): void {\n set(refreshTokenKey, value, mode);\n}\n\nfunction setExpiryDate(value: number, mode: string): void {\n set(expiryDateKey, value, mode);\n}\n\nfunction setUserId(value: string, mode: string): void {\n set(userIdKey, value, mode);\n}\n\nfunction getTokenType(mode: string): string {\n return get(tokenTypeKey, mode);\n}\n\nfunction getAccessToken(mode: string): string {\n return get(accessTokenKey, mode);\n}\n\nfunction getRefreshToken(mode: string): string {\n return get(refreshTokenKey, mode);\n}\n\nfunction getExpiryDate(mode: string): number {\n return get(expiryDateKey, mode);\n}\n\nfunction getUserId(mode: string): string {\n return get(userIdKey, mode);\n}\n\nfunction deleteTokenType(mode: string): void {\n rm(tokenTypeKey, mode);\n}\n\nfunction deleteAccessToken(mode: string): void {\n rm(accessTokenKey, mode);\n}\n\nfunction deleteRefreshToken(mode: string): void {\n rm(refreshTokenKey, mode);\n}\n\nfunction deleteExpiryDate(mode: string): void {\n rm(expiryDateKey, mode);\n}\n\nfunction deleteUserId(mode: string): void {\n rm(userIdKey, mode);\n}\n\nexport {\n deleteAccessToken,\n deleteExpiryDate,\n deleteRefreshToken,\n deleteTokenType,\n deleteUserId,\n getAccessToken,\n getExpiryDate,\n getRefreshToken,\n getTokenType,\n getUserId,\n setAccessToken,\n setExpiryDate,\n setRefreshToken,\n setTokenType,\n setUserId,\n};\n","/**\n * See the LICENSE file distributed with this work for additional\n * information regarding copyright ownership.\n *\n * This is free software; you can redistribute it and/or modify it\n * under the terms of the GNU Lesser General Public License as\n * published by the Free Software Foundation; either version 2.1 of\n * the License, or (at your option) any later version.\n *\n * This software is distributed in the hope that it will be useful,\n * but WITHOUT ANY WARRANTY; without even the implied warranty of\n * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU\n * Lesser General Public License for more details.\n *\n * You should have received a copy of the GNU Lesser General Public\n * License along with this software; if not, write to the Free\n * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA\n * 02110-1301 USA, or see the FSF site: http://www.fsf.org.\n */\n\nimport {\n deleteAccessToken,\n deleteExpiryDate,\n deleteRefreshToken,\n deleteTokenType,\n deleteUserId,\n getAccessToken,\n getExpiryDate,\n getRefreshToken,\n getTokenType,\n getUserId,\n setAccessToken,\n setExpiryDate,\n setRefreshToken,\n setTokenType,\n setUserId,\n} from \"./storage.js\";\nimport { UserDetails } from \"@xwiki/cristal-authentication-api\";\nimport axios from \"axios\";\nimport { BrowserWindow, ipcMain, shell } from \"electron\";\n\nconst callbackUrl = \"http://callback/\";\n\nasync function getTokenFromCallbackCode(\n code: string,\n baseUrl: string,\n authenticationBaseUrl: string,\n) {\n const tokenUrl = new URL(`${authenticationBaseUrl}/token`);\n tokenUrl.searchParams.set(\"base_url\", baseUrl);\n tokenUrl.searchParams.set(\"code\", code);\n tokenUrl.searchParams.set(\"redirect_uri\", callbackUrl);\n const response = await axios.get(tokenUrl.toString());\n setTokenType(response.data.token_type, \"oauth2\");\n setAccessToken(response.data.access_token, \"oauth2\");\n setRefreshToken(response.data.refresh_token, \"oauth2\");\n // We apply a safety margin of 10s to the expiration date.\n setExpiryDate(Date.now() + (response.data.expires_in - 10) * 1000, \"oauth2\");\n setUserId(response.data.user_id, \"oauth2\");\n}\n\nfunction initAuth(\n win: BrowserWindow,\n reload: (win: BrowserWindow) => void,\n baseUrl: string,\n authenticationBaseUrl: string,\n) {\n win.webContents.session.webRequest.onBeforeRequest(\n {\n urls: [`${callbackUrl}*`],\n },\n async ({ url }, callback) => {\n if (url.startsWith(authenticationBaseUrl)) {\n // Allow for the redirects from the oidc server to be performed without being blocked.\n callback({ cancel: false });\n } else {\n const parsedURL = new URL(url);\n await getTokenFromCallbackCode(\n parsedURL.searchParams.get(\"code\")!,\n baseUrl,\n authenticationBaseUrl,\n );\n mainWin.show();\n reload(mainWin);\n win?.close();\n }\n },\n );\n}\n\nasync function createWindow(url: string) {\n const win = new BrowserWindow({\n width: 800,\n height: 600,\n webPreferences: {\n nodeIntegration: false,\n },\n });\n\n win.setMenu(null);\n\n await win.loadURL(url);\n\n return win;\n}\n\nlet authWin: BrowserWindow;\nlet mainWin: BrowserWindow;\n\nexport function load(\n browserWindow: BrowserWindow,\n reload: (win: BrowserWindow) => void,\n): void {\n ipcMain.handle(\n \"authentication:nextcloud:loginOauth2\",\n async (\n _event,\n {\n baseUrl,\n authenticationBaseUrl,\n }: {\n baseUrl: string;\n authenticationBaseUrl: string;\n },\n ): Promise<void> => {\n const authorizationUrl = new URL(`${authenticationBaseUrl}/authorize`);\n authorizationUrl.searchParams.set(\"base_url\", baseUrl);\n authorizationUrl.searchParams.set(\"redirect_uri\", callbackUrl);\n // Save the window asking for login (i.e., the main window), before creating\n // a new windows for the oidc web page. Then, hide the main window for the\n // duration of the authentication process.\n mainWin = browserWindow;\n authWin = await createWindow(authorizationUrl.toString());\n\n initAuth(authWin, reload, baseUrl, authenticationBaseUrl);\n mainWin.hide();\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:loginBasic\",\n async (\n _event,\n {\n username,\n password,\n }: {\n username: string;\n password: string;\n },\n ): Promise<void> => {\n setAccessToken(btoa(`${username}:${password}`), \"basic\");\n setTokenType(\"Basic\", \"basic\");\n setUserId(username, \"basic\");\n // We reload the content on successful login.\n reload(browserWindow);\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:loginFlow\",\n async (\n _event,\n {\n baseUrl,\n }: {\n baseUrl: string;\n },\n ): Promise<void> => {\n const loginFlowUrl = `${baseUrl}/index.php/login/v2`;\n\n const loginFlowResponse = await fetch(loginFlowUrl, { method: \"POST\" });\n const jsonLoginFlowResponse: {\n poll: { token: string; endpoint: string };\n login: string;\n } = await loginFlowResponse.json();\n\n shell.openExternal(jsonLoginFlowResponse.login);\n\n // This interval handles polling Nextcloud for the access token.\n // It will return a 404 error until the login process has succeeded.\n const intervalId = setInterval(async () => {\n const response = await fetch(jsonLoginFlowResponse.poll.endpoint, {\n method: \"POST\",\n body: new URLSearchParams({\n token: jsonLoginFlowResponse.poll.token,\n }),\n headers: {\n \"Content-Type\": \"application/x-www-form-urlencoded\",\n },\n });\n if (response.ok) {\n const jsonResponse: {\n loginName: string;\n appPassword: string;\n } = await response.json();\n setAccessToken(\n btoa(`${jsonResponse.loginName}:${jsonResponse.appPassword}`),\n \"login-flow\",\n );\n setTokenType(\"Basic\", \"login-flow\");\n setUserId(jsonResponse.loginName, \"login-flow\");\n clearInterval(intervalId);\n // We reload the content on successful login.\n reload(browserWindow);\n }\n }, 3000);\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:isLoggedIn\",\n async (_event, { mode }: { mode: string }) => {\n const tokenType = getTokenType(mode);\n const accessTokenKey = getAccessToken(mode);\n return tokenType && accessTokenKey;\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:userDetails\",\n async (\n _event,\n { baseUrl, mode }: { baseUrl: string; mode: string },\n ): Promise<UserDetails> => {\n const userId = getUserId(mode);\n return {\n profile: `${baseUrl}/u/${userId}`,\n username: userId,\n name: userId!, // TODO: Find a way to get the display name (CRISTAL-589).\n avatar: `${baseUrl}/avatar/${userId}/64`, // We want the 64x64 avatar.\n };\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:authorizationValue\",\n async (_event, { mode }: { mode: string }) => {\n return {\n tokenType: getTokenType(mode),\n accessToken: getAccessToken(mode),\n };\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:logout\",\n async (_event, { mode }: { mode: string }): Promise<void> => {\n deleteAccessToken(mode);\n deleteTokenType(mode);\n deleteRefreshToken(mode);\n deleteExpiryDate(mode);\n deleteUserId(mode);\n },\n );\n\n ipcMain.handle(\n \"authentication:nextcloud:refreshToken\",\n async (\n _event,\n {\n baseUrl,\n authenticationBaseUrl,\n }: { baseUrl: string; authenticationBaseUrl: string },\n ): Promise<void> => {\n if (Date.now() > getExpiryDate(\"oauth2\")) {\n const refreshUrl = new URL(`${authenticationBaseUrl}/refresh`);\n refreshUrl.searchParams.set(\"base_url\", baseUrl);\n refreshUrl.searchParams.set(\"refresh_token\", getRefreshToken(\"oauth2\"));\n\n const {\n data: {\n access_token: accessToken,\n refresh_token: refreshToken,\n expires_in: expiresIn,\n },\n } = await axios.get(refreshUrl.toString());\n\n setAccessToken(accessToken, \"oauth2\");\n setRefreshToken(refreshToken, \"oauth2\");\n // We apply a safety margin of 10s to the expiration date.\n setExpiryDate(Date.now() + (expiresIn - 10) * 1000, \"oauth2\");\n }\n },\n );\n}\n"],"names":["tokenTypeKey","accessTokenKey","refreshTokenKey","expiryDateKey","userIdKey","schema","storeInstance","Store","set","key","value","mode","get","rm","setTokenType","setAccessToken","setRefreshToken","setExpiryDate","setUserId","getTokenType","getAccessToken","getRefreshToken","getExpiryDate","getUserId","deleteTokenType","deleteAccessToken","deleteRefreshToken","deleteExpiryDate","deleteUserId","callbackUrl","getTokenFromCallbackCode","code","baseUrl","authenticationBaseUrl","tokenUrl","response","axios","initAuth","win","reload","url","callback","parsedURL","mainWin","createWindow","BrowserWindow","authWin","load","browserWindow","ipcMain","_event","authorizationUrl","username","password","loginFlowUrl","jsonLoginFlowResponse","shell","intervalId","jsonResponse","tokenType","userId","refreshUrl","accessToken","refreshToken","expiresIn"],"mappings":"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"}
package/dist/storage.d.ts CHANGED
@@ -1,3 +1,22 @@
1
+ /**
2
+ * See the LICENSE file distributed with this work for additional
3
+ * information regarding copyright ownership.
4
+ *
5
+ * This is free software; you can redistribute it and/or modify it
6
+ * under the terms of the GNU Lesser General Public License as
7
+ * published by the Free Software Foundation; either version 2.1 of
8
+ * the License, or (at your option) any later version.
9
+ *
10
+ * This software is distributed in the hope that it will be useful,
11
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
12
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13
+ * Lesser General Public License for more details.
14
+ *
15
+ * You should have received a copy of the GNU Lesser General Public
16
+ * License along with this software; if not, write to the Free
17
+ * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
18
+ * 02110-1301 USA, or see the FSF site: http://www.fsf.org.
19
+ */
1
20
  declare function setTokenType(value: string, mode: string): void;
2
21
  declare function setAccessToken(value: string, mode: string): void;
3
22
  declare function setRefreshToken(value: string, mode: string): void;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@xwiki/cristal-electron-authentication-nextcloud-main",
3
- "version": "0.20.0",
3
+ "version": "0.21.1",
4
4
  "license": "LGPL 2.1",
5
5
  "author": "XWiki Org Community <contact@xwiki.org>",
6
6
  "homepage": "https://cristal.xwiki.org/",
@@ -27,17 +27,17 @@
27
27
  },
28
28
  "main": "./dist/index.es.js",
29
29
  "dependencies": {
30
- "axios": "1.10.0",
30
+ "axios": "1.11.0",
31
31
  "electron-store": "10.1.0",
32
- "@xwiki/cristal-authentication-api": "0.20.0"
32
+ "@xwiki/cristal-authentication-api": "0.21.1"
33
33
  },
34
34
  "peerDependencies": {
35
- "electron": "37.1.0"
35
+ "electron": "37.2.5"
36
36
  },
37
37
  "devDependencies": {
38
- "typescript": "5.8.3",
39
- "vite": "7.0.0",
40
- "@xwiki/cristal-dev-config": "0.20.0"
38
+ "typescript": "5.9.2",
39
+ "vite": "7.0.6",
40
+ "@xwiki/cristal-dev-config": "0.21.1"
41
41
  },
42
42
  "scripts": {
43
43
  "build": "tsc --project tsconfig.json && vite build",
package/src/index.ts CHANGED
@@ -1,4 +1,4 @@
1
- /*
1
+ /**
2
2
  * See the LICENSE file distributed with this work for additional
3
3
  * information regarding copyright ownership.
4
4
  *
package/src/storage.ts CHANGED
@@ -1,4 +1,4 @@
1
- /*
1
+ /**
2
2
  * See the LICENSE file distributed with this work for additional
3
3
  * information regarding copyright ownership.
4
4
  *
package/vite.config.ts CHANGED
@@ -1,4 +1,4 @@
1
- /*
1
+ /**
2
2
  * See the LICENSE file distributed with this work for additional
3
3
  * information regarding copyright ownership.
4
4
  *
package/vitest.config.ts CHANGED
@@ -1,4 +1,4 @@
1
- /*
1
+ /**
2
2
  * See the LICENSE file distributed with this work for additional
3
3
  * information regarding copyright ownership.
4
4
  *