@xwiki/cristal-electron-authentication-nextcloud-main 0.24.0 → 0.25.0
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/dist/index.es.js.map +1 -1
- package/dist/index.umd.js.map +1 -1
- package/package.json +7 -7
- package/src/index.ts +1 -1
- package/CHANGELOG.md +0 -100
package/dist/index.es.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
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\ntype StoreType = {\n tokenType: string;\n accessToken: string;\n refreshToken: string;\n expiryDate: number;\n userId: string;\n};\n\nconst storeInstance: Store<StoreType> = new Store<StoreType>({\n name: \"authentication-nextcloud-main\",\n schema,\n // TODO: add encryption key\n});\n\nfunction setTokenType(value: string, mode: string): void {\n storeInstance.set(`${tokenTypeKey}-${mode}`, value);\n}\n\nfunction setAccessToken(value: string, mode: string): void {\n storeInstance.set(`${accessTokenKey}-${mode}`, value);\n}\n\nfunction setRefreshToken(value: string, mode: string): void {\n storeInstance.set(`${refreshTokenKey}-${mode}`, value);\n}\n\nfunction setExpiryDate(value: number, mode: string): void {\n storeInstance.set(`${expiryDateKey}-${mode}`, value);\n}\n\nfunction setUserId(value: string, mode: string): void {\n storeInstance.set(`${userIdKey}-${mode}`, value);\n}\n\nfunction getTokenType(mode: string): string {\n return storeInstance.get(`${tokenTypeKey}-${mode}`);\n}\n\nfunction getAccessToken(mode: string): string {\n return storeInstance.get(`${accessTokenKey}-${mode}`);\n}\n\nfunction getRefreshToken(mode: string): string {\n return storeInstance.get(`${refreshTokenKey}-${mode}`);\n}\n\nfunction getExpiryDate(mode: string): number {\n return storeInstance.get(`${expiryDateKey}-${mode}`);\n}\n\nfunction getUserId(mode: string): string {\n return storeInstance.get(`${userIdKey}-${mode}`);\n}\n\nfunction deleteTokenType(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${tokenTypeKey}-${mode}`);\n}\n\nfunction deleteAccessToken(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${accessTokenKey}-${mode}`);\n}\n\nfunction deleteRefreshToken(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${refreshTokenKey}-${mode}`);\n}\n\nfunction deleteExpiryDate(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${expiryDateKey}-${mode}`);\n}\n\nfunction deleteUserId(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${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 axios from \"axios\";\nimport { BrowserWindow, ipcMain, shell } from \"electron\";\nimport type { UserDetails } from \"@xwiki/cristal-authentication-api\";\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\n/**\n * @param browserWindow - the browser window to use for the authentication process\n * @param reload - the reload function\n *\n * @since 0.16\n * @beta\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","setTokenType","value","mode","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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"}
|
|
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\ntype StoreType = {\n tokenType: string;\n accessToken: string;\n refreshToken: string;\n expiryDate: number;\n userId: string;\n};\n\nconst storeInstance: Store<StoreType> = new Store<StoreType>({\n name: \"authentication-nextcloud-main\",\n schema,\n // TODO: add encryption key\n});\n\nfunction setTokenType(value: string, mode: string): void {\n storeInstance.set(`${tokenTypeKey}-${mode}`, value);\n}\n\nfunction setAccessToken(value: string, mode: string): void {\n storeInstance.set(`${accessTokenKey}-${mode}`, value);\n}\n\nfunction setRefreshToken(value: string, mode: string): void {\n storeInstance.set(`${refreshTokenKey}-${mode}`, value);\n}\n\nfunction setExpiryDate(value: number, mode: string): void {\n storeInstance.set(`${expiryDateKey}-${mode}`, value);\n}\n\nfunction setUserId(value: string, mode: string): void {\n storeInstance.set(`${userIdKey}-${mode}`, value);\n}\n\nfunction getTokenType(mode: string): string {\n return storeInstance.get(`${tokenTypeKey}-${mode}`);\n}\n\nfunction getAccessToken(mode: string): string {\n return storeInstance.get(`${accessTokenKey}-${mode}`);\n}\n\nfunction getRefreshToken(mode: string): string {\n return storeInstance.get(`${refreshTokenKey}-${mode}`);\n}\n\nfunction getExpiryDate(mode: string): number {\n return storeInstance.get(`${expiryDateKey}-${mode}`);\n}\n\nfunction getUserId(mode: string): string {\n return storeInstance.get(`${userIdKey}-${mode}`);\n}\n\nfunction deleteTokenType(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${tokenTypeKey}-${mode}`);\n}\n\nfunction deleteAccessToken(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${accessTokenKey}-${mode}`);\n}\n\nfunction deleteRefreshToken(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${refreshTokenKey}-${mode}`);\n}\n\nfunction deleteExpiryDate(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${expiryDateKey}-${mode}`);\n}\n\nfunction deleteUserId(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${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 axios from \"axios\";\nimport { BrowserWindow, ipcMain, shell } from \"electron\";\nimport type { UserDetails } from \"@xwiki/platform-authentication-api\";\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\n/**\n * @param browserWindow - the browser window to use for the authentication process\n * @param reload - the reload function\n *\n * @since 0.16\n * @beta\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","setTokenType","value","mode","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.map
CHANGED
|
@@ -1 +1 @@
|
|
|
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\ntype StoreType = {\n tokenType: string;\n accessToken: string;\n refreshToken: string;\n expiryDate: number;\n userId: string;\n};\n\nconst storeInstance: Store<StoreType> = new Store<StoreType>({\n name: \"authentication-nextcloud-main\",\n schema,\n // TODO: add encryption key\n});\n\nfunction setTokenType(value: string, mode: string): void {\n storeInstance.set(`${tokenTypeKey}-${mode}`, value);\n}\n\nfunction setAccessToken(value: string, mode: string): void {\n storeInstance.set(`${accessTokenKey}-${mode}`, value);\n}\n\nfunction setRefreshToken(value: string, mode: string): void {\n storeInstance.set(`${refreshTokenKey}-${mode}`, value);\n}\n\nfunction setExpiryDate(value: number, mode: string): void {\n storeInstance.set(`${expiryDateKey}-${mode}`, value);\n}\n\nfunction setUserId(value: string, mode: string): void {\n storeInstance.set(`${userIdKey}-${mode}`, value);\n}\n\nfunction getTokenType(mode: string): string {\n return storeInstance.get(`${tokenTypeKey}-${mode}`);\n}\n\nfunction getAccessToken(mode: string): string {\n return storeInstance.get(`${accessTokenKey}-${mode}`);\n}\n\nfunction getRefreshToken(mode: string): string {\n return storeInstance.get(`${refreshTokenKey}-${mode}`);\n}\n\nfunction getExpiryDate(mode: string): number {\n return storeInstance.get(`${expiryDateKey}-${mode}`);\n}\n\nfunction getUserId(mode: string): string {\n return storeInstance.get(`${userIdKey}-${mode}`);\n}\n\nfunction deleteTokenType(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${tokenTypeKey}-${mode}`);\n}\n\nfunction deleteAccessToken(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${accessTokenKey}-${mode}`);\n}\n\nfunction deleteRefreshToken(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${refreshTokenKey}-${mode}`);\n}\n\nfunction deleteExpiryDate(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${expiryDateKey}-${mode}`);\n}\n\nfunction deleteUserId(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${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 axios from \"axios\";\nimport { BrowserWindow, ipcMain, shell } from \"electron\";\nimport type { UserDetails } from \"@xwiki/cristal-authentication-api\";\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\n/**\n * @param browserWindow - the browser window to use for the authentication process\n * @param reload - the reload function\n *\n * @since 0.16\n * @beta\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","setTokenType","value","mode","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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"}
|
|
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\ntype StoreType = {\n tokenType: string;\n accessToken: string;\n refreshToken: string;\n expiryDate: number;\n userId: string;\n};\n\nconst storeInstance: Store<StoreType> = new Store<StoreType>({\n name: \"authentication-nextcloud-main\",\n schema,\n // TODO: add encryption key\n});\n\nfunction setTokenType(value: string, mode: string): void {\n storeInstance.set(`${tokenTypeKey}-${mode}`, value);\n}\n\nfunction setAccessToken(value: string, mode: string): void {\n storeInstance.set(`${accessTokenKey}-${mode}`, value);\n}\n\nfunction setRefreshToken(value: string, mode: string): void {\n storeInstance.set(`${refreshTokenKey}-${mode}`, value);\n}\n\nfunction setExpiryDate(value: number, mode: string): void {\n storeInstance.set(`${expiryDateKey}-${mode}`, value);\n}\n\nfunction setUserId(value: string, mode: string): void {\n storeInstance.set(`${userIdKey}-${mode}`, value);\n}\n\nfunction getTokenType(mode: string): string {\n return storeInstance.get(`${tokenTypeKey}-${mode}`);\n}\n\nfunction getAccessToken(mode: string): string {\n return storeInstance.get(`${accessTokenKey}-${mode}`);\n}\n\nfunction getRefreshToken(mode: string): string {\n return storeInstance.get(`${refreshTokenKey}-${mode}`);\n}\n\nfunction getExpiryDate(mode: string): number {\n return storeInstance.get(`${expiryDateKey}-${mode}`);\n}\n\nfunction getUserId(mode: string): string {\n return storeInstance.get(`${userIdKey}-${mode}`);\n}\n\nfunction deleteTokenType(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${tokenTypeKey}-${mode}`);\n}\n\nfunction deleteAccessToken(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${accessTokenKey}-${mode}`);\n}\n\nfunction deleteRefreshToken(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${refreshTokenKey}-${mode}`);\n}\n\nfunction deleteExpiryDate(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${expiryDateKey}-${mode}`);\n}\n\nfunction deleteUserId(mode: string): void {\n // @ts-expect-error resolution failing because of electron-store library bug\n storeInstance.delete(`${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 axios from \"axios\";\nimport { BrowserWindow, ipcMain, shell } from \"electron\";\nimport type { UserDetails } from \"@xwiki/platform-authentication-api\";\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\n/**\n * @param browserWindow - the browser window to use for the authentication process\n * @param reload - the reload function\n *\n * @since 0.16\n * @beta\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","setTokenType","value","mode","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/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@xwiki/cristal-electron-authentication-nextcloud-main",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.25.0",
|
|
4
4
|
"license": "LGPL 2.1",
|
|
5
5
|
"author": "XWiki Org Community <contact@xwiki.org>",
|
|
6
6
|
"homepage": "https://cristal.xwiki.org/",
|
|
@@ -28,17 +28,17 @@
|
|
|
28
28
|
"main": "./dist/index.es.js",
|
|
29
29
|
"types": "./dist/index.d.ts",
|
|
30
30
|
"dependencies": {
|
|
31
|
-
"
|
|
32
|
-
"
|
|
33
|
-
"
|
|
31
|
+
"@xwiki/platform-authentication-api": "18.0.0-1766398944",
|
|
32
|
+
"axios": "1.13.2",
|
|
33
|
+
"electron-store": "11.0.2"
|
|
34
34
|
},
|
|
35
35
|
"peerDependencies": {
|
|
36
|
-
"electron": "39.
|
|
36
|
+
"electron": "39.2.6"
|
|
37
37
|
},
|
|
38
38
|
"devDependencies": {
|
|
39
39
|
"typescript": "5.9.3",
|
|
40
|
-
"vite": "7.
|
|
41
|
-
"@xwiki/cristal-dev-config": "0.
|
|
40
|
+
"vite": "7.2.7",
|
|
41
|
+
"@xwiki/cristal-dev-config": "0.25.0"
|
|
42
42
|
},
|
|
43
43
|
"scripts": {
|
|
44
44
|
"api-extractor:local": "api-extractor run --local",
|
package/src/index.ts
CHANGED
|
@@ -37,7 +37,7 @@ import {
|
|
|
37
37
|
} from "./storage.js";
|
|
38
38
|
import axios from "axios";
|
|
39
39
|
import { BrowserWindow, ipcMain, shell } from "electron";
|
|
40
|
-
import type { UserDetails } from "@xwiki/
|
|
40
|
+
import type { UserDetails } from "@xwiki/platform-authentication-api";
|
|
41
41
|
|
|
42
42
|
const callbackUrl = "http://callback/";
|
|
43
43
|
|
package/CHANGELOG.md
DELETED
|
@@ -1,100 +0,0 @@
|
|
|
1
|
-
# @xwiki/cristal-electron-authentication-nextcloud-main
|
|
2
|
-
|
|
3
|
-
## 0.23.0
|
|
4
|
-
|
|
5
|
-
### Minor Changes
|
|
6
|
-
|
|
7
|
-
- Cristal 0.23 Release
|
|
8
|
-
|
|
9
|
-
### Patch Changes
|
|
10
|
-
|
|
11
|
-
- Updated dependencies
|
|
12
|
-
- @xwiki/cristal-authentication-api@0.23.0
|
|
13
|
-
|
|
14
|
-
## 0.22.0
|
|
15
|
-
|
|
16
|
-
### Minor Changes
|
|
17
|
-
|
|
18
|
-
- Cristal 0.22 Release
|
|
19
|
-
|
|
20
|
-
### Patch Changes
|
|
21
|
-
|
|
22
|
-
- Updated dependencies
|
|
23
|
-
- @xwiki/cristal-authentication-api@0.22.0
|
|
24
|
-
|
|
25
|
-
## 0.21.0
|
|
26
|
-
|
|
27
|
-
### Minor Changes
|
|
28
|
-
|
|
29
|
-
- Cristal 0.21 Release
|
|
30
|
-
|
|
31
|
-
### Patch Changes
|
|
32
|
-
|
|
33
|
-
- Updated dependencies
|
|
34
|
-
- @xwiki/cristal-authentication-api@0.21.0
|
|
35
|
-
|
|
36
|
-
## 0.20.0
|
|
37
|
-
|
|
38
|
-
### Minor Changes
|
|
39
|
-
|
|
40
|
-
- Cristal 0.20 Release
|
|
41
|
-
|
|
42
|
-
### Patch Changes
|
|
43
|
-
|
|
44
|
-
- Updated dependencies
|
|
45
|
-
- @xwiki/cristal-authentication-api@0.20.0
|
|
46
|
-
|
|
47
|
-
## 0.20.0-rc.0
|
|
48
|
-
|
|
49
|
-
### Minor Changes
|
|
50
|
-
|
|
51
|
-
- Cristal 0.20-rc-1 Release
|
|
52
|
-
|
|
53
|
-
### Patch Changes
|
|
54
|
-
|
|
55
|
-
- Updated dependencies
|
|
56
|
-
- @xwiki/cristal-authentication-api@0.20.0-rc.0
|
|
57
|
-
|
|
58
|
-
## 0.19.0
|
|
59
|
-
|
|
60
|
-
### Minor Changes
|
|
61
|
-
|
|
62
|
-
- 3eea53e: Cristal 0.19 Release
|
|
63
|
-
|
|
64
|
-
### Patch Changes
|
|
65
|
-
|
|
66
|
-
- Updated dependencies [3eea53e]
|
|
67
|
-
- @xwiki/cristal-authentication-api@0.19.0
|
|
68
|
-
|
|
69
|
-
## 0.18.0
|
|
70
|
-
|
|
71
|
-
### Minor Changes
|
|
72
|
-
|
|
73
|
-
- Cristal 0.18 Release
|
|
74
|
-
|
|
75
|
-
### Patch Changes
|
|
76
|
-
|
|
77
|
-
- Updated dependencies
|
|
78
|
-
- @xwiki/cristal-authentication-api@0.18.0
|
|
79
|
-
|
|
80
|
-
## 0.17.0
|
|
81
|
-
|
|
82
|
-
### Minor Changes
|
|
83
|
-
|
|
84
|
-
- Cristal 0.17 Release
|
|
85
|
-
|
|
86
|
-
### Patch Changes
|
|
87
|
-
|
|
88
|
-
- Updated dependencies
|
|
89
|
-
- @xwiki/cristal-authentication-api@0.17.0
|
|
90
|
-
|
|
91
|
-
## 0.16.0
|
|
92
|
-
|
|
93
|
-
### Minor Changes
|
|
94
|
-
|
|
95
|
-
- Cristal 0.16 Release
|
|
96
|
-
|
|
97
|
-
### Patch Changes
|
|
98
|
-
|
|
99
|
-
- Updated dependencies
|
|
100
|
-
- @xwiki/cristal-authentication-api@0.16.0
|