Compare commits
	
		
			36 Commits
		
	
	
		
			v0.10.3-et
			...
			v0.10.3-et
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | abc922c956 | ||
|   | 4f2cd38344 | ||
|   | ca71038874 | ||
|   | be867b6b0d | ||
|   | f2f540b429 | ||
|   | 7feec4ba07 | ||
|   | 1d5fef1e53 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9c40efde17 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 53dff66978 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3a595247e8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 33f5f60e31 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9dd2ea57c9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fae7a696de | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 49e8b2d0f5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 281d908d3f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bacc42fe9c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1c26a28ca9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d3a04cd132 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e6060a23ac | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4b7fbf483a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bc3c30da92 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1896f770d1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 99d0b9ad72 | ||
|   | 944afb9056 | ||
|   | 23f5a24803 | ||
|   | 60ae00ac14 | ||
|   | 26862fa708 | ||
|   | 853d14c1ce | ||
|   | 11a5cac709 | ||
|   | 0d021021df | ||
|   | 19302466ef | ||
|   | 0594259ae4 | ||
|   | ba485bbb18 | ||
|   | 9fc005032c | ||
|   | f5d6f24b30 | ||
|   | a42efe7eda | 
							
								
								
									
										16
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | version: 2 | ||||||
|  | updates: | ||||||
|  |   - package-ecosystem: "npm" | ||||||
|  |     directory: "/" | ||||||
|  |     schedule: | ||||||
|  |       interval: "weekly" | ||||||
|  |  | ||||||
|  |   - package-ecosystem: "docker" | ||||||
|  |     directory: "/" | ||||||
|  |     schedule: | ||||||
|  |       interval: "weekly" | ||||||
|  |  | ||||||
|  |   - package-ecosystem: "github-actions" | ||||||
|  |     directory: "/" | ||||||
|  |     schedule: | ||||||
|  |       interval: "weekly" | ||||||
							
								
								
									
										2
									
								
								.github/workflows/workflow.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/workflow.yml
									
									
									
									
										vendored
									
									
								
							| @@ -52,7 +52,7 @@ jobs: | |||||||
|           name: dist |           name: dist | ||||||
|           path: dist/ |           path: dist/ | ||||||
|       - name: Set up Docker Buildx |       - name: Set up Docker Buildx | ||||||
|         uses: docker/setup-buildx-action@v1 |         uses: docker/setup-buildx-action@v3 | ||||||
|       - name: Login to ghcr.io |       - name: Login to ghcr.io | ||||||
|         uses: docker/login-action@v3 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|   | |||||||
							
								
								
									
										36
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								README.md
									
									
									
									
									
								
							| @@ -7,12 +7,13 @@ This project is built using [react-admin](https://marmelab.com/react-admin/). | |||||||
| <!-- vim-markdown-toc GFM --> | <!-- vim-markdown-toc GFM --> | ||||||
|  |  | ||||||
| * [Fork differences](#fork-differences) | * [Fork differences](#fork-differences) | ||||||
|   * [Available via CDN](#available-via-cdn) |   * [Availability](#availability) | ||||||
|   * [Changes](#changes) |   * [Changes](#changes) | ||||||
|   * [Development](#development) |   * [Development](#development) | ||||||
| * [Configuration](#configuration) | * [Configuration](#configuration) | ||||||
|   * [Restricting available homeserver](#restricting-available-homeserver) |   * [Restricting available homeserver](#restricting-available-homeserver) | ||||||
|   * [Protecting appservice managed users](#protecting-appservice-managed-users) |   * [Protecting appservice managed users](#protecting-appservice-managed-users) | ||||||
|  |   * [Adding custom menu items](#adding-custom-menu-items) | ||||||
|   * [Providing support URL](#providing-support-url) |   * [Providing support URL](#providing-support-url) | ||||||
| * [Usage](#usage) | * [Usage](#usage) | ||||||
|   * [Supported Synapse](#supported-synapse) |   * [Supported Synapse](#supported-synapse) | ||||||
| @@ -33,9 +34,11 @@ With [Awesome-Technologies/synapse-admin](https://github.com/Awesome-Technologie | |||||||
| fork is intended to be a more feature-rich version of the original project. The main goal is to provide a more | fork is intended to be a more feature-rich version of the original project. The main goal is to provide a more | ||||||
| user-friendly interface for managing Synapse homeservers. | user-friendly interface for managing Synapse homeservers. | ||||||
|  |  | ||||||
| ### Available via CDN | ### Availability | ||||||
|  |  | ||||||
| On [admin.etke.cc](https://admin.etke.cc) you can find the latest version of this fork. | * As a core/default component on [etke.cc](https://etke.cc/?utm_source=github&utm_medium=readme&utm_campaign=synapse-admin) | ||||||
|  | * Via CDN on [admin.etke.cc](https://admin.etke.cc) | ||||||
|  | * As a component in [Matrix-Docker-Ansible-Deploy Playbook](https://github.com/spantaleev/matrix-docker-ansible-deploy/blob/master/docs/configuring-playbook-synapse-admin.md) | ||||||
|  |  | ||||||
| ### Changes | ### Changes | ||||||
|  |  | ||||||
| @@ -63,6 +66,11 @@ The following changes are already implemented: | |||||||
| * [Provide options to delete media and redact events on user erase](https://github.com/etkecc/synapse-admin/pull/49) | * [Provide options to delete media and redact events on user erase](https://github.com/etkecc/synapse-admin/pull/49) | ||||||
| * [Authenticated Media support](https://github.com/etkecc/synapse-admin/pull/51) | * [Authenticated Media support](https://github.com/etkecc/synapse-admin/pull/51) | ||||||
| * [Better media preview/download](https://github.com/etkecc/synapse-admin/pull/53) | * [Better media preview/download](https://github.com/etkecc/synapse-admin/pull/53) | ||||||
|  | * [Login with access token](https://github.com/etkecc/synapse-admin/pull/58) | ||||||
|  | * [Fix footer causing vertical scrollbar](https://github.com/etkecc/synapse-admin/pull/60) | ||||||
|  | * [Custom Menu Items](https://github.com/etkecc/synapse-admin/pull/79) | ||||||
|  | * [Add user profile to the top menu](https://github.com/etkecc/synapse-admin/pull/80) | ||||||
|  | * [Enable visual customization](https://github.com/etkecc/synapse-admin/pull/81) | ||||||
|  |  | ||||||
| _the list will be updated as new changes are added_ | _the list will be updated as new changes are added_ | ||||||
|  |  | ||||||
| @@ -125,9 +133,29 @@ Example for [mautrix-telegram](https://github.com/mautrix/telegram) | |||||||
| } | } | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|  | ### Adding custom menu items | ||||||
|  |  | ||||||
|  | You can add custom menu items to the main menu by providing a `menu` array in the `config.json`. | ||||||
|  |  | ||||||
|  | ```json | ||||||
|  | { | ||||||
|  |   "menu": [ | ||||||
|  |     { | ||||||
|  |       "label": "Contact support", | ||||||
|  |       "icon": "SupportAgent", | ||||||
|  |       "url": "https://github.com/etkecc/synapse-admin/issues" | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Where `icon` is one of the [preloaded icons](./src/components/icons.ts) | ||||||
|  |  | ||||||
| ### Providing support URL | ### Providing support URL | ||||||
|  |  | ||||||
| Synapse-Admin provides a support link in the main menu - `Contact support`. By default, the link points to the GitHub issues page of the project. You can change this link by providing a `supportURL` in the `config.json`. | **Deprecated**: use `menu` config option described above. Automatically migrated to the `menu` if the `supportURL` is present. | ||||||
|  |  | ||||||
|  | ~~Synapse-Admin provides a support link in the main menu - `Contact support`. By default, the link points to the GitHub issues page of the project. You can change this link by providing a `supportURL` in the `config.json`.~~ | ||||||
|  |  | ||||||
| ```json | ```json | ||||||
| { | { | ||||||
|   | |||||||
							
								
								
									
										15
									
								
								index.html
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								index.html
									
									
									
									
									
								
							| @@ -22,6 +22,11 @@ | |||||||
|         font-family: sans-serif; |         font-family: sans-serif; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  |       .layout { | ||||||
|  |         min-height: 90vh !important; | ||||||
|  |         height: 90vh; | ||||||
|  |       } | ||||||
|  |  | ||||||
|       .loader-container { |       .loader-container { | ||||||
|         display: flex; |         display: flex; | ||||||
|         align-items: center; |         align-items: center; | ||||||
| @@ -120,13 +125,7 @@ | |||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|     <script type="module" src="/src/index.tsx"></script> |     <script type="module" src="/src/index.tsx"></script> | ||||||
|     <footer |     <span id="js-version" style="display: none;"></span> | ||||||
|       style="position: relative; z-index: 2; height: 2em; margin-top: 0; line-height: 2em; background-color: #eee; border: 0.5px solid #ddd"> |  | ||||||
|       <a id="copyright" href="https://github.com/etkecc/synapse-admin" |  | ||||||
|         style="margin-left: 1em; color: #888; font-family: Roboto, Helvetica, Arial, sans-serif; font-weight: 100; font-size: 0.8em; text-decoration: none;"> |  | ||||||
|         Synapse-Admin <b><span id="version"></span></b> by Awesome Technologies Innovationslabor GmbH |  | ||||||
|       </a> |  | ||||||
|     </footer> |  | ||||||
|   </body> |   </body> | ||||||
|   <script>document.getElementById("version").textContent = __SYNAPSE_ADMIN_VERSION__</script> |   <script>document.getElementById("js-version").textContent = __SYNAPSE_ADMIN_VERSION__</script> | ||||||
| </html> | </html> | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								justfile
									
									
									
									
									
								
							
							
						
						
									
										8
									
								
								justfile
									
									
									
									
									
								
							| @@ -25,11 +25,15 @@ run-dev: | |||||||
| stop-dev: | stop-dev: | ||||||
|     @docker-compose -f docker-compose-dev.yml stop |     @docker-compose -f docker-compose-dev.yml stop | ||||||
|  |  | ||||||
|  | # register a user in the dev stack | ||||||
| register-user localpart password *admin: | register-user localpart password *admin: | ||||||
| 	docker-compose exec synapse register_new_matrix_user {{ if admin == "1" {"--admin"} else {"--no-admin"} }} -u {{ localpart }} -p {{ password }} -c /config/homeserver.yaml http://localhost:8008 | 	docker-compose exec synapse register_new_matrix_user {{ if admin == "1" {"--admin"} else {"--no-admin"} }} -u {{ localpart }} -p {{ password }} -c /config/homeserver.yaml http://localhost:8008 | ||||||
|  |  | ||||||
|  | # run yarn {fix,lint,test} commands | ||||||
|  | test: | ||||||
|  |     @-yarn run fix | ||||||
|  |     @-yarn run lint | ||||||
|  |     @-yarn run test | ||||||
|  |  | ||||||
| # run the app in a production mode | # run the app in a production mode | ||||||
| run-prod: build | run-prod: build | ||||||
|   | |||||||
							
								
								
									
										35
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										35
									
								
								package.json
									
									
									
									
									
								
							| @@ -3,7 +3,7 @@ | |||||||
|   "version": "0.10.3", |   "version": "0.10.3", | ||||||
|   "description": "Admin GUI for the Matrix.org server Synapse", |   "description": "Admin GUI for the Matrix.org server Synapse", | ||||||
|   "type": "module", |   "type": "module", | ||||||
|   "author": "Awesome Technologies Innovationslabor GmbH", |   "author": "etke.cc (originally by Awesome Technologies Innovationslabor GmbH)", | ||||||
|   "license": "Apache-2.0", |   "license": "Apache-2.0", | ||||||
|   "homepage": ".", |   "homepage": ".", | ||||||
|   "repository": { |   "repository": { | ||||||
| @@ -13,20 +13,20 @@ | |||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@eslint/js": "^9.7.0", |     "@eslint/js": "^9.7.0", | ||||||
|     "@testing-library/dom": "^10.0.0", |     "@testing-library/dom": "^10.0.0", | ||||||
|     "@testing-library/jest-dom": "^6.0.0", |     "@testing-library/jest-dom": "^6.6.2", | ||||||
|     "@testing-library/react": "^16.0.0", |     "@testing-library/react": "^16.0.0", | ||||||
|     "@testing-library/user-event": "^14.5.2", |     "@testing-library/user-event": "^14.5.2", | ||||||
|     "@types/jest": "^29.5.13", |     "@types/jest": "^29.5.13", | ||||||
|     "@types/lodash": "^4.17.7", |     "@types/lodash": "^4.17.7", | ||||||
|     "@types/node": "^20.14.12", |     "@types/node": "^22.7.7", | ||||||
|     "@types/papaparse": "^5.3.14", |     "@types/papaparse": "^5.3.15", | ||||||
|     "@types/react": "^18.3.3", |     "@types/react": "^18.3.3", | ||||||
|     "@typescript-eslint/eslint-plugin": "^7.16.1", |     "@typescript-eslint/eslint-plugin": "^8.10.0", | ||||||
|     "@typescript-eslint/parser": "^7.16.1", |     "@typescript-eslint/parser": "^8.10.0", | ||||||
|     "@vitejs/plugin-react": "^4.3.1", |     "@vitejs/plugin-react": "^4.3.2", | ||||||
|     "eslint": "^8.57.0", |     "eslint": "^8.57.0", | ||||||
|     "eslint-config-prettier": "^9.1.0", |     "eslint-config-prettier": "^9.1.0", | ||||||
|     "eslint-plugin-import": "^2.29.1", |     "eslint-plugin-import": "^2.31.0", | ||||||
|     "eslint-plugin-jsx-a11y": "^6.9.0", |     "eslint-plugin-jsx-a11y": "^6.9.0", | ||||||
|     "eslint-plugin-prettier": "^5.2.1", |     "eslint-plugin-prettier": "^5.2.1", | ||||||
|     "eslint-plugin-unused-imports": "^3.2.0", |     "eslint-plugin-unused-imports": "^3.2.0", | ||||||
| @@ -38,10 +38,10 @@ | |||||||
|     "react-test-renderer": "^18.3.1", |     "react-test-renderer": "^18.3.1", | ||||||
|     "ts-jest": "^29.2.5", |     "ts-jest": "^29.2.5", | ||||||
|     "ts-node": "^10.9.2", |     "ts-node": "^10.9.2", | ||||||
|     "typescript": "^5.4.5", |     "typescript": "^5.6.3", | ||||||
|     "typescript-eslint": "^7.16.1", |     "typescript-eslint": "^8.10.0", | ||||||
|     "vite": "^5.4.6", |     "vite": "^5.4.6", | ||||||
|     "vite-plugin-version-mark": "^0.1.0" |     "vite-plugin-version-mark": "^0.1.2" | ||||||
|   }, |   }, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@emotion/react": "^11.13.0", |     "@emotion/react": "^11.13.0", | ||||||
| @@ -50,24 +50,25 @@ | |||||||
|     "@haxqer/ra-language-chinese": "^4.16.2", |     "@haxqer/ra-language-chinese": "^4.16.2", | ||||||
|     "@mui/icons-material": "^6.1.1", |     "@mui/icons-material": "^6.1.1", | ||||||
|     "@mui/material": "^6.1.1", |     "@mui/material": "^6.1.1", | ||||||
|  |     "@mui/utils": "^5.16.6", | ||||||
|     "@tanstack/react-query": "^5.56.2", |     "@tanstack/react-query": "^5.56.2", | ||||||
|     "history": "^5.3.0", |     "history": "^5.3.0", | ||||||
|     "lodash": "^4.17.21", |     "lodash": "^4.17.21", | ||||||
|     "papaparse": "^5.4.1", |     "papaparse": "^5.4.1", | ||||||
|     "ra-core": "^5.2.0", |     "ra-core": "^5.3.0", | ||||||
|     "ra-i18n-polyglot": "^5.2.0", |     "ra-i18n-polyglot": "^5.3.0", | ||||||
|     "ra-language-english": "^5.2.0", |     "ra-language-english": "^5.3.0", | ||||||
|     "ra-language-farsi": "^5.0.0", |     "ra-language-farsi": "^5.0.0", | ||||||
|     "ra-language-french": "^5.2.0", |     "ra-language-french": "^5.2.0", | ||||||
|     "ra-language-italian": "^3.13.1", |     "ra-language-italian": "^3.13.1", | ||||||
|     "ra-language-russian": "^4.14.2", |     "ra-language-russian": "^4.14.2", | ||||||
|     "react": "^18.3.1", |     "react": "^18.3.1", | ||||||
|     "react-admin": "^5.2.0", |     "react-admin": "^5.3.0", | ||||||
|     "react-dom": "^18.3.1", |     "react-dom": "^18.3.1", | ||||||
|     "react-hook-form": "^7.53.0", |     "react-hook-form": "^7.53.1", | ||||||
|     "react-is": "^18.3.1", |     "react-is": "^18.3.1", | ||||||
|     "react-router": "^6.26.2", |     "react-router": "^6.26.2", | ||||||
|     "react-router-dom": "^6.26.2" |     "react-router-dom": "^6.27.0" | ||||||
|   }, |   }, | ||||||
|   "scripts": { |   "scripts": { | ||||||
|     "start": "vite serve", |     "start": "vite serve", | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { render, screen, waitFor } from "@testing-library/react"; | import { render, screen } from "@testing-library/react"; | ||||||
| import fetchMock from "jest-fetch-mock"; | import fetchMock from "jest-fetch-mock"; | ||||||
| fetchMock.enableMocks(); | fetchMock.enableMocks(); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -23,6 +23,7 @@ import users from "./resources/users"; | |||||||
| import authProvider from "./synapse/authProvider"; | import authProvider from "./synapse/authProvider"; | ||||||
| import dataProvider from "./synapse/dataProvider"; | import dataProvider from "./synapse/dataProvider"; | ||||||
| import { QueryClient, QueryClientProvider } from "@tanstack/react-query"; | import { QueryClient, QueryClientProvider } from "@tanstack/react-query"; | ||||||
|  | import Footer from "./components/Footer"; | ||||||
|  |  | ||||||
| // TODO: Can we use lazy loading together with browser locale? | // TODO: Can we use lazy loading together with browser locale? | ||||||
| const messages = { | const messages = { | ||||||
| @@ -81,6 +82,7 @@ const App = () => ( | |||||||
|       <Resource name="room_state" /> |       <Resource name="room_state" /> | ||||||
|       <Resource name="destination_rooms" /> |       <Resource name="destination_rooms" /> | ||||||
|     </Admin> |     </Admin> | ||||||
|  |     <Footer /> | ||||||
|   </QueryClientProvider> |   </QueryClientProvider> | ||||||
| ); | ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -4,6 +4,13 @@ interface AppContextType { | |||||||
|   restrictBaseUrl: string | string[]; |   restrictBaseUrl: string | string[]; | ||||||
|   asManagedUsers: string[]; |   asManagedUsers: string[]; | ||||||
|   supportURL: string; |   supportURL: string; | ||||||
|  |   menu: MenuItem[]; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | interface MenuItem { | ||||||
|  |   label: string; | ||||||
|  |   icon: string; | ||||||
|  |   url: string; | ||||||
| } | } | ||||||
|  |  | ||||||
| export const AppContext = createContext({}); | export const AppContext = createContext({}); | ||||||
|   | |||||||
| @@ -1,26 +1,103 @@ | |||||||
| import { Layout, Menu } from 'react-admin'; | import { AppBar, TitlePortal, InspectorButton, Confirm, Layout, Logout, Menu, useLogout, UserMenu } from "react-admin"; | ||||||
| import LiveHelpIcon from '@mui/icons-material/LiveHelp'; | import { LoginMethod } from "../pages/LoginPage"; | ||||||
|  | import { useEffect, useState, Suspense } from "react"; | ||||||
|  | import { Icons, DefaultIcon } from "./icons"; | ||||||
|  |  | ||||||
| const DEFAULT_SUPPORT_LINK = "https://github.com/etkecc/synapse-admin/issues"; | const AdminUserMenu = () => { | ||||||
| const supportLink = (): string => { |   const [open, setOpen] = useState(false); | ||||||
|     try { |   const logout = useLogout(); | ||||||
|         new URL(localStorage.getItem("support_url") || ''); // Check if the URL is valid |   const checkLoginType = (ev: React.MouseEvent<HTMLDivElement>) => { | ||||||
|         return localStorage.getItem("support_url") || DEFAULT_SUPPORT_LINK; |     const loginType: LoginMethod = (localStorage.getItem("login_type") || "credentials") as LoginMethod; | ||||||
|     } catch (e) { |     if (loginType === "accessToken") { | ||||||
|         return DEFAULT_SUPPORT_LINK; |       ev.stopPropagation(); | ||||||
|  |       setOpen(true); | ||||||
|     } |     } | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|  |   const handleConfirm = () => { | ||||||
|  |     setOpen(false); | ||||||
|  |     logout(); | ||||||
|  |   }; | ||||||
|  |  | ||||||
| const AdminMenu = () => ( |   const handleDialogClose = () => { | ||||||
|     <Menu> |     setOpen(false); | ||||||
|  |     localStorage.removeItem("access_token"); | ||||||
|  |     localStorage.removeItem("login_type"); | ||||||
|  |     window.location.reload(); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <UserMenu> | ||||||
|  |       <div onClickCapture={checkLoginType}> | ||||||
|  |         <Logout /> | ||||||
|  |       </div> | ||||||
|  |       <Confirm | ||||||
|  |         isOpen={open} | ||||||
|  |         title="synapseadmin.auth.logout_acces_token_dialog.title" | ||||||
|  |         content="synapseadmin.auth.logout_acces_token_dialog.content" | ||||||
|  |         onConfirm={handleConfirm} | ||||||
|  |         onClose={handleDialogClose} | ||||||
|  |         confirm="synapseadmin.auth.logout_acces_token_dialog.confirm" | ||||||
|  |         cancel="synapseadmin.auth.logout_acces_token_dialog.cancel" | ||||||
|  |       /> | ||||||
|  |     </UserMenu> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const AdminAppBar = () => { | ||||||
|  |   return (<AppBar userMenu={<AdminUserMenu />}> | ||||||
|  |     <TitlePortal /> | ||||||
|  |     <InspectorButton /> | ||||||
|  |   </AppBar>); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const AdminMenu = (props) => { | ||||||
|  |   const [menu, setMenu] = useState([]); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     const menuConfig = localStorage.getItem('menu'); | ||||||
|  |     if (menuConfig) { | ||||||
|  |       try { | ||||||
|  |         setMenu(JSON.parse(menuConfig)); | ||||||
|  |       } catch (e) { | ||||||
|  |         console.error('Error parsing menu configuration', e); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   }, []); | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <Menu {...props}> | ||||||
|       <Menu.ResourceItems /> |       <Menu.ResourceItems /> | ||||||
|         <Menu.Item to={supportLink()} target="_blank" primaryText="Contact support" leftIcon={<LiveHelpIcon />} /> |       {menu.map((item, index) => { | ||||||
|  |         const { url, icon, label } = item; | ||||||
|  |         const IconComponent = Icons[icon] as React.ComponentType<any> | undefined; | ||||||
|  |  | ||||||
|  |         return ( | ||||||
|  |           <Suspense key={index}> | ||||||
|  |             <Menu.Item | ||||||
|  |               to={url} | ||||||
|  |               target="_blank" | ||||||
|  |               primaryText={label} | ||||||
|  |               leftIcon={IconComponent ? <IconComponent /> : <DefaultIcon />} | ||||||
|  |               onClick={props.onMenuClick} | ||||||
|  |             /> | ||||||
|  |           </Suspense> | ||||||
|  |         ); | ||||||
|  |       })} | ||||||
|     </Menu> |     </Menu> | ||||||
|   ); |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
| export const AdminLayout = ({ children }) => ( | export const AdminLayout = ({ children }) => ( | ||||||
|     <Layout menu={AdminMenu}> |   <Layout appBar={AdminAppBar} menu={AdminMenu} sx={{ | ||||||
|  |       ['& .RaLayout-appFrame']: { | ||||||
|  |         minHeight: '90vh', | ||||||
|  |         height: '90vh', | ||||||
|  |       }, | ||||||
|  |       ['& .RaLayout-content']: { | ||||||
|  |         marginBottom: '3rem', | ||||||
|  |       }, | ||||||
|  |     }}> | ||||||
|     {children} |     {children} | ||||||
|   </Layout> |   </Layout> | ||||||
| ); | ); | ||||||
|   | |||||||
| @@ -1,12 +1,11 @@ | |||||||
| import { get } from "lodash"; | import { get } from "lodash"; | ||||||
| import { Avatar, AvatarProps } from "@mui/material"; | import { Avatar, AvatarProps } from "@mui/material"; | ||||||
| import { useRecordContext } from "react-admin"; | import { FieldProps, useRecordContext } from "react-admin"; | ||||||
| import { useState, useEffect, useCallback } from "react"; | import { useState, useEffect, useCallback } from "react"; | ||||||
| import { fetchAuthenticatedMedia } from "../utils/fetchMedia"; | import { fetchAuthenticatedMedia } from "../utils/fetchMedia"; | ||||||
|  |  | ||||||
| const AvatarField = ({ source, ...rest }: AvatarProps & { source: string, label?: string }) => { | const AvatarField = ({ source, ...rest }: AvatarProps & FieldProps) => { | ||||||
|   const { alt, classes, sizes, sx, variant } = rest; |   const { alt, classes, sizes, sx, variant } = rest; | ||||||
|  |  | ||||||
|   const record = useRecordContext(rest); |   const record = useRecordContext(rest); | ||||||
|   const mxcURL = get(record, source)?.toString(); |   const mxcURL = get(record, source)?.toString(); | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										40
									
								
								src/components/Footer.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								src/components/Footer.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | import { Box, Link, Typography } from "@mui/material"; | ||||||
|  | import { useEffect, useState } from "react"; | ||||||
|  |  | ||||||
|  | const Footer = () => { | ||||||
|  |   const [version, setVersion] = useState<string | null>(null); | ||||||
|  |   useEffect(() => { | ||||||
|  |     const version = document.getElementById("js-version")?.textContent; | ||||||
|  |     if (version) { | ||||||
|  |       setVersion(version); | ||||||
|  |     } | ||||||
|  |   }, []); | ||||||
|  |  | ||||||
|  |   return (<Box | ||||||
|  |     component="footer" | ||||||
|  |     sx={{ | ||||||
|  |       position: 'fixed', | ||||||
|  |       zIndex: 100, | ||||||
|  |       bottom: 0, | ||||||
|  |       width: '100%', | ||||||
|  |       bgcolor: "#eee", | ||||||
|  |       borderTop: '1px solid', | ||||||
|  |       borderColor: '#ddd', | ||||||
|  |       p: 1, | ||||||
|  |     }}> | ||||||
|  |     <Typography variant="body2"> | ||||||
|  |       <Link sx={{ color: "#888", textDecoration: 'none' }} href="https://github.com/etkecc/synapse-admin" target="_blank"> | ||||||
|  |         Synapse-Admin | ||||||
|  |       </Link> <Link href={`https://github.com/etkecc/synapse-admin/releases/tag/`+version} target="_blank"> | ||||||
|  |         <span style={{ fontWeight: 'bold', color: "#000" }}>{version}</span> | ||||||
|  |       </Link> <Link sx={{ color: "#888", textDecoration: 'none' }} href="https://etke.cc/?utm_source=synapse-admin&utm_medium=footer&utm_campaign=synapse-admin" target="_blank"> | ||||||
|  |         by etke.cc | ||||||
|  |       </Link> <Link sx={{ color: "#888", textDecoration: 'none' }} href="https://github.com/awesome-technologies/synapse-admin" target="_blank"> | ||||||
|  |         (originally developed by Awesome Technologies Innovationslabor GmbH) | ||||||
|  |       </Link> | ||||||
|  |     </Typography> | ||||||
|  |   </Box> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default Footer; | ||||||
| @@ -121,11 +121,7 @@ const FilePicker = () => { | |||||||
|  |  | ||||||
|   const verifyCsv = ({ data, meta, errors }: ParseResult<ImportLine>, { setValues, setStats, setError }) => { |   const verifyCsv = ({ data, meta, errors }: ParseResult<ImportLine>, { setValues, setStats, setError }) => { | ||||||
|     /* First, verify the presence of required fields */ |     /* First, verify the presence of required fields */ | ||||||
|     const missingFields = expectedFields.filter(eF => { |     const missingFields = expectedFields.filter(eF => !meta.fields?.find(mF => eF === mF)); | ||||||
|       const result = meta.fields?.find(mF => eF === mF); |  | ||||||
|       if (result === undefined) { return eF; } // missing field |  | ||||||
|       return undefined; // field found |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     if (missingFields.length > 0) { |     if (missingFields.length > 0) { | ||||||
|       setError(translate("import_users.error.required_field", { field: missingFields[0] })); |       setError(translate("import_users.error.required_field", { field: missingFields[0] })); | ||||||
|   | |||||||
							
								
								
									
										58
									
								
								src/components/LoginFormBox.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								src/components/LoginFormBox.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | |||||||
|  | import { styled } from "@mui/material/styles"; | ||||||
|  | import { Box } from "@mui/material"; | ||||||
|  |  | ||||||
|  | const LoginFormBox = styled(Box)(({ theme }) => ({ | ||||||
|  |   display: "flex", | ||||||
|  |   flexDirection: "column", | ||||||
|  |   minHeight: "calc(100vh - 1rem)", | ||||||
|  |   alignItems: "center", | ||||||
|  |   justifyContent: "flex-start", | ||||||
|  |   background: "url(./images/floating-cogs.svg)", | ||||||
|  |   backgroundColor: "#f9f9f9", | ||||||
|  |   backgroundRepeat: "no-repeat", | ||||||
|  |   backgroundSize: "cover", | ||||||
|  |  | ||||||
|  |   [`& .card`]: { | ||||||
|  |     maxWidth: "30rem", | ||||||
|  |     marginTop: "6rem", | ||||||
|  |     marginBottom: "6rem", | ||||||
|  |   }, | ||||||
|  |   [`& .avatar`]: { | ||||||
|  |     margin: "1rem", | ||||||
|  |     display: "flex", | ||||||
|  |     justifyContent: "center", | ||||||
|  |   }, | ||||||
|  |   [`& .icon`]: { | ||||||
|  |     backgroundColor: theme.palette.grey[500], | ||||||
|  |   }, | ||||||
|  |   [`& .hint`]: { | ||||||
|  |     marginTop: "1em", | ||||||
|  |     marginBottom: "1em", | ||||||
|  |     display: "flex", | ||||||
|  |     justifyContent: "center", | ||||||
|  |     color: theme.palette.grey[600], | ||||||
|  |   }, | ||||||
|  |   [`& .form`]: { | ||||||
|  |     padding: "0 1rem 1rem 1rem", | ||||||
|  |   }, | ||||||
|  |   [`& .select`]: { | ||||||
|  |     marginBottom: "2rem", | ||||||
|  |   }, | ||||||
|  |   [`& .actions`]: { | ||||||
|  |     padding: "0 1rem 1rem 1rem", | ||||||
|  |   }, | ||||||
|  |   [`& .serverVersion`]: { | ||||||
|  |     color: theme.palette.grey[500], | ||||||
|  |     fontFamily: "Roboto, Helvetica, Arial, sans-serif", | ||||||
|  |     marginLeft: "0.5rem", | ||||||
|  |   }, | ||||||
|  |   [`& .matrixVersions`]: { | ||||||
|  |     color: theme.palette.grey[500], | ||||||
|  |     fontFamily: "Roboto, Helvetica, Arial, sans-serif", | ||||||
|  |     fontSize: "0.8rem", | ||||||
|  |     marginBottom: "1rem", | ||||||
|  |     marginLeft: "0.5rem", | ||||||
|  |   }, | ||||||
|  | })); | ||||||
|  |  | ||||||
|  | export default LoginFormBox; | ||||||
							
								
								
									
										12
									
								
								src/components/icons.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								src/components/icons.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | import { lazy } from "react"; | ||||||
|  |  | ||||||
|  | export const Icons = { | ||||||
|  |   Announcement: lazy(() => import('@mui/icons-material/Announcement')), | ||||||
|  |   Engineering: lazy(() => import('@mui/icons-material/Engineering')), | ||||||
|  |   HelpCenter: lazy(() => import('@mui/icons-material/HelpCenter')), | ||||||
|  |   SupportAgent: lazy(() => import('@mui/icons-material/SupportAgent')), | ||||||
|  |   Default: lazy(() => import('@mui/icons-material/OpenInNew')), | ||||||
|  |   // Add more icons as needed | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export const DefaultIcon = Icons.Default; | ||||||
| @@ -2,7 +2,7 @@ import { formalGermanMessages } from "@haleos/ra-language-german"; | |||||||
|  |  | ||||||
| import { SynapseTranslationMessages } from "."; | import { SynapseTranslationMessages } from "."; | ||||||
|  |  | ||||||
| const de: SynapseTranslationMessages = { | const fixedGermanMessages = { | ||||||
|   ...formalGermanMessages, |   ...formalGermanMessages, | ||||||
|   ra: { |   ra: { | ||||||
|     ...formalGermanMessages.ra, |     ...formalGermanMessages.ra, | ||||||
| @@ -10,8 +10,30 @@ const de: SynapseTranslationMessages = { | |||||||
|       ...formalGermanMessages.ra.navigation, |       ...formalGermanMessages.ra.navigation, | ||||||
|       no_filtered_results: "Keine Ergebnisse", |       no_filtered_results: "Keine Ergebnisse", | ||||||
|       clear_filters: "Alle Filter entfernen", |       clear_filters: "Alle Filter entfernen", | ||||||
|  |       add_filter: "Filter hinzufügen", | ||||||
|  |     }, | ||||||
|  |     action: { | ||||||
|  |       ...formalGermanMessages.ra.action, | ||||||
|  |       update_application: "Anwendung aktualisieren", | ||||||
|  |     }, | ||||||
|  |     page: { | ||||||
|  |       ...formalGermanMessages.ra.page, | ||||||
|  |       empty: "Leer", | ||||||
|  |       access_denied: "Zugriff verweigert", | ||||||
|  |       authentication_error: "Authentifizierungsfehler", | ||||||
|  |     }, | ||||||
|  |     message: { | ||||||
|  |       ...formalGermanMessages.ra.message, | ||||||
|  |       access_denied: | ||||||
|  |         "Sie haben nicht die richtigen Berechtigungen um auf diese Seite zuzugreifen.", | ||||||
|  |       authentication_error: | ||||||
|  |         "Der Authentifizierungsserver hat einen Fehler zurückgegeben und Ihre Anmeldedaten konnten nicht überprüft werden.", | ||||||
|     }, |     }, | ||||||
|   }, |   }, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const de: SynapseTranslationMessages = { | ||||||
|  |   ...fixedGermanMessages, | ||||||
|   synapseadmin: { |   synapseadmin: { | ||||||
|     auth: { |     auth: { | ||||||
|       base_url: "Heimserver URL", |       base_url: "Heimserver URL", | ||||||
| @@ -22,6 +44,14 @@ const de: SynapseTranslationMessages = { | |||||||
|       protocol_error: "Die URL muss mit 'http://' oder 'https://' beginnen", |       protocol_error: "Die URL muss mit 'http://' oder 'https://' beginnen", | ||||||
|       url_error: "Keine gültige Matrix Server URL", |       url_error: "Keine gültige Matrix Server URL", | ||||||
|       sso_sign_in: "Anmeldung mit SSO", |       sso_sign_in: "Anmeldung mit SSO", | ||||||
|  |       credentials: "Anmeldedaten", | ||||||
|  |       access_token: "Zugriffstoken", | ||||||
|  |       logout_acces_token_dialog: { | ||||||
|  |         title: "Sie verwenden ein bestehendes Matrix-Zugriffstoken.", | ||||||
|  |         content: "Möchten Sie diese Sitzung (die anderswo, z.B. in einem Matrix-Client, verwendet werden könnte) beenden oder sich nur vom Admin-Panel abmelden?", | ||||||
|  |         confirm: "Sitzung beenden", | ||||||
|  |         cancel: "Nur vom Admin-Panel abmelden", | ||||||
|  |       }, | ||||||
|     }, |     }, | ||||||
|     users: { |     users: { | ||||||
|       invalid_user_id: "Lokaler Anteil der Matrix Benutzer-ID ohne Homeserver.", |       invalid_user_id: "Lokaler Anteil der Matrix Benutzer-ID ohne Homeserver.", | ||||||
|   | |||||||
| @@ -14,6 +14,14 @@ const en: SynapseTranslationMessages = { | |||||||
|       protocol_error: "URL has to start with 'http://' or 'https://'", |       protocol_error: "URL has to start with 'http://' or 'https://'", | ||||||
|       url_error: "Not a valid Matrix server URL", |       url_error: "Not a valid Matrix server URL", | ||||||
|       sso_sign_in: "Sign in with SSO", |       sso_sign_in: "Sign in with SSO", | ||||||
|  |       credentials: "Credentials", | ||||||
|  |       access_token: "Access token", | ||||||
|  |       logout_acces_token_dialog: { | ||||||
|  |         title: "You are using an existing Matrix access token.", | ||||||
|  |         content: "Do you want to destroy this session (that could be used elsewhere, e.g. in a Matrix client) or just logout from the admin panel?", | ||||||
|  |         confirm: "Destroy session", | ||||||
|  |         cancel: "Just logout from admin panel", | ||||||
|  |       }, | ||||||
|     }, |     }, | ||||||
|     users: { |     users: { | ||||||
|       invalid_user_id: "Localpart of a Matrix user-id without homeserver.", |       invalid_user_id: "Localpart of a Matrix user-id without homeserver.", | ||||||
|   | |||||||
| @@ -13,6 +13,14 @@ const fa: SynapseTranslationMessages = { | |||||||
|       protocol_error: "URL باید با 'http://' یا 'https://' شروع شود", |       protocol_error: "URL باید با 'http://' یا 'https://' شروع شود", | ||||||
|       url_error: "آدرس وارد شده یک سرور معتبر نیست", |       url_error: "آدرس وارد شده یک سرور معتبر نیست", | ||||||
|       sso_sign_in: "با SSO وارد شوید", |       sso_sign_in: "با SSO وارد شوید", | ||||||
|  |       credentials: "اعتبارنامه", | ||||||
|  |       access_token: "توکن دسترسی", | ||||||
|  |       logout_acces_token_dialog: { | ||||||
|  |         title: "شما در حال استفاده از یک نشانه دسترسی ماتریکس موجود هستید.", | ||||||
|  |         content: "آیا میخواهید این جلسه (که میتواند در جای دیگر، مانند یک کلاینت ماتریکس استفاده شود) را نابود کنید یا فقط از پنل مدیریت خارج شوید؟", | ||||||
|  |         confirm: "نابودی جلسه", | ||||||
|  |         cancel: "فقط خروج از پنل مدیریت", | ||||||
|  |       }, | ||||||
|     }, |     }, | ||||||
|     users: { |     users: { | ||||||
|       invalid_user_id: "بخش محلی یک شناسه کاربری ماتریکس بدون سرور خانگی.", |       invalid_user_id: "بخش محلی یک شناسه کاربری ماتریکس بدون سرور خانگی.", | ||||||
|   | |||||||
| @@ -13,6 +13,14 @@ const fr: SynapseTranslationMessages = { | |||||||
|       protocol_error: "L'URL doit commencer par « http:// » ou « https:// »", |       protocol_error: "L'URL doit commencer par « http:// » ou « https:// »", | ||||||
|       url_error: "L'URL du serveur Matrix n'est pas valide", |       url_error: "L'URL du serveur Matrix n'est pas valide", | ||||||
|       sso_sign_in: "Se connecter avec l’authentification unique", |       sso_sign_in: "Se connecter avec l’authentification unique", | ||||||
|  |       credentials: "Identifiants", | ||||||
|  |       access_token: "Jeton d'accès", | ||||||
|  |       logout_acces_token_dialog: { | ||||||
|  |         title: "Vous utilisez un jeton d'accès Matrix existant.", | ||||||
|  |         content: "Voulez-vous détruire cette session (qui pourrait être utilisée ailleurs, par exemple dans un client Matrix) ou simplement vous déconnecter du panneau d'administration?", | ||||||
|  |         confirm: "Détruire la session", | ||||||
|  |         cancel: "Se déconnecter simplement du panneau d'administration", | ||||||
|  |       }, | ||||||
|     }, |     }, | ||||||
|     users: { |     users: { | ||||||
|       invalid_user_id: "Partie locale d'un identifiant utilisateur Matrix sans le nom du serveur d’accueil.", |       invalid_user_id: "Partie locale d'un identifiant utilisateur Matrix sans le nom du serveur d’accueil.", | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								src/i18n/index.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								src/i18n/index.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -11,6 +11,14 @@ interface SynapseTranslationMessages extends TranslationMessages { | |||||||
|       protocol_error: string; |       protocol_error: string; | ||||||
|       url_error: string; |       url_error: string; | ||||||
|       sso_sign_in: string; |       sso_sign_in: string; | ||||||
|  |       credentials: string; | ||||||
|  |       access_token: string; | ||||||
|  |       logout_acces_token_dialog: { | ||||||
|  |         title: string; | ||||||
|  |         content: string; | ||||||
|  |         confirm: string; | ||||||
|  |         cancel: string; | ||||||
|  |       }; | ||||||
|     }; |     }; | ||||||
|     users: { |     users: { | ||||||
|       invalid_user_id: string; |       invalid_user_id: string; | ||||||
|   | |||||||
| @@ -13,6 +13,14 @@ const it: SynapseTranslationMessages = { | |||||||
|       protocol_error: "L'URL deve iniziare per 'http://' o 'https://'", |       protocol_error: "L'URL deve iniziare per 'http://' o 'https://'", | ||||||
|       url_error: "URL del server Matrix non valido", |       url_error: "URL del server Matrix non valido", | ||||||
|       sso_sign_in: "Accedi con SSO", |       sso_sign_in: "Accedi con SSO", | ||||||
|  |       credentials: "Credenziali", | ||||||
|  |       access_token: "Token di accesso", | ||||||
|  |       logout_acces_token_dialog: { | ||||||
|  |         title: "Stai utilizzando un token di accesso Matrix esistente.", | ||||||
|  |         content: "Vuoi distruggere questa sessione (che potrebbe essere utilizzata altrove, ad esempio in un client Matrix) o semplicemente disconnetterti dal pannello di amministrazione?", | ||||||
|  |         confirm: "Distruggi sessione", | ||||||
|  |         cancel: "Disconnetti solo dal pannello di amministrazione", | ||||||
|  |       }, | ||||||
|     }, |     }, | ||||||
|     users: { |     users: { | ||||||
|       invalid_user_id: "ID utente non valido su questo homeserver.", |       invalid_user_id: "ID utente non valido su questo homeserver.", | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ import russianMessages from "ra-language-russian"; | |||||||
|  |  | ||||||
| import { SynapseTranslationMessages } from "."; | import { SynapseTranslationMessages } from "."; | ||||||
|  |  | ||||||
| const ru: SynapseTranslationMessages = { | const fixedRussianMessages = { | ||||||
|   ...russianMessages, |   ...russianMessages, | ||||||
|   ra: { |   ra: { | ||||||
|     ...russianMessages.ra, |     ...russianMessages.ra, | ||||||
| @@ -11,7 +11,24 @@ const ru: SynapseTranslationMessages = { | |||||||
|       no_filtered_results: "Нет результатов", |       no_filtered_results: "Нет результатов", | ||||||
|       clear_filters: "Все фильтры сбросить", |       clear_filters: "Все фильтры сбросить", | ||||||
|     }, |     }, | ||||||
|  |     page: { | ||||||
|  |       ...russianMessages.ra.page, | ||||||
|  |       empty: "Пусто", | ||||||
|  |       access_denied: "Доступ запрещен", | ||||||
|  |       authentication_error: "Ошибка аутентификации", | ||||||
|     }, |     }, | ||||||
|  |     message: { | ||||||
|  |       ...russianMessages.ra.message, | ||||||
|  |       access_denied: | ||||||
|  |         "У вас нет прав доступа к этой странице.", | ||||||
|  |       authentication_error: | ||||||
|  |         "Сервер аутентификации вернул ошибку и не смог проверить ваши учетные данные.", | ||||||
|  |     }, | ||||||
|  |   }, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const ru: SynapseTranslationMessages = { | ||||||
|  |   ...fixedRussianMessages, | ||||||
|   synapseadmin: { |   synapseadmin: { | ||||||
|     auth: { |     auth: { | ||||||
|       base_url: "Адрес домашнего сервера", |       base_url: "Адрес домашнего сервера", | ||||||
| @@ -22,6 +39,14 @@ const ru: SynapseTranslationMessages = { | |||||||
|       protocol_error: "Адрес должен начинаться с 'http://' или 'https://'", |       protocol_error: "Адрес должен начинаться с 'http://' или 'https://'", | ||||||
|       url_error: "Неверный адрес сервера Matrix", |       url_error: "Неверный адрес сервера Matrix", | ||||||
|       sso_sign_in: "Вход через SSO", |       sso_sign_in: "Вход через SSO", | ||||||
|  |       credentials: "Учетные данные", | ||||||
|  |       access_token: "Токен доступа", | ||||||
|  |       logout_acces_token_dialog: { | ||||||
|  |         title: "Вы используете существующий токен доступа Matrix.", | ||||||
|  |         content: "Вы хотите завершить эту сессию (которая может быть использована в другом месте, например, в клиенте Matrix) или просто выйти из панели администрирования?", | ||||||
|  |         confirm: "Завершить сессию", | ||||||
|  |         cancel: "Просто выйти из панели администрирования", | ||||||
|  |       }, | ||||||
|     }, |     }, | ||||||
|     users: { |     users: { | ||||||
|       invalid_user_id: "Локальная часть ID пользователя Matrix без адреса домашнего сервера.", |       invalid_user_id: "Локальная часть ID пользователя Matrix без адреса домашнего сервера.", | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ import chineseMessages from "@haxqer/ra-language-chinese"; | |||||||
|  |  | ||||||
| import { SynapseTranslationMessages } from "."; | import { SynapseTranslationMessages } from "."; | ||||||
|  |  | ||||||
| const zh: SynapseTranslationMessages = { | const fixedChineseMessages = { | ||||||
|   ...chineseMessages, |   ...chineseMessages, | ||||||
|   ra: { |   ra: { | ||||||
|     ...chineseMessages.ra, |     ...chineseMessages.ra, | ||||||
| @@ -11,7 +11,27 @@ const zh: SynapseTranslationMessages = { | |||||||
|       no_filtered_results: "没有结果", |       no_filtered_results: "没有结果", | ||||||
|       clear_filters: "清除所有过滤器", |       clear_filters: "清除所有过滤器", | ||||||
|     }, |     }, | ||||||
|  |     action: { | ||||||
|  |       ...chineseMessages.ra.action, | ||||||
|  |       update_application: "Anwendung aktualisieren", | ||||||
|     }, |     }, | ||||||
|  |     page: { | ||||||
|  |       ...chineseMessages.ra.page, | ||||||
|  |       access_denied: "拒绝访问", | ||||||
|  |       authentication_error: "认证错误", | ||||||
|  |     }, | ||||||
|  |     message: { | ||||||
|  |       ...chineseMessages.ra.message, | ||||||
|  |       access_denied: | ||||||
|  |         "您没有访问此页面的权限。", | ||||||
|  |       authentication_error: | ||||||
|  |         "身份验证服务器返回错误,无法验证您的凭据。", | ||||||
|  |     }, | ||||||
|  |   }, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const zh: SynapseTranslationMessages = { | ||||||
|  |   ...fixedChineseMessages, | ||||||
|   synapseadmin: { |   synapseadmin: { | ||||||
|     auth: { |     auth: { | ||||||
|       base_url: "服务器 URL", |       base_url: "服务器 URL", | ||||||
| @@ -21,6 +41,14 @@ const zh: SynapseTranslationMessages = { | |||||||
|       protocol_error: "URL 需要以'http://'或'https://'作为起始", |       protocol_error: "URL 需要以'http://'或'https://'作为起始", | ||||||
|       url_error: "不是一个有效的 Matrix 服务器地址", |       url_error: "不是一个有效的 Matrix 服务器地址", | ||||||
|       sso_sign_in: "使用 SSO 登录", |       sso_sign_in: "使用 SSO 登录", | ||||||
|  |       credentials: "凭证", | ||||||
|  |       access_token: "访问令牌", | ||||||
|  |       logout_acces_token_dialog: { | ||||||
|  |         title: "您正在使用现有的 Matrix 访问令牌。", | ||||||
|  |         content: "您想销毁此会话(可能在其他地方使用,例如在 Matrix 客户端中)还是仅从管理面板退出?", | ||||||
|  |         confirm: "销毁会话", | ||||||
|  |         cancel: "仅从管理面板退出", | ||||||
|  |       }, | ||||||
|     }, |     }, | ||||||
|     users: { |     users: { | ||||||
|       invalid_user_id: "必须要是一个有效的 Matrix 用户 ID ,例如 @user_id:homeserver", |       invalid_user_id: "必须要是一个有效的 Matrix 用户 ID ,例如 @user_id:homeserver", | ||||||
|   | |||||||
| @@ -3,14 +3,33 @@ import React from "react"; | |||||||
| import { createRoot } from "react-dom/client"; | import { createRoot } from "react-dom/client"; | ||||||
|  |  | ||||||
| import App from "./App"; | import App from "./App"; | ||||||
| import { AppContext } from "./AppContext"; | import { AppContext, MenuItem } from "./AppContext"; | ||||||
| import storage from "./storage"; | import storage from "./storage"; | ||||||
|  |  | ||||||
| fetch("config.json") | fetch("config.json") | ||||||
|   .then(res => res.json()) |   .then(res => res.json()) | ||||||
|   .then(props => { |   .then(props => { | ||||||
|  |     if (props.asManagedUsers) { | ||||||
|       storage.setItem("as_managed_users", JSON.stringify(props.asManagedUsers)); |       storage.setItem("as_managed_users", JSON.stringify(props.asManagedUsers)); | ||||||
|     storage.setItem("support_url", props.supportURL); |     } | ||||||
|  |  | ||||||
|  |     let menu: MenuItem[] = []; | ||||||
|  |     if (props.menu) { | ||||||
|  |       menu = props.menu; | ||||||
|  |     } | ||||||
|  |     if (props.supportURL) { | ||||||
|  |       const migratedSupportURL = { | ||||||
|  |         label: "Contact support", | ||||||
|  |         icon: "SupportAgent", | ||||||
|  |         url: props.supportURL, | ||||||
|  |       }; | ||||||
|  |       console.warn("supportURL config option is deprecated. Please, use the menu option instead. Automatically migrated to the new menu option:", migratedSupportURL); | ||||||
|  |       menu.push(migratedSupportURL as MenuItem); | ||||||
|  |     } | ||||||
|  |     if (menu.length > 0) { | ||||||
|  |       storage.setItem("menu", JSON.stringify(menu)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     return createRoot(document.getElementById("root")).render( |     return createRoot(document.getElementById("root")).render( | ||||||
|       <React.StrictMode> |       <React.StrictMode> | ||||||
|         <AppContext.Provider value={props}> |         <AppContext.Provider value={props}> | ||||||
|   | |||||||
| @@ -1,8 +1,7 @@ | |||||||
| import { useState, useEffect } from "react"; | import { useState, useEffect } from "react"; | ||||||
|  |  | ||||||
| import LockIcon from "@mui/icons-material/Lock"; | import LockIcon from "@mui/icons-material/Lock"; | ||||||
| import { Avatar, Box, Button, Card, CardActions, CircularProgress, MenuItem, Select, Typography } from "@mui/material"; | import { Avatar, Box, Button, Card, CardActions, CircularProgress, MenuItem, Select, Tab, Tabs, Typography } from "@mui/material"; | ||||||
| import { styled } from "@mui/material/styles"; |  | ||||||
| import { | import { | ||||||
|   Form, |   Form, | ||||||
|   FormDataConsumer, |   FormDataConsumer, | ||||||
| @@ -17,7 +16,7 @@ import { | |||||||
|   useLocales, |   useLocales, | ||||||
| } from "react-admin"; | } from "react-admin"; | ||||||
| import { useFormContext } from "react-hook-form"; | import { useFormContext } from "react-hook-form"; | ||||||
|  | import LoginFormBox from "../components/LoginFormBox"; | ||||||
| import { useAppContext } from "../AppContext"; | import { useAppContext } from "../AppContext"; | ||||||
| import { | import { | ||||||
|   getServerVersion, |   getServerVersion, | ||||||
| @@ -29,66 +28,18 @@ import { | |||||||
| } from "../synapse/synapse"; | } from "../synapse/synapse"; | ||||||
| import storage from "../storage"; | import storage from "../storage"; | ||||||
|  |  | ||||||
| const FormBox = styled(Box)(({ theme }) => ({ | export type LoginMethod = "credentials" | "accessToken"; | ||||||
|   display: "flex", |  | ||||||
|   flexDirection: "column", |  | ||||||
|   minHeight: "calc(100vh - 1rem)", |  | ||||||
|   alignItems: "center", |  | ||||||
|   justifyContent: "flex-start", |  | ||||||
|   background: "url(./images/floating-cogs.svg)", |  | ||||||
|   backgroundColor: "#f9f9f9", |  | ||||||
|   backgroundRepeat: "no-repeat", |  | ||||||
|   backgroundSize: "cover", |  | ||||||
|  |  | ||||||
|   [`& .card`]: { |  | ||||||
|     width: "30rem", |  | ||||||
|     marginTop: "6rem", |  | ||||||
|     marginBottom: "6rem", |  | ||||||
|   }, |  | ||||||
|   [`& .avatar`]: { |  | ||||||
|     margin: "1rem", |  | ||||||
|     display: "flex", |  | ||||||
|     justifyContent: "center", |  | ||||||
|   }, |  | ||||||
|   [`& .icon`]: { |  | ||||||
|     backgroundColor: theme.palette.grey[500], |  | ||||||
|   }, |  | ||||||
|   [`& .hint`]: { |  | ||||||
|     marginTop: "1em", |  | ||||||
|     marginBottom: "1em", |  | ||||||
|     display: "flex", |  | ||||||
|     justifyContent: "center", |  | ||||||
|     color: theme.palette.grey[600], |  | ||||||
|   }, |  | ||||||
|   [`& .form`]: { |  | ||||||
|     padding: "0 1rem 1rem 1rem", |  | ||||||
|   }, |  | ||||||
|   [`& .select`]: { |  | ||||||
|     marginBottom: "2rem", |  | ||||||
|   }, |  | ||||||
|   [`& .actions`]: { |  | ||||||
|     padding: "0 1rem 1rem 1rem", |  | ||||||
|   }, |  | ||||||
|   [`& .serverVersion`]: { |  | ||||||
|     color: theme.palette.grey[500], |  | ||||||
|     fontFamily: "Roboto, Helvetica, Arial, sans-serif", |  | ||||||
|     marginLeft: "0.5rem", |  | ||||||
|   }, |  | ||||||
|   [`& .matrixVersions`]: { |  | ||||||
|     color: theme.palette.grey[500], |  | ||||||
|     fontFamily: "Roboto, Helvetica, Arial, sans-serif", |  | ||||||
|     fontSize: "0.8rem", |  | ||||||
|     marginBottom: "1rem", |  | ||||||
|     marginLeft: "0.5rem", |  | ||||||
|   }, |  | ||||||
| })); |  | ||||||
|  |  | ||||||
| const LoginPage = () => { | const LoginPage = () => { | ||||||
|   const login = useLogin(); |   const login = useLogin(); | ||||||
|   const notify = useNotify(); |   const notify = useNotify(); | ||||||
|   const { restrictBaseUrl } = useAppContext(); |   const { restrictBaseUrl } = useAppContext(); | ||||||
|   const allowSingleBaseUrl = typeof restrictBaseUrl === "string"; |   const allowSingleBaseUrl = typeof restrictBaseUrl === "string"; | ||||||
|   const allowMultipleBaseUrls = Array.isArray(restrictBaseUrl); |   const allowMultipleBaseUrls = | ||||||
|  |     Array.isArray(restrictBaseUrl) && | ||||||
|  |     restrictBaseUrl.length > 0 && | ||||||
|  |     restrictBaseUrl[0] !== "" && | ||||||
|  |     restrictBaseUrl[0] !== null; | ||||||
|   const allowAnyBaseUrl = !(allowSingleBaseUrl || allowMultipleBaseUrls); |   const allowAnyBaseUrl = !(allowSingleBaseUrl || allowMultipleBaseUrls); | ||||||
|   const [loading, setLoading] = useState(false); |   const [loading, setLoading] = useState(false); | ||||||
|   const [supportPassAuth, setSupportPassAuth] = useState(true); |   const [supportPassAuth, setSupportPassAuth] = useState(true); | ||||||
| @@ -98,8 +49,13 @@ const LoginPage = () => { | |||||||
|   const base_url = allowSingleBaseUrl ? restrictBaseUrl : storage.getItem("base_url"); |   const base_url = allowSingleBaseUrl ? restrictBaseUrl : storage.getItem("base_url"); | ||||||
|   const [ssoBaseUrl, setSSOBaseUrl] = useState(""); |   const [ssoBaseUrl, setSSOBaseUrl] = useState(""); | ||||||
|   const loginToken = /\?loginToken=([a-zA-Z0-9_-]+)/.exec(window.location.href); |   const loginToken = /\?loginToken=([a-zA-Z0-9_-]+)/.exec(window.location.href); | ||||||
|  |   const [loginMethod, setLoginMethod] = useState<LoginMethod>("credentials"); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     if (!loginToken) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|   if (loginToken) { |  | ||||||
|     const ssoToken = loginToken[1]; |     const ssoToken = loginToken[1]; | ||||||
|     console.log("SSO token is", ssoToken); |     console.log("SSO token is", ssoToken); | ||||||
|     // Prevent further requests |     // Prevent further requests | ||||||
| @@ -127,7 +83,7 @@ const LoginPage = () => { | |||||||
|         console.error(error); |         console.error(error); | ||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
|   } |   }, [loginToken]); | ||||||
|  |  | ||||||
|   const validateBaseUrl = value => { |   const validateBaseUrl = value => { | ||||||
|     if (!value.match(/^(http|https):\/\//)) { |     if (!value.match(/^(http|https):\/\//)) { | ||||||
| @@ -212,6 +168,18 @@ const LoginPage = () => { | |||||||
|     }, [formData.base_url, form]); |     }, [formData.base_url, form]); | ||||||
|  |  | ||||||
|     return ( |     return ( | ||||||
|  |       <> | ||||||
|  |         <Tabs | ||||||
|  |           value={loginMethod} | ||||||
|  |           onChange={(_, newValue) => setLoginMethod(newValue as LoginMethod)} | ||||||
|  |           indicatorColor="primary" | ||||||
|  |           textColor="primary" | ||||||
|  |           centered | ||||||
|  |         > | ||||||
|  |           <Tab label={translate("synapseadmin.auth.credentials")} value="credentials" /> | ||||||
|  |           <Tab label={translate("synapseadmin.auth.access_token")} value="accessToken" /> | ||||||
|  |         </Tabs> | ||||||
|  |         {loginMethod === "credentials" ? ( | ||||||
|           <> |           <> | ||||||
|             <Box> |             <Box> | ||||||
|               <TextInput |               <TextInput | ||||||
| @@ -236,6 +204,18 @@ const LoginPage = () => { | |||||||
|                 validate={required()} |                 validate={required()} | ||||||
|               /> |               /> | ||||||
|             </Box> |             </Box> | ||||||
|  |           </> | ||||||
|  |         ) : ( | ||||||
|  |           <Box> | ||||||
|  |             <TextInput | ||||||
|  |               source="accessToken" | ||||||
|  |               label="synapseadmin.auth.access_token" | ||||||
|  |               disabled={loading} | ||||||
|  |               resettable | ||||||
|  |               validate={required()} | ||||||
|  |             /> | ||||||
|  |           </Box> | ||||||
|  |         )} | ||||||
|         <Box> |         <Box> | ||||||
|           <TextInput |           <TextInput | ||||||
|             source="base_url" |             source="base_url" | ||||||
| @@ -263,7 +243,7 @@ const LoginPage = () => { | |||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <Form defaultValues={{ base_url: base_url }} onSubmit={handleSubmit} mode="onTouched"> |     <Form defaultValues={{ base_url: base_url }} onSubmit={handleSubmit} mode="onTouched"> | ||||||
|       <FormBox> |       <LoginFormBox> | ||||||
|         <Card className="card"> |         <Card className="card"> | ||||||
|           <Box className="avatar"> |           <Box className="avatar"> | ||||||
|             {loading ? ( |             {loading ? ( | ||||||
| @@ -290,7 +270,7 @@ const LoginPage = () => { | |||||||
|               ))} |               ))} | ||||||
|             </Select> |             </Select> | ||||||
|             <FormDataConsumer>{formDataProps => <UserData {...formDataProps} />}</FormDataConsumer> |             <FormDataConsumer>{formDataProps => <UserData {...formDataProps} />}</FormDataConsumer> | ||||||
|             <CardActions className="actions"> |             {loginMethod === "credentials" && <CardActions className="actions"> | ||||||
|               <Button |               <Button | ||||||
|                 variant="contained" |                 variant="contained" | ||||||
|                 type="submit" |                 type="submit" | ||||||
| @@ -309,10 +289,21 @@ const LoginPage = () => { | |||||||
|               > |               > | ||||||
|                 {translate("synapseadmin.auth.sso_sign_in")} |                 {translate("synapseadmin.auth.sso_sign_in")} | ||||||
|               </Button> |               </Button> | ||||||
|             </CardActions> |             </CardActions>} | ||||||
|  |             {loginMethod === "accessToken" && <CardActions className="actions"> | ||||||
|  |               <Button | ||||||
|  |                 variant="contained" | ||||||
|  |                 type="submit" | ||||||
|  |                 color="primary" | ||||||
|  |                 disabled={loading} | ||||||
|  |                 fullWidth | ||||||
|  |               > | ||||||
|  |                 {translate("ra.auth.sign_in")} | ||||||
|  |               </Button> | ||||||
|  |             </CardActions>} | ||||||
|           </Box> |           </Box> | ||||||
|         </Card> |         </Card> | ||||||
|       </FormBox> |       </LoginFormBox> | ||||||
|       <Notification /> |       <Notification /> | ||||||
|     </Form> |     </Form> | ||||||
|   ); |   ); | ||||||
|   | |||||||
| @@ -4,9 +4,11 @@ import AutorenewIcon from "@mui/icons-material/Autorenew"; | |||||||
| import DestinationsIcon from "@mui/icons-material/CloudQueue"; | import DestinationsIcon from "@mui/icons-material/CloudQueue"; | ||||||
| import FolderSharedIcon from "@mui/icons-material/FolderShared"; | import FolderSharedIcon from "@mui/icons-material/FolderShared"; | ||||||
| import ViewListIcon from "@mui/icons-material/ViewList"; | import ViewListIcon from "@mui/icons-material/ViewList"; | ||||||
|  | import ErrorIcon from '@mui/icons-material/Error'; | ||||||
| import { | import { | ||||||
|   Button, |   Button, | ||||||
|   Datagrid, |   Datagrid, | ||||||
|  |   DatagridConfigurable, | ||||||
|   DateField, |   DateField, | ||||||
|   List, |   List, | ||||||
|   ListProps, |   ListProps, | ||||||
| @@ -21,6 +23,7 @@ import { | |||||||
|   Tab, |   Tab, | ||||||
|   TabbedShowLayout, |   TabbedShowLayout, | ||||||
|   TextField, |   TextField, | ||||||
|  |   FunctionField, | ||||||
|   TopToolbar, |   TopToolbar, | ||||||
|   useRecordContext, |   useRecordContext, | ||||||
|   useDelete, |   useDelete, | ||||||
| @@ -35,13 +38,6 @@ import { get } from "lodash"; | |||||||
|  |  | ||||||
| const DestinationPagination = () => <Pagination rowsPerPageOptions={[10, 25, 50, 100, 500, 1000]} />; | const DestinationPagination = () => <Pagination rowsPerPageOptions={[10, 25, 50, 100, 500, 1000]} />; | ||||||
|  |  | ||||||
| const destinationRowSx = (record: RaRecord) => ({ |  | ||||||
|   backgroundColor: record.retry_last_ts > 0 ? "warning.light" : "primary.contrastText", |  | ||||||
|   "& .MuiButtonBase-root": { |  | ||||||
|     color: "primary.dark", |  | ||||||
|   }, |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| const destinationFilters = [<SearchInput source="destination" alwaysOn />]; | const destinationFilters = [<SearchInput source="destination" alwaysOn />]; | ||||||
|  |  | ||||||
| export const DestinationReconnectButton = () => { | export const DestinationReconnectButton = () => { | ||||||
| @@ -105,7 +101,22 @@ const RetryDateField = (props: DateFieldProps) => { | |||||||
|   return <DateField {...props} />; |   return <DateField {...props} />; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | const destinationFieldRender = (record: RaRecord) => { | ||||||
|  |   if (record.retry_last_ts > 0) { | ||||||
|  |     return ( | ||||||
|  |       <> | ||||||
|  |         <ErrorIcon fontSize="inherit" color="error" sx={{verticalAlign: "middle"}}/> | ||||||
|  |  | ||||||
|  |         {record.destination} | ||||||
|  |       </> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |   return <> {record.destination} </>; | ||||||
|  | } | ||||||
|  |  | ||||||
| export const DestinationList = (props: ListProps) => { | export const DestinationList = (props: ListProps) => { | ||||||
|  |   const record = useRecordContext(props); | ||||||
|   return ( |   return ( | ||||||
|     <List |     <List | ||||||
|       {...props} |       {...props} | ||||||
| @@ -113,14 +124,14 @@ export const DestinationList = (props: ListProps) => { | |||||||
|       pagination={<DestinationPagination />} |       pagination={<DestinationPagination />} | ||||||
|       sort={{ field: "destination", order: "ASC" }} |       sort={{ field: "destination", order: "ASC" }} | ||||||
|     > |     > | ||||||
|       <Datagrid rowSx={destinationRowSx} rowClick={id => `${id}/show/rooms`} bulkActionButtons={false}> |       <DatagridConfigurable rowClick={id => `${id}/show/rooms`} bulkActionButtons={false}> | ||||||
|         <TextField source="destination" /> |         <FunctionField source="destination" render={destinationFieldRender} /> | ||||||
|         <DateField source="failure_ts" showTime options={DATE_FORMAT} /> |         <DateField source="failure_ts" showTime options={DATE_FORMAT} /> | ||||||
|         <RetryDateField source="retry_last_ts" showTime options={DATE_FORMAT} /> |         <RetryDateField source="retry_last_ts" showTime options={DATE_FORMAT} /> | ||||||
|         <TextField source="retry_interval" /> |         <TextField source="retry_interval" /> | ||||||
|         <TextField source="last_successful_stream_ordering" /> |         <TextField source="last_successful_stream_ordering" /> | ||||||
|         <DestinationReconnectButton /> |         <DestinationReconnectButton /> | ||||||
|       </Datagrid> |       </DatagridConfigurable> | ||||||
|     </List> |     </List> | ||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ import { | |||||||
|   Create, |   Create, | ||||||
|   CreateProps, |   CreateProps, | ||||||
|   Datagrid, |   Datagrid, | ||||||
|  |   DatagridConfigurable, | ||||||
|   DateField, |   DateField, | ||||||
|   DateTimeInput, |   DateTimeInput, | ||||||
|   Edit, |   Edit, | ||||||
| @@ -39,13 +40,13 @@ export const RegistrationTokenList = (props: ListProps) => ( | |||||||
|     pagination={false} |     pagination={false} | ||||||
|     perPage={500} |     perPage={500} | ||||||
|   > |   > | ||||||
|     <Datagrid rowClick="edit"> |     <DatagridConfigurable rowClick="edit"> | ||||||
|       <TextField source="token" sortable={false} /> |       <TextField source="token" sortable={false} /> | ||||||
|       <NumberField source="uses_allowed" sortable={false} /> |       <NumberField source="uses_allowed" sortable={false} /> | ||||||
|       <NumberField source="pending" sortable={false} /> |       <NumberField source="pending" sortable={false} /> | ||||||
|       <NumberField source="completed" sortable={false} /> |       <NumberField source="completed" sortable={false} /> | ||||||
|       <DateField source="expiry_time" showTime options={DATE_FORMAT} sortable={false} /> |       <DateField source="expiry_time" showTime options={DATE_FORMAT} sortable={false} /> | ||||||
|     </Datagrid> |     </DatagridConfigurable> | ||||||
|   </List> |   </List> | ||||||
| ); | ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ import ViewListIcon from "@mui/icons-material/ViewList"; | |||||||
| import ReportIcon from "@mui/icons-material/Warning"; | import ReportIcon from "@mui/icons-material/Warning"; | ||||||
| import { | import { | ||||||
|   Datagrid, |   Datagrid, | ||||||
|  |   DatagridConfigurable, | ||||||
|   DateField, |   DateField, | ||||||
|   DeleteButton, |   DeleteButton, | ||||||
|   List, |   List, | ||||||
| @@ -90,13 +91,13 @@ const ReportShowActions = () => { | |||||||
|  |  | ||||||
| export const ReportList = (props: ListProps) => ( | export const ReportList = (props: ListProps) => ( | ||||||
|   <List {...props} pagination={<ReportPagination />} sort={{ field: "received_ts", order: "DESC" }}> |   <List {...props} pagination={<ReportPagination />} sort={{ field: "received_ts", order: "DESC" }}> | ||||||
|     <Datagrid rowClick="show" bulkActionButtons={false}> |     <DatagridConfigurable rowClick="show" bulkActionButtons={false}> | ||||||
|       <TextField source="id" sortable={false} /> |       <TextField source="id" sortable={false} /> | ||||||
|       <DateField source="received_ts" showTime options={DATE_FORMAT} sortable={true} /> |       <DateField source="received_ts" showTime options={DATE_FORMAT} sortable={true} /> | ||||||
|       <TextField sortable={false} source="user_id" /> |       <TextField sortable={false} source="user_id" /> | ||||||
|       <TextField sortable={false} source="name" /> |       <TextField sortable={false} source="name" /> | ||||||
|       <TextField sortable={false} source="score" /> |       <TextField sortable={false} source="score" /> | ||||||
|     </Datagrid> |     </DatagridConfigurable> | ||||||
|   </List> |   </List> | ||||||
| ); | ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| import PermMediaIcon from "@mui/icons-material/PermMedia"; | import PermMediaIcon from "@mui/icons-material/PermMedia"; | ||||||
| import { | import { | ||||||
|   Datagrid, |   Datagrid, | ||||||
|  |   DatagridConfigurable, | ||||||
|   ExportButton, |   ExportButton, | ||||||
|   List, |   List, | ||||||
|   ListProps, |   ListProps, | ||||||
| @@ -37,12 +38,12 @@ export const UserMediaStatsList = (props: ListProps) => ( | |||||||
|     pagination={<UserMediaStatsPagination />} |     pagination={<UserMediaStatsPagination />} | ||||||
|     sort={{ field: "media_length", order: "DESC" }} |     sort={{ field: "media_length", order: "DESC" }} | ||||||
|   > |   > | ||||||
|     <Datagrid rowClick={id => "/users/" + id + "/media"} bulkActionButtons={false}> |     <DatagridConfigurable rowClick={id => "/users/" + id + "/media"} bulkActionButtons={false}> | ||||||
|       <TextField source="user_id" label="resources.users.fields.id" /> |       <TextField source="user_id" label="resources.users.fields.id" /> | ||||||
|       <TextField source="displayname" label="resources.users.fields.displayname" /> |       <TextField source="displayname" label="resources.users.fields.displayname" /> | ||||||
|       <NumberField source="media_count" /> |       <NumberField source="media_count" /> | ||||||
|       <NumberField source="media_length" /> |       <NumberField source="media_length" /> | ||||||
|     </Datagrid> |     </DatagridConfigurable> | ||||||
|   </List> |   </List> | ||||||
| ); | ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -15,6 +15,7 @@ import { | |||||||
|   ArrayField, |   ArrayField, | ||||||
|   Button, |   Button, | ||||||
|   Datagrid, |   Datagrid, | ||||||
|  |   DatagridConfigurable, | ||||||
|   DateField, |   DateField, | ||||||
|   Create, |   Create, | ||||||
|   CreateProps, |   CreateProps, | ||||||
| @@ -51,8 +52,8 @@ import { | |||||||
|   NumberField, |   NumberField, | ||||||
|   useListContext, |   useListContext, | ||||||
|   useNotify, |   useNotify, | ||||||
|   ToolbarClasses, |  | ||||||
|   Identifier, |   Identifier, | ||||||
|  |   ToolbarClasses, | ||||||
|   RaRecord, |   RaRecord, | ||||||
|   ImageInput, |   ImageInput, | ||||||
|   ImageField, |   ImageField, | ||||||
| @@ -147,10 +148,6 @@ const UserBulkActionButtons = () => { | |||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const usersRowClick = (id: Identifier, resource: string, record: RaRecord): string => { |  | ||||||
|   return `/users/${id}`; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export const UserList = (props: ListProps) => ( | export const UserList = (props: ListProps) => ( | ||||||
|   <List |   <List | ||||||
|     {...props} |     {...props} | ||||||
| @@ -160,8 +157,11 @@ export const UserList = (props: ListProps) => ( | |||||||
|     actions={<UserListActions />} |     actions={<UserListActions />} | ||||||
|     pagination={<UserPagination />} |     pagination={<UserPagination />} | ||||||
|   > |   > | ||||||
|     <Datagrid rowClick={usersRowClick} bulkActionButtons={<UserBulkActionButtons />}> |     <DatagridConfigurable | ||||||
|       <AvatarField source="avatar_src" sx={{ height: "40px", width: "40px" }} /> |       rowClick={(id: Identifier, resource: string) => `/${resource}/${id}`} | ||||||
|  |       bulkActionButtons={<UserBulkActionButtons />} | ||||||
|  |     > | ||||||
|  |       <AvatarField source="avatar_src" sx={{ height: "40px", width: "40px" }} sortBy="avatar_url" /> | ||||||
|       <TextField source="id" sortBy="name" /> |       <TextField source="id" sortBy="name" /> | ||||||
|       <TextField source="displayname" /> |       <TextField source="displayname" /> | ||||||
|       <BooleanField source="is_guest" /> |       <BooleanField source="is_guest" /> | ||||||
| @@ -170,7 +170,7 @@ export const UserList = (props: ListProps) => ( | |||||||
|       <BooleanField source="locked" /> |       <BooleanField source="locked" /> | ||||||
|       <BooleanField source="erased" sortable={false} /> |       <BooleanField source="erased" sortable={false} /> | ||||||
|       <DateField source="creation_ts" label="resources.users.fields.creation_ts_ms" showTime options={DATE_FORMAT} /> |       <DateField source="creation_ts" label="resources.users.fields.creation_ts_ms" showTime options={DATE_FORMAT} /> | ||||||
|     </Datagrid> |     </DatagridConfigurable> | ||||||
|   </List> |   </List> | ||||||
| ); | ); | ||||||
|  |  | ||||||
| @@ -212,9 +212,7 @@ const UserEditActions = () => { | |||||||
| export const UserCreate = (props: CreateProps) => ( | export const UserCreate = (props: CreateProps) => ( | ||||||
|   <Create |   <Create | ||||||
|     {...props} |     {...props} | ||||||
|     redirect={(resource, id, data) => { |     redirect={(resource: string | undefined, id: Identifier | undefined) => `${resource}/${id}`} | ||||||
|       return `users/${id}`; |  | ||||||
|     }} |  | ||||||
|   > |   > | ||||||
|     <SimpleForm> |     <SimpleForm> | ||||||
|       <TextInput source="id" autoComplete="off" validate={validateUser} /> |       <TextInput source="id" autoComplete="off" validate={validateUser} /> | ||||||
|   | |||||||
| @@ -23,14 +23,14 @@ describe("authProvider", () => { | |||||||
|         }) |         }) | ||||||
|       ); |       ); | ||||||
|  |  | ||||||
|       const ret: undefined = await authProvider.login({ |       const ret = await authProvider.login({ | ||||||
|         base_url: "http://example.com", |         base_url: "http://example.com", | ||||||
|         username: "@user:example.com", |         username: "@user:example.com", | ||||||
|         password: "secret", |         password: "secret", | ||||||
|       }); |       }); | ||||||
|  |  | ||||||
|       expect(ret).toBe(undefined); |       expect(ret).toEqual({redirectTo: "/"}); | ||||||
|       expect(fetch).toBeCalledWith("http://example.com/_matrix/client/r0/login", { |       expect(fetch).toHaveBeenCalledWith("http://example.com/_matrix/client/v3/login", { | ||||||
|         body: '{"device_id":null,"initial_device_display_name":"Synapse Admin","type":"m.login.password","identifier":{"type":"m.id.user","user":"@user:example.com"},"password":"secret"}', |         body: '{"device_id":null,"initial_device_display_name":"Synapse Admin","type":"m.login.password","identifier":{"type":"m.id.user","user":"@user:example.com"},"password":"secret"}', | ||||||
|         headers: new Headers({ |         headers: new Headers({ | ||||||
|           Accept: "application/json", |           Accept: "application/json", | ||||||
| @@ -55,13 +55,13 @@ describe("authProvider", () => { | |||||||
|       }) |       }) | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|     const ret: undefined = await authProvider.login({ |     const ret = await authProvider.login({ | ||||||
|       base_url: "https://example.com/", |       base_url: "https://example.com/", | ||||||
|       loginToken: "login_token", |       loginToken: "login_token", | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     expect(ret).toBe(undefined); |     expect(ret).toEqual({redirectTo: "/"}); | ||||||
|     expect(fetch).toHaveBeenCalledWith("https://example.com/_matrix/client/r0/login", { |     expect(fetch).toHaveBeenCalledWith("https://example.com/_matrix/client/v3/login", { | ||||||
|       body: '{"device_id":null,"initial_device_display_name":"Synapse Admin","type":"m.login.token","token":"login_token"}', |       body: '{"device_id":null,"initial_device_display_name":"Synapse Admin","type":"m.login.token","token":"login_token"}', | ||||||
|       headers: new Headers({ |       headers: new Headers({ | ||||||
|         Accept: "application/json", |         Accept: "application/json", | ||||||
| @@ -83,7 +83,7 @@ describe("authProvider", () => { | |||||||
|  |  | ||||||
|       await authProvider.logout(null); |       await authProvider.logout(null); | ||||||
|  |  | ||||||
|       expect(fetch).toBeCalledWith("example.com/_matrix/client/r0/logout", { |       expect(fetch).toHaveBeenCalledWith("example.com/_matrix/client/v3/logout", { | ||||||
|         headers: new Headers({ |         headers: new Headers({ | ||||||
|           Accept: "application/json", |           Accept: "application/json", | ||||||
|           Authorization: "Bearer foo", |           Authorization: "Bearer foo", | ||||||
| @@ -123,7 +123,9 @@ describe("authProvider", () => { | |||||||
|  |  | ||||||
|   describe("getPermissions", () => { |   describe("getPermissions", () => { | ||||||
|     it("should do nothing", async () => { |     it("should do nothing", async () => { | ||||||
|  |       if (authProvider.getPermissions) { | ||||||
|         await expect(authProvider.getPermissions(null)).resolves.toBeUndefined(); |         await expect(authProvider.getPermissions(null)).resolves.toBeUndefined(); | ||||||
|  |       } | ||||||
|     }); |     }); | ||||||
|   }); |   }); | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -1,7 +1,8 @@ | |||||||
| import { AuthProvider, HttpError, Options, fetchUtils, useTranslate } from "react-admin"; | import { AuthProvider, HttpError, Options, fetchUtils } from "react-admin"; | ||||||
|  |  | ||||||
| import storage from "../storage"; | import storage from "../storage"; | ||||||
| import { MatrixError, displayError } from "../components/error"; | import { MatrixError, displayError } from "../components/error"; | ||||||
|  | import { fetchAuthenticatedMedia } from "../utils/fetchMedia"; | ||||||
|  |  | ||||||
| const authProvider: AuthProvider = { | const authProvider: AuthProvider = { | ||||||
|   // called when the user attempts to log in |   // called when the user attempts to log in | ||||||
| @@ -10,14 +11,16 @@ const authProvider: AuthProvider = { | |||||||
|     username, |     username, | ||||||
|     password, |     password, | ||||||
|     loginToken, |     loginToken, | ||||||
|  |     accessToken, | ||||||
|   }: { |   }: { | ||||||
|     base_url: string; |     base_url: string; | ||||||
|     username: string; |     username: string; | ||||||
|     password: string; |     password: string; | ||||||
|     loginToken: string; |     loginToken: string; | ||||||
|  |     accessToken: string; | ||||||
|   }) => { |   }) => { | ||||||
|     console.log("login "); |     console.log("login "); | ||||||
|     const options: Options = { |     let options: Options = { | ||||||
|       method: "POST", |       method: "POST", | ||||||
|       body: JSON.stringify( |       body: JSON.stringify( | ||||||
|         Object.assign( |         Object.assign( | ||||||
| @@ -55,11 +58,30 @@ const authProvider: AuthProvider = { | |||||||
|     storage.setItem("base_url", base_url); |     storage.setItem("base_url", base_url); | ||||||
|  |  | ||||||
|     const decoded_base_url = window.decodeURIComponent(base_url); |     const decoded_base_url = window.decodeURIComponent(base_url); | ||||||
|     const login_api_url = decoded_base_url + "/_matrix/client/r0/login"; |     let login_api_url = decoded_base_url + (accessToken ? "/_matrix/client/v3/account/whoami" : "/_matrix/client/v3/login"); | ||||||
|  |  | ||||||
|     let response; |     let response; | ||||||
|  |  | ||||||
|     try { |     try { | ||||||
|  |       if (accessToken) { | ||||||
|  |         // this a login with an already obtained access token, let's just validate it | ||||||
|  |         options = { | ||||||
|  |           headers: new Headers({ | ||||||
|  |             Accept: 'application/json', | ||||||
|  |             Authorization: `Bearer ${accessToken}`, | ||||||
|  |           }), | ||||||
|  |         }; | ||||||
|  |       } | ||||||
|  |  | ||||||
|       response = await fetchUtils.fetchJson(login_api_url, options); |       response = await fetchUtils.fetchJson(login_api_url, options); | ||||||
|  |       const json = response.json; | ||||||
|  |       storage.setItem("home_server", accessToken ? base_url : json.home_server); | ||||||
|  |       storage.setItem("user_id", json.user_id); | ||||||
|  |       storage.setItem("access_token", accessToken ? accessToken : json.access_token); | ||||||
|  |       storage.setItem("device_id", json.device_id); | ||||||
|  |       storage.setItem("login_type", accessToken ? "accessToken" : "credentials"); | ||||||
|  |  | ||||||
|  |       return Promise.resolve({redirectTo: "/"}); | ||||||
|     } catch(err) { |     } catch(err) { | ||||||
|       const error = err as HttpError; |       const error = err as HttpError; | ||||||
|       const errorStatus = error.status; |       const errorStatus = error.status; | ||||||
| @@ -73,18 +95,49 @@ const authProvider: AuthProvider = { | |||||||
|         ) |         ) | ||||||
|       ); |       ); | ||||||
|     } |     } | ||||||
|  |   }, | ||||||
|  |   getIdentity: async () => { | ||||||
|  |     const access_token = storage.getItem("access_token"); | ||||||
|  |     const user_id = storage.getItem("user_id"); | ||||||
|  |     const base_url = storage.getItem("base_url"); | ||||||
|  |  | ||||||
|     const json = response.json; |     if (typeof access_token !== "string" || typeof user_id !== "string" || typeof base_url !== "string") { | ||||||
|     storage.setItem("home_server", json.home_server); |       return Promise.reject(); | ||||||
|     storage.setItem("user_id", json.user_id); |     } | ||||||
|     storage.setItem("access_token", json.access_token); |  | ||||||
|     storage.setItem("device_id", json.device_id); |     const options: Options = { | ||||||
|  |       headers: new Headers({ | ||||||
|  |         Accept: "application/json", | ||||||
|  |         Authorization: `Bearer ${access_token}`, | ||||||
|  |       }), | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const whoami_api_url = base_url + `/_matrix/client/v3/profile/${user_id}`; | ||||||
|  |  | ||||||
|  |     try { | ||||||
|  |       let avatar_url = ""; | ||||||
|  |       const response = await fetchUtils.fetchJson(whoami_api_url, options); | ||||||
|  |       if (response.json.avatar_url) { | ||||||
|  |         const mediaresp = await fetchAuthenticatedMedia(response.json.avatar_url, "thumbnail"); | ||||||
|  |         const blob = await mediaresp.blob(); | ||||||
|  |         avatar_url = URL.createObjectURL(blob); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       return Promise.resolve({ | ||||||
|  |         id: user_id, | ||||||
|  |         fullName: response.json.displayname, | ||||||
|  |         avatar: avatar_url, | ||||||
|  |       }); | ||||||
|  |     } catch (err) { | ||||||
|  |       console.log("Error getting identity", err); | ||||||
|  |       return Promise.reject(); | ||||||
|  |     } | ||||||
|   }, |   }, | ||||||
|   // called when the user clicks on the logout button |   // called when the user clicks on the logout button | ||||||
|   logout: async () => { |   logout: async () => { | ||||||
|     console.log("logout"); |     console.log("logout"); | ||||||
|  |  | ||||||
|     const logout_api_url = storage.getItem("base_url") + "/_matrix/client/r0/logout"; |     const logout_api_url = storage.getItem("base_url") + "/_matrix/client/v3/logout"; | ||||||
|     const access_token = storage.getItem("access_token"); |     const access_token = storage.getItem("access_token"); | ||||||
|  |  | ||||||
|     const options: Options = { |     const options: Options = { | ||||||
| @@ -102,6 +155,7 @@ const authProvider: AuthProvider = { | |||||||
|         console.log("Error logging out", err); |         console.log("Error logging out", err); | ||||||
|       } finally { |       } finally { | ||||||
|         storage.removeItem("access_token"); |         storage.removeItem("access_token"); | ||||||
|  |         storage.removeItem("login_type"); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   | |||||||
							
								
								
									
										560
									
								
								yarn.lock
									
									
									
									
									
								
							
							
						
						
									
										560
									
								
								yarn.lock
									
									
									
									
									
								
							| @@ -23,33 +23,41 @@ | |||||||
|     "@babel/highlight" "^7.24.7" |     "@babel/highlight" "^7.24.7" | ||||||
|     picocolors "^1.0.0" |     picocolors "^1.0.0" | ||||||
|  |  | ||||||
| "@babel/compat-data@^7.25.2": | "@babel/code-frame@^7.25.7": | ||||||
|   version "7.25.4" |   version "7.25.7" | ||||||
|   resolved "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.25.4.tgz" |   resolved "https://registry.yarnpkg.com/@babel/code-frame/-/code-frame-7.25.7.tgz#438f2c524071531d643c6f0188e1e28f130cebc7" | ||||||
|   integrity sha512-+LGRog6RAsCJrrrg/IO6LGmpphNe5DiK30dGjCoxxeGv49B10/3XYGxPsAwrDlMFcFEvdAUavDT8r9k/hSyQqQ== |   integrity sha512-0xZJFNE5XMpENsgfHYTw8FbX4kv53mFLn2i3XPoq69LyhYSCBJtitaHx9QnsVTrsogI4Z3+HtEfZ2/GFPOtf5g== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/highlight" "^7.25.7" | ||||||
|  |     picocolors "^1.0.0" | ||||||
|  |  | ||||||
| "@babel/core@^7.11.6", "@babel/core@^7.12.3", "@babel/core@^7.23.9", "@babel/core@^7.24.5": | "@babel/compat-data@^7.25.7": | ||||||
|   version "7.25.2" |   version "7.25.8" | ||||||
|   resolved "https://registry.npmjs.org/@babel/core/-/core-7.25.2.tgz" |   resolved "https://registry.yarnpkg.com/@babel/compat-data/-/compat-data-7.25.8.tgz#0376e83df5ab0eb0da18885c0140041f0747a402" | ||||||
|   integrity sha512-BBt3opiCOxUr9euZ5/ro/Xv8/V7yJ5bjYMqG/C1YAo8MIKAnumZalCN+msbci3Pigy4lIQfPUpfMM27HMGaYEA== |   integrity sha512-ZsysZyXY4Tlx+Q53XdnOFmqwfB9QDTHYxaZYajWRoBLuLEAwI2UIbtxOjWh/cFaa9IKUlcB+DDuoskLuKu56JA== | ||||||
|  |  | ||||||
|  | "@babel/core@^7.11.6", "@babel/core@^7.12.3", "@babel/core@^7.23.9", "@babel/core@^7.25.2": | ||||||
|  |   version "7.25.8" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@babel/core/-/core-7.25.8.tgz#a57137d2a51bbcffcfaeba43cb4dd33ae3e0e1c6" | ||||||
|  |   integrity sha512-Oixnb+DzmRT30qu9d3tJSQkxuygWm32DFykT4bRoORPa9hZ/L4KhVB/XiRm6KG+roIEM7DBQlmg27kw2HZkdZg== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@ampproject/remapping" "^2.2.0" |     "@ampproject/remapping" "^2.2.0" | ||||||
|     "@babel/code-frame" "^7.24.7" |     "@babel/code-frame" "^7.25.7" | ||||||
|     "@babel/generator" "^7.25.0" |     "@babel/generator" "^7.25.7" | ||||||
|     "@babel/helper-compilation-targets" "^7.25.2" |     "@babel/helper-compilation-targets" "^7.25.7" | ||||||
|     "@babel/helper-module-transforms" "^7.25.2" |     "@babel/helper-module-transforms" "^7.25.7" | ||||||
|     "@babel/helpers" "^7.25.0" |     "@babel/helpers" "^7.25.7" | ||||||
|     "@babel/parser" "^7.25.0" |     "@babel/parser" "^7.25.8" | ||||||
|     "@babel/template" "^7.25.0" |     "@babel/template" "^7.25.7" | ||||||
|     "@babel/traverse" "^7.25.2" |     "@babel/traverse" "^7.25.7" | ||||||
|     "@babel/types" "^7.25.2" |     "@babel/types" "^7.25.8" | ||||||
|     convert-source-map "^2.0.0" |     convert-source-map "^2.0.0" | ||||||
|     debug "^4.1.0" |     debug "^4.1.0" | ||||||
|     gensync "^1.0.0-beta.2" |     gensync "^1.0.0-beta.2" | ||||||
|     json5 "^2.2.3" |     json5 "^2.2.3" | ||||||
|     semver "^6.3.1" |     semver "^6.3.1" | ||||||
|  |  | ||||||
| "@babel/generator@^7.25.0", "@babel/generator@^7.25.6", "@babel/generator@^7.7.2": | "@babel/generator@^7.25.6", "@babel/generator@^7.7.2": | ||||||
|   version "7.25.6" |   version "7.25.6" | ||||||
|   resolved "https://registry.npmjs.org/@babel/generator/-/generator-7.25.6.tgz" |   resolved "https://registry.npmjs.org/@babel/generator/-/generator-7.25.6.tgz" | ||||||
|   integrity sha512-VPC82gr1seXOpkjAAKoLhP50vx4vGNlF4msF64dSFq1P8RfB+QAuJWGHPXXPc8QyfVWwwB/TNNU4+ayZmHNbZw== |   integrity sha512-VPC82gr1seXOpkjAAKoLhP50vx4vGNlF4msF64dSFq1P8RfB+QAuJWGHPXXPc8QyfVWwwB/TNNU4+ayZmHNbZw== | ||||||
| @@ -59,18 +67,28 @@ | |||||||
|     "@jridgewell/trace-mapping" "^0.3.25" |     "@jridgewell/trace-mapping" "^0.3.25" | ||||||
|     jsesc "^2.5.1" |     jsesc "^2.5.1" | ||||||
|  |  | ||||||
| "@babel/helper-compilation-targets@^7.25.2": | "@babel/generator@^7.25.7": | ||||||
|   version "7.25.2" |   version "7.25.7" | ||||||
|   resolved "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.25.2.tgz" |   resolved "https://registry.yarnpkg.com/@babel/generator/-/generator-7.25.7.tgz#de86acbeb975a3e11ee92dd52223e6b03b479c56" | ||||||
|   integrity sha512-U2U5LsSaZ7TAt3cfaymQ8WHh0pxvdHoEk6HVpaexxixjyEquMh0L0YNJNM6CTGKMXV1iksi0iZkGw4AcFkPaaw== |   integrity sha512-5Dqpl5fyV9pIAD62yK9P7fcA768uVPUyrQmqpqstHWgMma4feF1x/oFysBCVZLY5wJ2GkMUCdsNDnGZrPoR6rA== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@babel/compat-data" "^7.25.2" |     "@babel/types" "^7.25.7" | ||||||
|     "@babel/helper-validator-option" "^7.24.8" |     "@jridgewell/gen-mapping" "^0.3.5" | ||||||
|     browserslist "^4.23.1" |     "@jridgewell/trace-mapping" "^0.3.25" | ||||||
|  |     jsesc "^3.0.2" | ||||||
|  |  | ||||||
|  | "@babel/helper-compilation-targets@^7.25.7": | ||||||
|  |   version "7.25.7" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@babel/helper-compilation-targets/-/helper-compilation-targets-7.25.7.tgz#11260ac3322dda0ef53edfae6e97b961449f5fa4" | ||||||
|  |   integrity sha512-DniTEax0sv6isaw6qSQSfV4gVRNtw2rte8HHM45t9ZR0xILaufBRNkpMifCRiAPyvL4ACD6v0gfCwCmtOQaV4A== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/compat-data" "^7.25.7" | ||||||
|  |     "@babel/helper-validator-option" "^7.25.7" | ||||||
|  |     browserslist "^4.24.0" | ||||||
|     lru-cache "^5.1.1" |     lru-cache "^5.1.1" | ||||||
|     semver "^6.3.1" |     semver "^6.3.1" | ||||||
|  |  | ||||||
| "@babel/helper-module-imports@^7.16.7", "@babel/helper-module-imports@^7.24.7": | "@babel/helper-module-imports@^7.16.7": | ||||||
|   version "7.24.7" |   version "7.24.7" | ||||||
|   resolved "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.24.7.tgz" |   resolved "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.24.7.tgz" | ||||||
|   integrity sha512-8AyH3C+74cgCVVXow/myrynrAGv+nTVg5vKu2nZph9x7RcRwzmh0VFallJuFTZ9mx6u4eSdXZfcOzSqTUm0HCA== |   integrity sha512-8AyH3C+74cgCVVXow/myrynrAGv+nTVg5vKu2nZph9x7RcRwzmh0VFallJuFTZ9mx6u4eSdXZfcOzSqTUm0HCA== | ||||||
| @@ -78,51 +96,74 @@ | |||||||
|     "@babel/traverse" "^7.24.7" |     "@babel/traverse" "^7.24.7" | ||||||
|     "@babel/types" "^7.24.7" |     "@babel/types" "^7.24.7" | ||||||
|  |  | ||||||
| "@babel/helper-module-transforms@^7.25.2": | "@babel/helper-module-imports@^7.25.7": | ||||||
|   version "7.25.2" |   version "7.25.7" | ||||||
|   resolved "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.25.2.tgz" |   resolved "https://registry.yarnpkg.com/@babel/helper-module-imports/-/helper-module-imports-7.25.7.tgz#dba00d9523539152906ba49263e36d7261040472" | ||||||
|   integrity sha512-BjyRAbix6j/wv83ftcVJmBt72QtHI56C7JXZoG2xATiLpmoC7dpd8WnkikExHDVPpi/3qCmO6WY1EaXOluiecQ== |   integrity sha512-o0xCgpNmRohmnoWKQ0Ij8IdddjyBFE4T2kagL/x6M3+4zUgc+4qTOUBoNe4XxDskt1HPKO007ZPiMgLDq2s7Kw== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@babel/helper-module-imports" "^7.24.7" |     "@babel/traverse" "^7.25.7" | ||||||
|     "@babel/helper-simple-access" "^7.24.7" |     "@babel/types" "^7.25.7" | ||||||
|     "@babel/helper-validator-identifier" "^7.24.7" |  | ||||||
|     "@babel/traverse" "^7.25.2" | "@babel/helper-module-transforms@^7.25.7": | ||||||
|  |   version "7.25.7" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@babel/helper-module-transforms/-/helper-module-transforms-7.25.7.tgz#2ac9372c5e001b19bc62f1fe7d96a18cb0901d1a" | ||||||
|  |   integrity sha512-k/6f8dKG3yDz/qCwSM+RKovjMix563SLxQFo0UhRNo239SP6n9u5/eLtKD6EAjwta2JHJ49CsD8pms2HdNiMMQ== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/helper-module-imports" "^7.25.7" | ||||||
|  |     "@babel/helper-simple-access" "^7.25.7" | ||||||
|  |     "@babel/helper-validator-identifier" "^7.25.7" | ||||||
|  |     "@babel/traverse" "^7.25.7" | ||||||
|  |  | ||||||
| "@babel/helper-plugin-utils@^7.0.0", "@babel/helper-plugin-utils@^7.10.4", "@babel/helper-plugin-utils@^7.12.13", "@babel/helper-plugin-utils@^7.14.5", "@babel/helper-plugin-utils@^7.24.7", "@babel/helper-plugin-utils@^7.24.8", "@babel/helper-plugin-utils@^7.8.0": | "@babel/helper-plugin-utils@^7.0.0", "@babel/helper-plugin-utils@^7.10.4", "@babel/helper-plugin-utils@^7.12.13", "@babel/helper-plugin-utils@^7.14.5", "@babel/helper-plugin-utils@^7.24.7", "@babel/helper-plugin-utils@^7.24.8", "@babel/helper-plugin-utils@^7.8.0": | ||||||
|   version "7.24.8" |   version "7.24.8" | ||||||
|   resolved "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.24.8.tgz" |   resolved "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.24.8.tgz" | ||||||
|   integrity sha512-FFWx5142D8h2Mgr/iPVGH5G7w6jDn4jUSpZTyDnQO0Yn7Ks2Kuz6Pci8H6MPCoUJegd/UZQ3tAvfLCxQSnWWwg== |   integrity sha512-FFWx5142D8h2Mgr/iPVGH5G7w6jDn4jUSpZTyDnQO0Yn7Ks2Kuz6Pci8H6MPCoUJegd/UZQ3tAvfLCxQSnWWwg== | ||||||
|  |  | ||||||
| "@babel/helper-simple-access@^7.24.7": | "@babel/helper-plugin-utils@^7.25.7": | ||||||
|   version "7.24.7" |   version "7.25.7" | ||||||
|   resolved "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.24.7.tgz" |   resolved "https://registry.yarnpkg.com/@babel/helper-plugin-utils/-/helper-plugin-utils-7.25.7.tgz#8ec5b21812d992e1ef88a9b068260537b6f0e36c" | ||||||
|   integrity sha512-zBAIvbCMh5Ts+b86r/CjU+4XGYIs+R1j951gxI3KmmxBMhCg4oQMsv6ZXQ64XOm/cvzfU1FmoCyt6+owc5QMYg== |   integrity sha512-eaPZai0PiqCi09pPs3pAFfl/zYgGaE6IdXtYvmf0qlcDTd3WCtO7JWCcRd64e0EQrcYgiHibEZnOGsSY4QSgaw== | ||||||
|  |  | ||||||
|  | "@babel/helper-simple-access@^7.25.7": | ||||||
|  |   version "7.25.7" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@babel/helper-simple-access/-/helper-simple-access-7.25.7.tgz#5eb9f6a60c5d6b2e0f76057004f8dacbddfae1c0" | ||||||
|  |   integrity sha512-FPGAkJmyoChQeM+ruBGIDyrT2tKfZJO8NcxdC+CWNJi7N8/rZpSxK7yvBJ5O/nF1gfu5KzN7VKG3YVSLFfRSxQ== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@babel/traverse" "^7.24.7" |     "@babel/traverse" "^7.25.7" | ||||||
|     "@babel/types" "^7.24.7" |     "@babel/types" "^7.25.7" | ||||||
|  |  | ||||||
| "@babel/helper-string-parser@^7.24.8": | "@babel/helper-string-parser@^7.24.8": | ||||||
|   version "7.24.8" |   version "7.24.8" | ||||||
|   resolved "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.24.8.tgz" |   resolved "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.24.8.tgz" | ||||||
|   integrity sha512-pO9KhhRcuUyGnJWwyEgnRJTSIZHiT+vMD0kPeD+so0l7mxkMT19g3pjY9GTnHySck/hDzq+dtW/4VgnMkippsQ== |   integrity sha512-pO9KhhRcuUyGnJWwyEgnRJTSIZHiT+vMD0kPeD+so0l7mxkMT19g3pjY9GTnHySck/hDzq+dtW/4VgnMkippsQ== | ||||||
|  |  | ||||||
|  | "@babel/helper-string-parser@^7.25.7": | ||||||
|  |   version "7.25.7" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@babel/helper-string-parser/-/helper-string-parser-7.25.7.tgz#d50e8d37b1176207b4fe9acedec386c565a44a54" | ||||||
|  |   integrity sha512-CbkjYdsJNHFk8uqpEkpCvRs3YRp9tY6FmFY7wLMSYuGYkrdUi7r2lc4/wqsvlHoMznX3WJ9IP8giGPq68T/Y6g== | ||||||
|  |  | ||||||
| "@babel/helper-validator-identifier@^7.24.7": | "@babel/helper-validator-identifier@^7.24.7": | ||||||
|   version "7.24.7" |   version "7.24.7" | ||||||
|   resolved "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.24.7.tgz" |   resolved "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.24.7.tgz" | ||||||
|   integrity sha512-rR+PBcQ1SMQDDyF6X0wxtG8QyLCgUB0eRAGguqRLfkCA87l7yAP7ehq8SNj96OOGTO8OBV70KhuFYcIkHXOg0w== |   integrity sha512-rR+PBcQ1SMQDDyF6X0wxtG8QyLCgUB0eRAGguqRLfkCA87l7yAP7ehq8SNj96OOGTO8OBV70KhuFYcIkHXOg0w== | ||||||
|  |  | ||||||
| "@babel/helper-validator-option@^7.24.8": | "@babel/helper-validator-identifier@^7.25.7": | ||||||
|   version "7.24.8" |   version "7.25.7" | ||||||
|   resolved "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.24.8.tgz" |   resolved "https://registry.yarnpkg.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.25.7.tgz#77b7f60c40b15c97df735b38a66ba1d7c3e93da5" | ||||||
|   integrity sha512-xb8t9tD1MHLungh/AIoWYN+gVHaB9kwlu8gffXGSt3FFEIT7RjS+xWbc2vUD1UTZdIpKj/ab3rdqJ7ufngyi2Q== |   integrity sha512-AM6TzwYqGChO45oiuPqwL2t20/HdMC1rTPAesnBCgPCSF1x3oN9MVUwQV2iyz4xqWrctwK5RNC8LV22kaQCNYg== | ||||||
|  |  | ||||||
| "@babel/helpers@^7.25.0": | "@babel/helper-validator-option@^7.25.7": | ||||||
|   version "7.25.6" |   version "7.25.7" | ||||||
|   resolved "https://registry.npmjs.org/@babel/helpers/-/helpers-7.25.6.tgz" |   resolved "https://registry.yarnpkg.com/@babel/helper-validator-option/-/helper-validator-option-7.25.7.tgz#97d1d684448228b30b506d90cace495d6f492729" | ||||||
|   integrity sha512-Xg0tn4HcfTijTwfDwYlvVCl43V6h4KyVVX2aEm4qdO/PC6L2YvzLHFdmxhoeSA3eslcE6+ZVXHgWwopXYLNq4Q== |   integrity sha512-ytbPLsm+GjArDYXJ8Ydr1c/KJuutjF2besPNbIZnZ6MKUxi/uTA22t2ymmA4WFjZFpjiAMO0xuuJPqK2nvDVfQ== | ||||||
|  |  | ||||||
|  | "@babel/helpers@^7.25.7": | ||||||
|  |   version "7.25.7" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@babel/helpers/-/helpers-7.25.7.tgz#091b52cb697a171fe0136ab62e54e407211f09c2" | ||||||
|  |   integrity sha512-Sv6pASx7Esm38KQpF/U/OXLwPPrdGHNKoeblRxgZRLXnAtnkEe4ptJPDtAZM7fBLadbc1Q07kQpSiGQ0Jg6tRA== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@babel/template" "^7.25.0" |     "@babel/template" "^7.25.7" | ||||||
|     "@babel/types" "^7.25.6" |     "@babel/types" "^7.25.7" | ||||||
|  |  | ||||||
| "@babel/highlight@^7.24.7": | "@babel/highlight@^7.24.7": | ||||||
|   version "7.24.7" |   version "7.24.7" | ||||||
| @@ -134,6 +175,16 @@ | |||||||
|     js-tokens "^4.0.0" |     js-tokens "^4.0.0" | ||||||
|     picocolors "^1.0.0" |     picocolors "^1.0.0" | ||||||
|  |  | ||||||
|  | "@babel/highlight@^7.25.7": | ||||||
|  |   version "7.25.7" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@babel/highlight/-/highlight-7.25.7.tgz#20383b5f442aa606e7b5e3043b0b1aafe9f37de5" | ||||||
|  |   integrity sha512-iYyACpW3iW8Fw+ZybQK+drQre+ns/tKpXbNESfrhNnPLIklLbXr7MYJ6gPEd0iETGLOK+SxMjVvKb/ffmk+FEw== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/helper-validator-identifier" "^7.25.7" | ||||||
|  |     chalk "^2.4.2" | ||||||
|  |     js-tokens "^4.0.0" | ||||||
|  |     picocolors "^1.0.0" | ||||||
|  |  | ||||||
| "@babel/parser@^7.1.0", "@babel/parser@^7.14.7", "@babel/parser@^7.20.7", "@babel/parser@^7.23.9", "@babel/parser@^7.25.0", "@babel/parser@^7.25.6": | "@babel/parser@^7.1.0", "@babel/parser@^7.14.7", "@babel/parser@^7.20.7", "@babel/parser@^7.23.9", "@babel/parser@^7.25.0", "@babel/parser@^7.25.6": | ||||||
|   version "7.25.6" |   version "7.25.6" | ||||||
|   resolved "https://registry.npmjs.org/@babel/parser/-/parser-7.25.6.tgz" |   resolved "https://registry.npmjs.org/@babel/parser/-/parser-7.25.6.tgz" | ||||||
| @@ -141,6 +192,13 @@ | |||||||
|   dependencies: |   dependencies: | ||||||
|     "@babel/types" "^7.25.6" |     "@babel/types" "^7.25.6" | ||||||
|  |  | ||||||
|  | "@babel/parser@^7.25.7", "@babel/parser@^7.25.8": | ||||||
|  |   version "7.25.8" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@babel/parser/-/parser-7.25.8.tgz#f6aaf38e80c36129460c1657c0762db584c9d5e2" | ||||||
|  |   integrity sha512-HcttkxzdPucv3nNFmfOOMfFf64KgdJVqm1KaCm25dPGMLElo9nsLvXeJECQg8UzPuBGLyTSA0ZzqCtDSzKTEoQ== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/types" "^7.25.8" | ||||||
|  |  | ||||||
| "@babel/plugin-syntax-async-generators@^7.8.4": | "@babel/plugin-syntax-async-generators@^7.8.4": | ||||||
|   version "7.8.4" |   version "7.8.4" | ||||||
|   resolved "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz" |   resolved "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz" | ||||||
| @@ -260,19 +318,19 @@ | |||||||
|   dependencies: |   dependencies: | ||||||
|     "@babel/helper-plugin-utils" "^7.24.8" |     "@babel/helper-plugin-utils" "^7.24.8" | ||||||
|  |  | ||||||
| "@babel/plugin-transform-react-jsx-self@^7.24.5": | "@babel/plugin-transform-react-jsx-self@^7.24.7": | ||||||
|   version "7.24.7" |   version "7.25.7" | ||||||
|   resolved "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-self/-/plugin-transform-react-jsx-self-7.24.7.tgz" |   resolved "https://registry.yarnpkg.com/@babel/plugin-transform-react-jsx-self/-/plugin-transform-react-jsx-self-7.25.7.tgz#3d11df143131fd8f5486a1f7d3839890f88f8c85" | ||||||
|   integrity sha512-fOPQYbGSgH0HUp4UJO4sMBFjY6DuWq+2i8rixyUMb3CdGixs/gccURvYOAhajBdKDoGajFr3mUq5rH3phtkGzw== |   integrity sha512-JD9MUnLbPL0WdVK8AWC7F7tTG2OS6u/AKKnsK+NdRhUiVdnzyR1S3kKQCaRLOiaULvUiqK6Z4JQE635VgtCFeg== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@babel/helper-plugin-utils" "^7.24.7" |     "@babel/helper-plugin-utils" "^7.25.7" | ||||||
|  |  | ||||||
| "@babel/plugin-transform-react-jsx-source@^7.24.1": | "@babel/plugin-transform-react-jsx-source@^7.24.7": | ||||||
|   version "7.24.7" |   version "7.25.7" | ||||||
|   resolved "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-source/-/plugin-transform-react-jsx-source-7.24.7.tgz" |   resolved "https://registry.yarnpkg.com/@babel/plugin-transform-react-jsx-source/-/plugin-transform-react-jsx-source-7.25.7.tgz#a0d8372310d5ea5b0447dfa03a8485f960eff7be" | ||||||
|   integrity sha512-J2z+MWzZHVOemyLweMqngXrgGC42jQ//R0KdxqkIz/OrbVIIlhFI3WigZ5fO+nwFvBlncr4MGapd8vTyc7RPNQ== |   integrity sha512-S/JXG/KrbIY06iyJPKfxr0qRxnhNOdkNXYBl/rmwgDd72cQLH9tEGkDm/yJPGvcSIUoikzfjMios9i+xT/uv9w== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@babel/helper-plugin-utils" "^7.24.7" |     "@babel/helper-plugin-utils" "^7.25.7" | ||||||
|  |  | ||||||
| "@babel/runtime@^7.12.5", "@babel/runtime@^7.18.3", "@babel/runtime@^7.21.0", "@babel/runtime@^7.23.8", "@babel/runtime@^7.23.9", "@babel/runtime@^7.25.6", "@babel/runtime@^7.5.5", "@babel/runtime@^7.6.2", "@babel/runtime@^7.7.2", "@babel/runtime@^7.7.6", "@babel/runtime@^7.8.7": | "@babel/runtime@^7.12.5", "@babel/runtime@^7.18.3", "@babel/runtime@^7.21.0", "@babel/runtime@^7.23.8", "@babel/runtime@^7.23.9", "@babel/runtime@^7.25.6", "@babel/runtime@^7.5.5", "@babel/runtime@^7.6.2", "@babel/runtime@^7.7.2", "@babel/runtime@^7.7.6", "@babel/runtime@^7.8.7": | ||||||
|   version "7.25.6" |   version "7.25.6" | ||||||
| @@ -290,7 +348,16 @@ | |||||||
|     "@babel/parser" "^7.25.0" |     "@babel/parser" "^7.25.0" | ||||||
|     "@babel/types" "^7.25.0" |     "@babel/types" "^7.25.0" | ||||||
|  |  | ||||||
| "@babel/traverse@^7.24.7", "@babel/traverse@^7.25.2": | "@babel/template@^7.25.7": | ||||||
|  |   version "7.25.7" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@babel/template/-/template-7.25.7.tgz#27f69ce382855d915b14ab0fe5fb4cbf88fa0769" | ||||||
|  |   integrity sha512-wRwtAgI3bAS+JGU2upWNL9lSlDcRCqD05BZ1n3X2ONLH1WilFP6O1otQjeMK/1g0pvYcXC7b/qVUB1keofjtZA== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/code-frame" "^7.25.7" | ||||||
|  |     "@babel/parser" "^7.25.7" | ||||||
|  |     "@babel/types" "^7.25.7" | ||||||
|  |  | ||||||
|  | "@babel/traverse@^7.24.7": | ||||||
|   version "7.25.6" |   version "7.25.6" | ||||||
|   resolved "https://registry.npmjs.org/@babel/traverse/-/traverse-7.25.6.tgz" |   resolved "https://registry.npmjs.org/@babel/traverse/-/traverse-7.25.6.tgz" | ||||||
|   integrity sha512-9Vrcx5ZW6UwK5tvqsj0nGpp/XzqthkT0dqIc9g1AdtygFToNtTF67XzYS//dm+SAK9cp3B9R4ZO/46p63SCjlQ== |   integrity sha512-9Vrcx5ZW6UwK5tvqsj0nGpp/XzqthkT0dqIc9g1AdtygFToNtTF67XzYS//dm+SAK9cp3B9R4ZO/46p63SCjlQ== | ||||||
| @@ -303,7 +370,20 @@ | |||||||
|     debug "^4.3.1" |     debug "^4.3.1" | ||||||
|     globals "^11.1.0" |     globals "^11.1.0" | ||||||
|  |  | ||||||
| "@babel/types@^7.0.0", "@babel/types@^7.20.7", "@babel/types@^7.24.7", "@babel/types@^7.25.0", "@babel/types@^7.25.2", "@babel/types@^7.25.6", "@babel/types@^7.3.3": | "@babel/traverse@^7.25.7": | ||||||
|  |   version "7.25.7" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@babel/traverse/-/traverse-7.25.7.tgz#83e367619be1cab8e4f2892ef30ba04c26a40fa8" | ||||||
|  |   integrity sha512-jatJPT1Zjqvh/1FyJs6qAHL+Dzb7sTb+xr7Q+gM1b+1oBsMsQQ4FkVKb6dFlJvLlVssqkRzV05Jzervt9yhnzg== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/code-frame" "^7.25.7" | ||||||
|  |     "@babel/generator" "^7.25.7" | ||||||
|  |     "@babel/parser" "^7.25.7" | ||||||
|  |     "@babel/template" "^7.25.7" | ||||||
|  |     "@babel/types" "^7.25.7" | ||||||
|  |     debug "^4.3.1" | ||||||
|  |     globals "^11.1.0" | ||||||
|  |  | ||||||
|  | "@babel/types@^7.0.0", "@babel/types@^7.20.7", "@babel/types@^7.24.7", "@babel/types@^7.25.0", "@babel/types@^7.25.6", "@babel/types@^7.3.3": | ||||||
|   version "7.25.6" |   version "7.25.6" | ||||||
|   resolved "https://registry.npmjs.org/@babel/types/-/types-7.25.6.tgz" |   resolved "https://registry.npmjs.org/@babel/types/-/types-7.25.6.tgz" | ||||||
|   integrity sha512-/l42B1qxpG6RdfYf343Uw1vmDjeNhneUXtzhojE7pDgfpEypmRhI6j1kr17XCVv4Cgl9HdAiQY2x0GwKm7rWCw== |   integrity sha512-/l42B1qxpG6RdfYf343Uw1vmDjeNhneUXtzhojE7pDgfpEypmRhI6j1kr17XCVv4Cgl9HdAiQY2x0GwKm7rWCw== | ||||||
| @@ -312,6 +392,15 @@ | |||||||
|     "@babel/helper-validator-identifier" "^7.24.7" |     "@babel/helper-validator-identifier" "^7.24.7" | ||||||
|     to-fast-properties "^2.0.0" |     to-fast-properties "^2.0.0" | ||||||
|  |  | ||||||
|  | "@babel/types@^7.25.7", "@babel/types@^7.25.8": | ||||||
|  |   version "7.25.8" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@babel/types/-/types-7.25.8.tgz#5cf6037258e8a9bcad533f4979025140cb9993e1" | ||||||
|  |   integrity sha512-JWtuCu8VQsMladxVz/P4HzHUGCAwpuqacmowgXFs5XjxIgKuNjnLokQzuVjlTvIzODaDmpjT3oxcC48vyk9EWg== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/helper-string-parser" "^7.25.7" | ||||||
|  |     "@babel/helper-validator-identifier" "^7.25.7" | ||||||
|  |     to-fast-properties "^2.0.0" | ||||||
|  |  | ||||||
| "@bcoe/v8-coverage@^0.2.3": | "@bcoe/v8-coverage@^0.2.3": | ||||||
|   version "0.2.3" |   version "0.2.3" | ||||||
|   resolved "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-0.2.3.tgz" |   resolved "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-0.2.3.tgz" | ||||||
| @@ -1053,10 +1142,10 @@ | |||||||
|   resolved "https://registry.npmjs.org/@popperjs/core/-/core-2.11.8.tgz" |   resolved "https://registry.npmjs.org/@popperjs/core/-/core-2.11.8.tgz" | ||||||
|   integrity sha512-P1st0aksCrn9sGZhp8GMYwBnQsbvAWsZAX44oXNNvLHGqAOcoVxmjZiohstwQ7SqKnbR47akdNi+uleWD8+g6A== |   integrity sha512-P1st0aksCrn9sGZhp8GMYwBnQsbvAWsZAX44oXNNvLHGqAOcoVxmjZiohstwQ7SqKnbR47akdNi+uleWD8+g6A== | ||||||
|  |  | ||||||
| "@remix-run/router@1.19.2": | "@remix-run/router@1.20.0": | ||||||
|   version "1.19.2" |   version "1.20.0" | ||||||
|   resolved "https://registry.npmjs.org/@remix-run/router/-/router-1.19.2.tgz" |   resolved "https://registry.yarnpkg.com/@remix-run/router/-/router-1.20.0.tgz#03554155b45d8b529adf635b2f6ad1165d70d8b4" | ||||||
|   integrity sha512-baiMx18+IMuD1yyvOGaHM9QrVUPGGG0jC+z+IPHnRJWUAUvaKuWKyE8gjDj2rzv3sz9zOGoRSPgeBVHRhZnBlA== |   integrity sha512-mUnk8rPJBI9loFDZ+YzPGdeniYK+FTmRD1TMCz7ev2SNIozyKKpnGgsxO34u6Z4z/t0ITuu7voi/AshfsGsgFg== | ||||||
|  |  | ||||||
| "@rollup/rollup-android-arm-eabi@4.22.4": | "@rollup/rollup-android-arm-eabi@4.22.4": | ||||||
|   version "4.22.4" |   version "4.22.4" | ||||||
| @@ -1188,10 +1277,10 @@ | |||||||
|     lz-string "^1.5.0" |     lz-string "^1.5.0" | ||||||
|     pretty-format "^27.0.2" |     pretty-format "^27.0.2" | ||||||
|  |  | ||||||
| "@testing-library/jest-dom@^6.0.0": | "@testing-library/jest-dom@^6.6.2": | ||||||
|   version "6.5.0" |   version "6.6.2" | ||||||
|   resolved "https://registry.npmjs.org/@testing-library/jest-dom/-/jest-dom-6.5.0.tgz" |   resolved "https://registry.yarnpkg.com/@testing-library/jest-dom/-/jest-dom-6.6.2.tgz#8186aa9a07263adef9cc5a59a4772db8c31f4a5b" | ||||||
|   integrity sha512-xGGHpBXYSHUUr6XsKBfs85TWlYKpTc37cSBBVrXcib2MkHLboWlkClhWF37JKlDb9KEq3dHs+f2xR7XJEWGBxA== |   integrity sha512-P6GJD4yqc9jZLbe98j/EkyQDTPgqftohZF5FBkHY5BUERZmcf4HeO2k0XaefEg329ux2p21i1A1DmyQ1kKw2Jw== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@adobe/css-tools" "^4.4.0" |     "@adobe/css-tools" "^4.4.0" | ||||||
|     aria-query "^5.0.0" |     aria-query "^5.0.0" | ||||||
| @@ -1334,17 +1423,17 @@ | |||||||
|   resolved "https://registry.npmjs.org/@types/lodash/-/lodash-4.17.7.tgz" |   resolved "https://registry.npmjs.org/@types/lodash/-/lodash-4.17.7.tgz" | ||||||
|   integrity sha512-8wTvZawATi/lsmNu10/j2hk1KEP0IvjubqPE3cu1Xz7xfXXt5oCq3SNUz4fMIP4XGF9Ky+Ue2tBA3hcS7LSBlA== |   integrity sha512-8wTvZawATi/lsmNu10/j2hk1KEP0IvjubqPE3cu1Xz7xfXXt5oCq3SNUz4fMIP4XGF9Ky+Ue2tBA3hcS7LSBlA== | ||||||
|  |  | ||||||
| "@types/node@*", "@types/node@^20.14.12": | "@types/node@*", "@types/node@^22.7.7": | ||||||
|   version "20.16.5" |   version "22.7.7" | ||||||
|   resolved "https://registry.npmjs.org/@types/node/-/node-20.16.5.tgz" |   resolved "https://registry.yarnpkg.com/@types/node/-/node-22.7.7.tgz#6cd9541c3dccb4f7e8b141b491443f4a1570e307" | ||||||
|   integrity sha512-VwYCweNo3ERajwy0IUlqqcyZ8/A7Zwa9ZP3MnENWcB11AejO+tLy3pu850goUW2FC/IJMdZUfKpX/yxL1gymCA== |   integrity sha512-SRxCrrg9CL/y54aiMCG3edPKdprgMVGDXjA3gB8UmmBW5TcXzRUYAh8EWzTnSJFAd1rgImPELza+A3bJ+qxz8Q== | ||||||
|   dependencies: |   dependencies: | ||||||
|     undici-types "~6.19.2" |     undici-types "~6.19.2" | ||||||
|  |  | ||||||
| "@types/papaparse@^5.3.14": | "@types/papaparse@^5.3.15": | ||||||
|   version "5.3.14" |   version "5.3.15" | ||||||
|   resolved "https://registry.npmjs.org/@types/papaparse/-/papaparse-5.3.14.tgz" |   resolved "https://registry.yarnpkg.com/@types/papaparse/-/papaparse-5.3.15.tgz#7cafa16757a1d121422deefbb10b6310b224ecc4" | ||||||
|   integrity sha512-LxJ4iEFcpqc6METwp9f6BV6VVc43m6MfH0VqFosHvrUgfXiFe6ww7R3itkOQ+TCK6Y+Iv/+RnnvtRZnkc5Kc9g== |   integrity sha512-JHe6vF6x/8Z85nCX4yFdDslN11d+1pr12E526X8WAfhadOeaOTx5AuIkvDKIBopfvlzpzkdMx4YyvSKCM9oqtw== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@types/node" "*" |     "@types/node" "*" | ||||||
|  |  | ||||||
| @@ -1395,85 +1484,85 @@ | |||||||
|   dependencies: |   dependencies: | ||||||
|     "@types/yargs-parser" "*" |     "@types/yargs-parser" "*" | ||||||
|  |  | ||||||
| "@typescript-eslint/eslint-plugin@7.18.0", "@typescript-eslint/eslint-plugin@^7.16.1": | "@typescript-eslint/eslint-plugin@8.10.0", "@typescript-eslint/eslint-plugin@^8.10.0": | ||||||
|   version "7.18.0" |   version "8.10.0" | ||||||
|   resolved "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-7.18.0.tgz" |   resolved "https://registry.yarnpkg.com/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.10.0.tgz#9c8218ed62f9a322df10ded7c34990f014df44f2" | ||||||
|   integrity sha512-94EQTWZ40mzBc42ATNIBimBEDltSJ9RQHCC8vc/PDbxi4k8dVwUAv4o98dk50M1zB+JGFxp43FP7f8+FP8R6Sw== |   integrity sha512-phuB3hoP7FFKbRXxjl+DRlQDuJqhpOnm5MmtROXyWi3uS/Xg2ZXqiQfcG2BJHiN4QKyzdOJi3NEn/qTnjUlkmQ== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@eslint-community/regexpp" "^4.10.0" |     "@eslint-community/regexpp" "^4.10.0" | ||||||
|     "@typescript-eslint/scope-manager" "7.18.0" |     "@typescript-eslint/scope-manager" "8.10.0" | ||||||
|     "@typescript-eslint/type-utils" "7.18.0" |     "@typescript-eslint/type-utils" "8.10.0" | ||||||
|     "@typescript-eslint/utils" "7.18.0" |     "@typescript-eslint/utils" "8.10.0" | ||||||
|     "@typescript-eslint/visitor-keys" "7.18.0" |     "@typescript-eslint/visitor-keys" "8.10.0" | ||||||
|     graphemer "^1.4.0" |     graphemer "^1.4.0" | ||||||
|     ignore "^5.3.1" |     ignore "^5.3.1" | ||||||
|     natural-compare "^1.4.0" |     natural-compare "^1.4.0" | ||||||
|     ts-api-utils "^1.3.0" |     ts-api-utils "^1.3.0" | ||||||
|  |  | ||||||
| "@typescript-eslint/parser@7.18.0", "@typescript-eslint/parser@^7.16.1": | "@typescript-eslint/parser@8.10.0", "@typescript-eslint/parser@^8.10.0": | ||||||
|   version "7.18.0" |   version "8.10.0" | ||||||
|   resolved "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-7.18.0.tgz" |   resolved "https://registry.yarnpkg.com/@typescript-eslint/parser/-/parser-8.10.0.tgz#3cbe7206f5e42835878a74a76da533549f977662" | ||||||
|   integrity sha512-4Z+L8I2OqhZV8qA132M4wNL30ypZGYOQVBfMgxDH/K5UX0PNqTu1c6za9ST5r9+tavvHiTWmBnKzpCJ/GlVFtg== |   integrity sha512-E24l90SxuJhytWJ0pTQydFT46Nk0Z+bsLKo/L8rtQSL93rQ6byd1V/QbDpHUTdLPOMsBCcYXZweADNCfOCmOAg== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@typescript-eslint/scope-manager" "7.18.0" |     "@typescript-eslint/scope-manager" "8.10.0" | ||||||
|     "@typescript-eslint/types" "7.18.0" |     "@typescript-eslint/types" "8.10.0" | ||||||
|     "@typescript-eslint/typescript-estree" "7.18.0" |     "@typescript-eslint/typescript-estree" "8.10.0" | ||||||
|     "@typescript-eslint/visitor-keys" "7.18.0" |     "@typescript-eslint/visitor-keys" "8.10.0" | ||||||
|     debug "^4.3.4" |     debug "^4.3.4" | ||||||
|  |  | ||||||
| "@typescript-eslint/scope-manager@7.18.0": | "@typescript-eslint/scope-manager@8.10.0": | ||||||
|   version "7.18.0" |   version "8.10.0" | ||||||
|   resolved "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-7.18.0.tgz" |   resolved "https://registry.yarnpkg.com/@typescript-eslint/scope-manager/-/scope-manager-8.10.0.tgz#606ffe18314d7b5c2f118f2f02aaa2958107a19c" | ||||||
|   integrity sha512-jjhdIE/FPF2B7Z1uzc6i3oWKbGcHb87Qw7AWj6jmEqNOfDFbJWtjt/XfwCpvNkpGWlcJaog5vTR+VV8+w9JflA== |   integrity sha512-AgCaEjhfql9MDKjMUxWvH7HjLeBqMCBfIaBbzzIcBbQPZE7CPh1m6FF+L75NUMJFMLYhCywJXIDEMa3//1A0dw== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@typescript-eslint/types" "7.18.0" |     "@typescript-eslint/types" "8.10.0" | ||||||
|     "@typescript-eslint/visitor-keys" "7.18.0" |     "@typescript-eslint/visitor-keys" "8.10.0" | ||||||
|  |  | ||||||
| "@typescript-eslint/type-utils@7.18.0": | "@typescript-eslint/type-utils@8.10.0": | ||||||
|   version "7.18.0" |   version "8.10.0" | ||||||
|   resolved "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-7.18.0.tgz" |   resolved "https://registry.yarnpkg.com/@typescript-eslint/type-utils/-/type-utils-8.10.0.tgz#99f1d2e21f8c74703e7d9c4a67a87271eaf57597" | ||||||
|   integrity sha512-XL0FJXuCLaDuX2sYqZUUSOJ2sG5/i1AAze+axqmLnSkNEVMVYLF+cbwlB2w8D1tinFuSikHmFta+P+HOofrLeA== |   integrity sha512-PCpUOpyQSpxBn230yIcK+LeCQaXuxrgCm2Zk1S+PTIRJsEfU6nJ0TtwyH8pIwPK/vJoA+7TZtzyAJSGBz+s/dg== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@typescript-eslint/typescript-estree" "7.18.0" |     "@typescript-eslint/typescript-estree" "8.10.0" | ||||||
|     "@typescript-eslint/utils" "7.18.0" |     "@typescript-eslint/utils" "8.10.0" | ||||||
|     debug "^4.3.4" |     debug "^4.3.4" | ||||||
|     ts-api-utils "^1.3.0" |     ts-api-utils "^1.3.0" | ||||||
|  |  | ||||||
| "@typescript-eslint/types@7.18.0": | "@typescript-eslint/types@8.10.0": | ||||||
|   version "7.18.0" |   version "8.10.0" | ||||||
|   resolved "https://registry.npmjs.org/@typescript-eslint/types/-/types-7.18.0.tgz" |   resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-8.10.0.tgz#eb29c4bc2ed23489348c297469c76d28c38fb618" | ||||||
|   integrity sha512-iZqi+Ds1y4EDYUtlOOC+aUmxnE9xS/yCigkjA7XpTKV6nCBd3Hp/PRGGmdwnfkV2ThMyYldP1wRpm/id99spTQ== |   integrity sha512-k/E48uzsfJCRRbGLapdZgrX52csmWJ2rcowwPvOZ8lwPUv3xW6CcFeJAXgx4uJm+Ge4+a4tFOkdYvSpxhRhg1w== | ||||||
|  |  | ||||||
| "@typescript-eslint/typescript-estree@7.18.0": | "@typescript-eslint/typescript-estree@8.10.0": | ||||||
|   version "7.18.0" |   version "8.10.0" | ||||||
|   resolved "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-7.18.0.tgz" |   resolved "https://registry.yarnpkg.com/@typescript-eslint/typescript-estree/-/typescript-estree-8.10.0.tgz#36cc66e06c5f44d6781f95cb03b132e985273a33" | ||||||
|   integrity sha512-aP1v/BSPnnyhMHts8cf1qQ6Q1IFwwRvAQGRvBFkWlo3/lH29OXA3Pts+c10nxRxIBrDnoMqzhgdwVe5f2D6OzA== |   integrity sha512-3OE0nlcOHaMvQ8Xu5gAfME3/tWVDpb/HxtpUZ1WeOAksZ/h/gwrBzCklaGzwZT97/lBbbxJ16dMA98JMEngW4w== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@typescript-eslint/types" "7.18.0" |     "@typescript-eslint/types" "8.10.0" | ||||||
|     "@typescript-eslint/visitor-keys" "7.18.0" |     "@typescript-eslint/visitor-keys" "8.10.0" | ||||||
|     debug "^4.3.4" |     debug "^4.3.4" | ||||||
|     globby "^11.1.0" |     fast-glob "^3.3.2" | ||||||
|     is-glob "^4.0.3" |     is-glob "^4.0.3" | ||||||
|     minimatch "^9.0.4" |     minimatch "^9.0.4" | ||||||
|     semver "^7.6.0" |     semver "^7.6.0" | ||||||
|     ts-api-utils "^1.3.0" |     ts-api-utils "^1.3.0" | ||||||
|  |  | ||||||
| "@typescript-eslint/utils@7.18.0": | "@typescript-eslint/utils@8.10.0": | ||||||
|   version "7.18.0" |   version "8.10.0" | ||||||
|   resolved "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-7.18.0.tgz" |   resolved "https://registry.yarnpkg.com/@typescript-eslint/utils/-/utils-8.10.0.tgz#d78d1ce3ea3d2a88a2593ebfb1c98490131d00bf" | ||||||
|   integrity sha512-kK0/rNa2j74XuHVcoCZxdFBMF+aq/vH83CXAOHieC+2Gis4mF8jJXT5eAfyD3K0sAxtPuwxaIOIOvhwzVDt/kw== |   integrity sha512-Oq4uZ7JFr9d1ZunE/QKy5egcDRXT/FrS2z/nlxzPua2VHFtmMvFNDvpq1m/hq0ra+T52aUezfcjGRIB7vNJF9w== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@eslint-community/eslint-utils" "^4.4.0" |     "@eslint-community/eslint-utils" "^4.4.0" | ||||||
|     "@typescript-eslint/scope-manager" "7.18.0" |     "@typescript-eslint/scope-manager" "8.10.0" | ||||||
|     "@typescript-eslint/types" "7.18.0" |     "@typescript-eslint/types" "8.10.0" | ||||||
|     "@typescript-eslint/typescript-estree" "7.18.0" |     "@typescript-eslint/typescript-estree" "8.10.0" | ||||||
|  |  | ||||||
| "@typescript-eslint/visitor-keys@7.18.0": | "@typescript-eslint/visitor-keys@8.10.0": | ||||||
|   version "7.18.0" |   version "8.10.0" | ||||||
|   resolved "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-7.18.0.tgz" |   resolved "https://registry.yarnpkg.com/@typescript-eslint/visitor-keys/-/visitor-keys-8.10.0.tgz#7ce4c0c3b82140415c9cd9babe09e0000b4e9979" | ||||||
|   integrity sha512-cDF0/Gf81QpY3xYyJKDV14Zwdmid5+uuENhjH2EqFaF0ni+yAyq/LzMaIJdhNJXZI7uLzwIlA+V7oWoyn6Curg== |   integrity sha512-k8nekgqwr7FadWk548Lfph6V3r9OVqjzAIVskE7orMZR23cGJjAOVazsZSJW+ElyjfTM4wx/1g88Mi70DDtG9A== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@typescript-eslint/types" "7.18.0" |     "@typescript-eslint/types" "8.10.0" | ||||||
|     eslint-visitor-keys "^3.4.3" |     eslint-visitor-keys "^3.4.3" | ||||||
|  |  | ||||||
| "@ungap/structured-clone@^1.2.0": | "@ungap/structured-clone@^1.2.0": | ||||||
| @@ -1481,14 +1570,14 @@ | |||||||
|   resolved "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.2.0.tgz" |   resolved "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.2.0.tgz" | ||||||
|   integrity sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ== |   integrity sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ== | ||||||
|  |  | ||||||
| "@vitejs/plugin-react@^4.3.1": | "@vitejs/plugin-react@^4.3.2": | ||||||
|   version "4.3.1" |   version "4.3.2" | ||||||
|   resolved "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-4.3.1.tgz" |   resolved "https://registry.yarnpkg.com/@vitejs/plugin-react/-/plugin-react-4.3.2.tgz#1e13f666fe3135b477220d3c13b783704636b6e4" | ||||||
|   integrity sha512-m/V2syj5CuVnaxcUJOQRel/Wr31FFXRFlnOoq1TVtkCxsY5veGMTEmpWHndrhB2U8ScHtCQB1e+4hWYExQc6Lg== |   integrity sha512-hieu+o05v4glEBucTcKMK3dlES0OeJlD9YVOAPraVMOInBCwzumaIFiUjr4bHK7NPgnAHgiskUoceKercrN8vg== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@babel/core" "^7.24.5" |     "@babel/core" "^7.25.2" | ||||||
|     "@babel/plugin-transform-react-jsx-self" "^7.24.5" |     "@babel/plugin-transform-react-jsx-self" "^7.24.7" | ||||||
|     "@babel/plugin-transform-react-jsx-source" "^7.24.1" |     "@babel/plugin-transform-react-jsx-source" "^7.24.7" | ||||||
|     "@types/babel__core" "^7.20.5" |     "@types/babel__core" "^7.20.5" | ||||||
|     react-refresh "^0.14.2" |     react-refresh "^0.14.2" | ||||||
|  |  | ||||||
| @@ -1634,11 +1723,6 @@ array-includes@^3.1.6, array-includes@^3.1.8: | |||||||
|     get-intrinsic "^1.2.4" |     get-intrinsic "^1.2.4" | ||||||
|     is-string "^1.0.7" |     is-string "^1.0.7" | ||||||
|  |  | ||||||
| array-union@^2.1.0: |  | ||||||
|   version "2.1.0" |  | ||||||
|   resolved "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz" |  | ||||||
|   integrity sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw== |  | ||||||
|  |  | ||||||
| array.prototype.findlastindex@^1.2.5: | array.prototype.findlastindex@^1.2.5: | ||||||
|   version "1.2.5" |   version "1.2.5" | ||||||
|   resolved "https://registry.npmjs.org/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.5.tgz" |   resolved "https://registry.npmjs.org/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.5.tgz" | ||||||
| @@ -1847,13 +1931,13 @@ broadcast-channel@^3.4.1: | |||||||
|     rimraf "3.0.2" |     rimraf "3.0.2" | ||||||
|     unload "2.2.0" |     unload "2.2.0" | ||||||
|  |  | ||||||
| browserslist@^4.23.1: | browserslist@^4.24.0: | ||||||
|   version "4.23.3" |   version "4.24.0" | ||||||
|   resolved "https://registry.npmjs.org/browserslist/-/browserslist-4.23.3.tgz" |   resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.24.0.tgz#a1325fe4bc80b64fda169629fc01b3d6cecd38d4" | ||||||
|   integrity sha512-btwCFJVjI4YWDNfau8RhZ+B1Q/VLoUITrm3RlP6y1tYGWIOa+InuYiRGXUBXo8nA1qKmHMyLB/iVQg5TT4eFoA== |   integrity sha512-Rmb62sR1Zpjql25eSanFGEhAxcFwfA1K0GuQcLoaJBAcENegrQut3hYdhXFF1obQfiDyqIW/cLM5HSJ/9k884A== | ||||||
|   dependencies: |   dependencies: | ||||||
|     caniuse-lite "^1.0.30001646" |     caniuse-lite "^1.0.30001663" | ||||||
|     electron-to-chromium "^1.5.4" |     electron-to-chromium "^1.5.28" | ||||||
|     node-releases "^2.0.18" |     node-releases "^2.0.18" | ||||||
|     update-browserslist-db "^1.1.0" |     update-browserslist-db "^1.1.0" | ||||||
|  |  | ||||||
| @@ -1902,10 +1986,10 @@ camelcase@^6.2.0: | |||||||
|   resolved "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz" |   resolved "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz" | ||||||
|   integrity sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA== |   integrity sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA== | ||||||
|  |  | ||||||
| caniuse-lite@^1.0.30001646: | caniuse-lite@^1.0.30001663: | ||||||
|   version "1.0.30001662" |   version "1.0.30001669" | ||||||
|   resolved "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001662.tgz" |   resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001669.tgz#fda8f1d29a8bfdc42de0c170d7f34a9cf19ed7a3" | ||||||
|   integrity sha512-sgMUVwLmGseH8ZIrm1d51UbrhqMCH3jvS7gF/M6byuHOnKyLOBL7W8yz5V02OHwgLGA36o/AFhWzzh4uc5aqTA== |   integrity sha512-DlWzFDJqstqtIVx1zeSpIMLjunf5SmwOw0N2Ck/QSQdS8PLS4+9HrLaYei4w8BIAL7IB/UEDu889d8vhCTPA0w== | ||||||
|  |  | ||||||
| chalk@^2.4.2: | chalk@^2.4.2: | ||||||
|   version "2.4.2" |   version "2.4.2" | ||||||
| @@ -2288,13 +2372,6 @@ diff@^4.0.1: | |||||||
|   resolved "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz" |   resolved "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz" | ||||||
|   integrity sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A== |   integrity sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A== | ||||||
|  |  | ||||||
| dir-glob@^3.0.1: |  | ||||||
|   version "3.0.1" |  | ||||||
|   resolved "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz" |  | ||||||
|   integrity sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA== |  | ||||||
|   dependencies: |  | ||||||
|     path-type "^4.0.0" |  | ||||||
|  |  | ||||||
| doctrine@^2.1.0: | doctrine@^2.1.0: | ||||||
|   version "2.1.0" |   version "2.1.0" | ||||||
|   resolved "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz" |   resolved "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz" | ||||||
| @@ -2379,10 +2456,10 @@ ejs@^3.1.10: | |||||||
|   dependencies: |   dependencies: | ||||||
|     jake "^10.8.5" |     jake "^10.8.5" | ||||||
|  |  | ||||||
| electron-to-chromium@^1.5.4: | electron-to-chromium@^1.5.28: | ||||||
|   version "1.5.25" |   version "1.5.41" | ||||||
|   resolved "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.25.tgz" |   resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.5.41.tgz#eae1ba6c49a1a61d84cf8263351d3513b2bcc534" | ||||||
|   integrity sha512-kMb204zvK3PsSlgvvwzI3wBIcAw15tRkYk+NQdsjdDtcQWTp2RABbMQ9rUBy8KNEOM+/E6ep+XC3AykiWZld4g== |   integrity sha512-dfdv/2xNjX0P8Vzme4cfzHqnPm5xsZXwsolTYr0eyW18IUmNyG08vL+fttvinTfhKfIKdRoqkDIC9e9iWQCNYQ== | ||||||
|  |  | ||||||
| emittery@^0.13.1: | emittery@^0.13.1: | ||||||
|   version "0.13.1" |   version "0.13.1" | ||||||
| @@ -2626,17 +2703,17 @@ eslint-import-resolver-node@^0.3.9: | |||||||
|     is-core-module "^2.13.0" |     is-core-module "^2.13.0" | ||||||
|     resolve "^1.22.4" |     resolve "^1.22.4" | ||||||
|  |  | ||||||
| eslint-module-utils@^2.9.0: | eslint-module-utils@^2.12.0: | ||||||
|   version "2.11.0" |   version "2.12.0" | ||||||
|   resolved "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.11.0.tgz" |   resolved "https://registry.yarnpkg.com/eslint-module-utils/-/eslint-module-utils-2.12.0.tgz#fe4cfb948d61f49203d7b08871982b65b9af0b0b" | ||||||
|   integrity sha512-gbBE5Hitek/oG6MUVj6sFuzEjA/ClzNflVrLovHi/JgLdC7fiN5gLAY1WIPW1a0V5I999MnsrvVrCOGmmVqDBQ== |   integrity sha512-wALZ0HFoytlyh/1+4wuZ9FJCD/leWHQzzrxJ8+rebyReSLk7LApMyd3WJaLVoN+D5+WIdJyDK1c6JnE65V4Zyg== | ||||||
|   dependencies: |   dependencies: | ||||||
|     debug "^3.2.7" |     debug "^3.2.7" | ||||||
|  |  | ||||||
| eslint-plugin-import@^2.29.1: | eslint-plugin-import@^2.31.0: | ||||||
|   version "2.30.0" |   version "2.31.0" | ||||||
|   resolved "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.30.0.tgz" |   resolved "https://registry.yarnpkg.com/eslint-plugin-import/-/eslint-plugin-import-2.31.0.tgz#310ce7e720ca1d9c0bb3f69adfd1c6bdd7d9e0e7" | ||||||
|   integrity sha512-/mHNE9jINJfiD2EKkg1BKyPyUk4zdnT54YgbOgfjSakWT5oyX/qQLVNTkehyfpcMxZXMy1zyonZ2v7hZTX43Yw== |   integrity sha512-ixmkI62Rbc2/w8Vfxyh1jQRTdRTF52VxwRVHl/ykPAmqG+Nb7/kNn+byLP0LxPgI7zWA16Jt82SybJInmMia3A== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@rtsao/scc" "^1.1.0" |     "@rtsao/scc" "^1.1.0" | ||||||
|     array-includes "^3.1.8" |     array-includes "^3.1.8" | ||||||
| @@ -2646,7 +2723,7 @@ eslint-plugin-import@^2.29.1: | |||||||
|     debug "^3.2.7" |     debug "^3.2.7" | ||||||
|     doctrine "^2.1.0" |     doctrine "^2.1.0" | ||||||
|     eslint-import-resolver-node "^0.3.9" |     eslint-import-resolver-node "^0.3.9" | ||||||
|     eslint-module-utils "^2.9.0" |     eslint-module-utils "^2.12.0" | ||||||
|     hasown "^2.0.2" |     hasown "^2.0.2" | ||||||
|     is-core-module "^2.15.1" |     is-core-module "^2.15.1" | ||||||
|     is-glob "^4.0.3" |     is-glob "^4.0.3" | ||||||
| @@ -2655,6 +2732,7 @@ eslint-plugin-import@^2.29.1: | |||||||
|     object.groupby "^1.0.3" |     object.groupby "^1.0.3" | ||||||
|     object.values "^1.2.0" |     object.values "^1.2.0" | ||||||
|     semver "^6.3.1" |     semver "^6.3.1" | ||||||
|  |     string.prototype.trimend "^1.0.8" | ||||||
|     tsconfig-paths "^3.15.0" |     tsconfig-paths "^3.15.0" | ||||||
|  |  | ||||||
| eslint-plugin-jsx-a11y@^6.9.0: | eslint-plugin-jsx-a11y@^6.9.0: | ||||||
| @@ -2853,7 +2931,7 @@ fast-diff@^1.1.2: | |||||||
|   resolved "https://registry.npmjs.org/fast-diff/-/fast-diff-1.3.0.tgz" |   resolved "https://registry.npmjs.org/fast-diff/-/fast-diff-1.3.0.tgz" | ||||||
|   integrity sha512-VxPP4NqbUjj6MaAOafWeUn2cXWLcCtljklUtZf0Ind4XQ+QPtmA0b18zZy0jIQx+ExRVCR/ZQpBmik5lXshNsw== |   integrity sha512-VxPP4NqbUjj6MaAOafWeUn2cXWLcCtljklUtZf0Ind4XQ+QPtmA0b18zZy0jIQx+ExRVCR/ZQpBmik5lXshNsw== | ||||||
|  |  | ||||||
| fast-glob@^3.2.9: | fast-glob@^3.3.2: | ||||||
|   version "3.3.2" |   version "3.3.2" | ||||||
|   resolved "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.2.tgz" |   resolved "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.2.tgz" | ||||||
|   integrity sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow== |   integrity sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow== | ||||||
| @@ -3088,18 +3166,6 @@ globalthis@^1.0.3: | |||||||
|     define-properties "^1.2.1" |     define-properties "^1.2.1" | ||||||
|     gopd "^1.0.1" |     gopd "^1.0.1" | ||||||
|  |  | ||||||
| globby@^11.1.0: |  | ||||||
|   version "11.1.0" |  | ||||||
|   resolved "https://registry.npmjs.org/globby/-/globby-11.1.0.tgz" |  | ||||||
|   integrity sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g== |  | ||||||
|   dependencies: |  | ||||||
|     array-union "^2.1.0" |  | ||||||
|     dir-glob "^3.0.1" |  | ||||||
|     fast-glob "^3.2.9" |  | ||||||
|     ignore "^5.2.0" |  | ||||||
|     merge2 "^1.4.1" |  | ||||||
|     slash "^3.0.0" |  | ||||||
|  |  | ||||||
| gopd@^1.0.1: | gopd@^1.0.1: | ||||||
|   version "1.0.1" |   version "1.0.1" | ||||||
|   resolved "https://registry.npmjs.org/gopd/-/gopd-1.0.1.tgz" |   resolved "https://registry.npmjs.org/gopd/-/gopd-1.0.1.tgz" | ||||||
| @@ -3177,11 +3243,6 @@ hoist-non-react-statics@^3.3.1: | |||||||
|   dependencies: |   dependencies: | ||||||
|     react-is "^16.7.0" |     react-is "^16.7.0" | ||||||
|  |  | ||||||
| hotscript@^1.0.12: |  | ||||||
|   version "1.0.13" |  | ||||||
|   resolved "https://registry.npmjs.org/hotscript/-/hotscript-1.0.13.tgz" |  | ||||||
|   integrity sha512-C++tTF1GqkGYecL+2S1wJTfoH6APGAsbb7PAWQ3iVIwgG/EFseAfEVOKFgAFq4yK3+6j1EjUD4UQ9dRJHX/sSQ== |  | ||||||
|  |  | ||||||
| html-encoding-sniffer@^3.0.0: | html-encoding-sniffer@^3.0.0: | ||||||
|   version "3.0.0" |   version "3.0.0" | ||||||
|   resolved "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-3.0.0.tgz" |   resolved "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-3.0.0.tgz" | ||||||
| @@ -4031,6 +4092,11 @@ jsesc@^2.5.1: | |||||||
|   resolved "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz" |   resolved "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz" | ||||||
|   integrity sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA== |   integrity sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA== | ||||||
|  |  | ||||||
|  | jsesc@^3.0.2: | ||||||
|  |   version "3.0.2" | ||||||
|  |   resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-3.0.2.tgz#bb8b09a6597ba426425f2e4a07245c3d00b9343e" | ||||||
|  |   integrity sha512-xKqzzWXDttJuOcawBt4KnKHHIf5oQ/Cxax+0PWFG+DFDgHNAdi+TXECADI+RYiFUMmx8792xsMbbgXj4CwnP4g== | ||||||
|  |  | ||||||
| jshint@^2.13.6: | jshint@^2.13.6: | ||||||
|   version "2.13.6" |   version "2.13.6" | ||||||
|   resolved "https://registry.npmjs.org/jshint/-/jshint-2.13.6.tgz" |   resolved "https://registry.npmjs.org/jshint/-/jshint-2.13.6.tgz" | ||||||
| @@ -4213,7 +4279,7 @@ merge-stream@^2.0.0: | |||||||
|   resolved "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz" |   resolved "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz" | ||||||
|   integrity sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w== |   integrity sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w== | ||||||
|  |  | ||||||
| merge2@^1.3.0, merge2@^1.4.1: | merge2@^1.3.0: | ||||||
|   version "1.4.1" |   version "1.4.1" | ||||||
|   resolved "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz" |   resolved "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz" | ||||||
|   integrity sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg== |   integrity sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg== | ||||||
| @@ -4683,16 +4749,15 @@ ra-core@^4.11.2, ra-core@^4.16.2: | |||||||
|     react-is "^17.0.2" |     react-is "^17.0.2" | ||||||
|     react-query "^3.32.1" |     react-query "^3.32.1" | ||||||
|  |  | ||||||
| ra-core@^5.2.0: | ra-core@^5.2.0, ra-core@^5.3.0: | ||||||
|   version "5.2.0" |   version "5.3.0" | ||||||
|   resolved "https://registry.npmjs.org/ra-core/-/ra-core-5.2.0.tgz" |   resolved "https://registry.yarnpkg.com/ra-core/-/ra-core-5.3.0.tgz#02ce7e7c5d7617dbb2b2805803fde1d0c88dc4bd" | ||||||
|   integrity sha512-HAl/rCLeGt55VvfimQd8emuErPIjI5R0f8fEpy7B+jucxGmGTOw2nes+Tf7UIZKjIoN2BHhb/pailjfd2n8Nlg== |   integrity sha512-e5Y0u+P+0Kx2XeYEAUDV63NdlAvMtlrb4ndYLtIbCaIeOx1o6GXlD2BedICaYSb6XXjz+KoNZ1nmWuZ6tS4f7Q== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@tanstack/react-query" "^5.8.4" |     "@tanstack/react-query" "^5.8.4" | ||||||
|     clsx "^2.1.1" |     clsx "^2.1.1" | ||||||
|     date-fns "^3.6.0" |     date-fns "^3.6.0" | ||||||
|     eventemitter3 "^5.0.1" |     eventemitter3 "^5.0.1" | ||||||
|     hotscript "^1.0.12" |  | ||||||
|     inflection "^3.0.0" |     inflection "^3.0.0" | ||||||
|     jsonexport "^3.2.0" |     jsonexport "^3.2.0" | ||||||
|     lodash "~4.17.5" |     lodash "~4.17.5" | ||||||
| @@ -4700,20 +4765,20 @@ ra-core@^5.2.0: | |||||||
|     react-error-boundary "^4.0.13" |     react-error-boundary "^4.0.13" | ||||||
|     react-is "^18.2.0" |     react-is "^18.2.0" | ||||||
|  |  | ||||||
| ra-i18n-polyglot@^5.2.0: | ra-i18n-polyglot@^5.3.0: | ||||||
|   version "5.2.0" |   version "5.3.0" | ||||||
|   resolved "https://registry.npmjs.org/ra-i18n-polyglot/-/ra-i18n-polyglot-5.2.0.tgz" |   resolved "https://registry.yarnpkg.com/ra-i18n-polyglot/-/ra-i18n-polyglot-5.3.0.tgz#ed10be1e3133b3e4a42990d045646ef336faf1ce" | ||||||
|   integrity sha512-w/3voyhjjlrj+ETeHBUj1Xy5trqmwy9g2qACssC/NJ+fAGtq+QhzZZfO27WKIR5LtM0DIT+lM0hdFxtrR5SnSg== |   integrity sha512-Dc2m6MlpF1qHeqH5iJukd7t6oVxnqC/k8AY+HEYQCXCTHhVWkzHDjLOilcKHbgiY6st1p6KoCL93CP8lyEjaqg== | ||||||
|   dependencies: |   dependencies: | ||||||
|     node-polyglot "^2.2.2" |     node-polyglot "^2.2.2" | ||||||
|     ra-core "^5.2.0" |     ra-core "^5.3.0" | ||||||
|  |  | ||||||
| ra-language-english@^5.2.0: | ra-language-english@^5.3.0: | ||||||
|   version "5.2.0" |   version "5.3.0" | ||||||
|   resolved "https://registry.npmjs.org/ra-language-english/-/ra-language-english-5.2.0.tgz" |   resolved "https://registry.yarnpkg.com/ra-language-english/-/ra-language-english-5.3.0.tgz#f2811d9c72513c18ff0f48498bfa92f97911b4ad" | ||||||
|   integrity sha512-vu5BjCPtJ6pEcn4/+/SJkqE5RuFU3DymQbnJSon/KoYUD7F9QOIPmRlVWoCdCKAJPBi8fR6Ffs2aGyAgmx90KQ== |   integrity sha512-Gp8dt4annXcxFd7BJ/EgngLmI74A25++iwE5lS8WqD6yBL3byU5Xwu9+CQ8YQkU9+cQEFhLhkWIGfXXczxSymw== | ||||||
|   dependencies: |   dependencies: | ||||||
|     ra-core "^5.2.0" |     ra-core "^5.3.0" | ||||||
|  |  | ||||||
| ra-language-farsi@^5.0.0: | ra-language-farsi@^5.0.0: | ||||||
|   version "5.0.0" |   version "5.0.0" | ||||||
| @@ -4739,17 +4804,16 @@ ra-language-russian@^4.14.2: | |||||||
|   dependencies: |   dependencies: | ||||||
|     ra-core "^4.16.2" |     ra-core "^4.16.2" | ||||||
|  |  | ||||||
| ra-ui-materialui@^5.2.0: | ra-ui-materialui@^5.3.0: | ||||||
|   version "5.2.0" |   version "5.3.0" | ||||||
|   resolved "https://registry.npmjs.org/ra-ui-materialui/-/ra-ui-materialui-5.2.0.tgz" |   resolved "https://registry.yarnpkg.com/ra-ui-materialui/-/ra-ui-materialui-5.3.0.tgz#14b1c35f98a4d133dd0ef6be85191de71da88bb2" | ||||||
|   integrity sha512-OTjj/CJSxDDojH+e8Ba2eN0S2E672JeVxO5h+KpNg/h8jEF/PTb4Vc3uNF1oXxoXakoMNiKF1DhfTNAZgOfsnQ== |   integrity sha512-jmEqI77LDcEbhrMC45QM1YmUMSOwzNH8dEjr9A/Xm+UW+YnzOFDkVJJumfP3jpFL4wviX48Qy5mBi54t19CYjw== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@tanstack/react-query" "^5.8.4" |     "@tanstack/react-query" "^5.8.4" | ||||||
|     autosuggest-highlight "^3.1.1" |     autosuggest-highlight "^3.1.1" | ||||||
|     clsx "^2.1.1" |     clsx "^2.1.1" | ||||||
|     css-mediaquery "^0.1.2" |     css-mediaquery "^0.1.2" | ||||||
|     dompurify "^2.4.3" |     dompurify "^2.4.3" | ||||||
|     hotscript "^1.0.12" |  | ||||||
|     inflection "^3.0.0" |     inflection "^3.0.0" | ||||||
|     jsonexport "^3.2.0" |     jsonexport "^3.2.0" | ||||||
|     lodash "~4.17.5" |     lodash "~4.17.5" | ||||||
| @@ -4758,19 +4822,19 @@ ra-ui-materialui@^5.2.0: | |||||||
|     react-error-boundary "^4.0.13" |     react-error-boundary "^4.0.13" | ||||||
|     react-transition-group "^4.4.5" |     react-transition-group "^4.4.5" | ||||||
|  |  | ||||||
| react-admin@^5.2.0: | react-admin@^5.3.0: | ||||||
|   version "5.2.0" |   version "5.3.0" | ||||||
|   resolved "https://registry.npmjs.org/react-admin/-/react-admin-5.2.0.tgz" |   resolved "https://registry.yarnpkg.com/react-admin/-/react-admin-5.3.0.tgz#c94dce0e96b949717e3c79df03b7cba133949ba1" | ||||||
|   integrity sha512-EEHz5jhB/OLMN1PPINVY5LF86hOhqMA6n7nTPwK8hJUT56OUfK9MTUdRkn4ucc8Cy+vxT5a3BlmH6BJSnDs5sw== |   integrity sha512-86M0c76ClN7mbMsONLY83lV32hAQREH1hnhHXANZPS4BnRSKSyFjI93Wke+TwUkWlYiqXlEwaYzISfDTTU3xqA== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@emotion/react" "^11.4.1" |     "@emotion/react" "^11.4.1" | ||||||
|     "@emotion/styled" "^11.3.0" |     "@emotion/styled" "^11.3.0" | ||||||
|     "@mui/icons-material" "^5.15.20" |     "@mui/icons-material" "^5.15.20" | ||||||
|     "@mui/material" "^5.15.20" |     "@mui/material" "^5.15.20" | ||||||
|     ra-core "^5.2.0" |     ra-core "^5.3.0" | ||||||
|     ra-i18n-polyglot "^5.2.0" |     ra-i18n-polyglot "^5.3.0" | ||||||
|     ra-language-english "^5.2.0" |     ra-language-english "^5.3.0" | ||||||
|     ra-ui-materialui "^5.2.0" |     ra-ui-materialui "^5.3.0" | ||||||
|     react-hook-form "^7.53.0" |     react-hook-form "^7.53.0" | ||||||
|     react-router "^6.22.0" |     react-router "^6.22.0" | ||||||
|     react-router-dom "^6.22.0" |     react-router-dom "^6.22.0" | ||||||
| @@ -4799,10 +4863,10 @@ react-error-boundary@^4.0.13: | |||||||
|   dependencies: |   dependencies: | ||||||
|     "@babel/runtime" "^7.12.5" |     "@babel/runtime" "^7.12.5" | ||||||
|  |  | ||||||
| react-hook-form@^7.53.0: | react-hook-form@^7.53.0, react-hook-form@^7.53.1: | ||||||
|   version "7.53.0" |   version "7.53.1" | ||||||
|   resolved "https://registry.npmjs.org/react-hook-form/-/react-hook-form-7.53.0.tgz" |   resolved "https://registry.yarnpkg.com/react-hook-form/-/react-hook-form-7.53.1.tgz#3f2cd1ed2b3af99416a4ac674da2d526625add67" | ||||||
|   integrity sha512-M1n3HhqCww6S2hxLxciEXy2oISPnAzxY7gvwVPrtlczTM/1dDadXgUxDpHMrMTblDOcm/AXtXxHwZ3jpg1mqKQ== |   integrity sha512-6aiQeBda4zjcuaugWvim9WsGqisoUk+etmFEsSUMm451/Ic8L/UAb7sRtMj3V+Hdzm6mMjU1VhiSzYUZeBm0Vg== | ||||||
|  |  | ||||||
| "react-is@^16.12.0 || ^17.0.0 || ^18.0.0", react-is@^18.0.0, react-is@^18.2.0, react-is@^18.3.1: | "react-is@^16.12.0 || ^17.0.0 || ^18.0.0", react-is@^18.0.0, react-is@^18.2.0, react-is@^18.3.1: | ||||||
|   version "18.3.1" |   version "18.3.1" | ||||||
| @@ -4833,20 +4897,20 @@ react-refresh@^0.14.2: | |||||||
|   resolved "https://registry.npmjs.org/react-refresh/-/react-refresh-0.14.2.tgz" |   resolved "https://registry.npmjs.org/react-refresh/-/react-refresh-0.14.2.tgz" | ||||||
|   integrity sha512-jCvmsr+1IUSMUyzOkRcvnVbX3ZYC6g9TDrDbFuFmRDq7PD4yaGbLKNQL6k2jnArV8hjYxh7hVhAZB6s9HDGpZA== |   integrity sha512-jCvmsr+1IUSMUyzOkRcvnVbX3ZYC6g9TDrDbFuFmRDq7PD4yaGbLKNQL6k2jnArV8hjYxh7hVhAZB6s9HDGpZA== | ||||||
|  |  | ||||||
| react-router-dom@^6.22.0, react-router-dom@^6.26.2: | react-router-dom@^6.22.0, react-router-dom@^6.27.0: | ||||||
|   version "6.26.2" |   version "6.27.0" | ||||||
|   resolved "https://registry.npmjs.org/react-router-dom/-/react-router-dom-6.26.2.tgz" |   resolved "https://registry.yarnpkg.com/react-router-dom/-/react-router-dom-6.27.0.tgz#8d7972a425fd75f91c1e1ff67e47240c5752dc3f" | ||||||
|   integrity sha512-z7YkaEW0Dy35T3/QKPYB1LjMK2R1fxnHO8kWpUMTBdfVzZrWOiY9a7CtN8HqdWtDUWd5FY6Dl8HFsqVwH4uOtQ== |   integrity sha512-+bvtFWMC0DgAFrfKXKG9Fc+BcXWRUO1aJIihbB79xaeq0v5UzfvnM5houGUm1Y461WVRcgAQ+Clh5rdb1eCx4g== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@remix-run/router" "1.19.2" |     "@remix-run/router" "1.20.0" | ||||||
|     react-router "6.26.2" |     react-router "6.27.0" | ||||||
|  |  | ||||||
| react-router@6.26.2, react-router@^6.22.0, react-router@^6.26.2: | react-router@6.27.0, react-router@^6.22.0, react-router@^6.26.2: | ||||||
|   version "6.26.2" |   version "6.27.0" | ||||||
|   resolved "https://registry.npmjs.org/react-router/-/react-router-6.26.2.tgz" |   resolved "https://registry.yarnpkg.com/react-router/-/react-router-6.27.0.tgz#db292474926c814c996c0ff3ef0162d1f9f60ed4" | ||||||
|   integrity sha512-tvN1iuT03kHgOFnLPfLJ8V95eijteveqdOSk+srqfePtQvqCExB8eHOYnlilbOcyJyKnYkr1vJvf7YqotAJu1A== |   integrity sha512-YA+HGZXz4jaAkVoYBE98VQl+nVzI+cVI2Oj/06F5ZM+0u3TgedN9Y9kmMRo2mnkSK2nCpNQn0DVob4HCsY/WLw== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@remix-run/router" "1.19.2" |     "@remix-run/router" "1.20.0" | ||||||
|  |  | ||||||
| react-shallow-renderer@^16.15.0: | react-shallow-renderer@^16.15.0: | ||||||
|   version "16.15.0" |   version "16.15.0" | ||||||
| @@ -5501,19 +5565,19 @@ typed-array-length@^1.0.6: | |||||||
|     is-typed-array "^1.1.13" |     is-typed-array "^1.1.13" | ||||||
|     possible-typed-array-names "^1.0.0" |     possible-typed-array-names "^1.0.0" | ||||||
|  |  | ||||||
| typescript-eslint@^7.16.1: | typescript-eslint@^8.10.0: | ||||||
|   version "7.18.0" |   version "8.10.0" | ||||||
|   resolved "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-7.18.0.tgz" |   resolved "https://registry.yarnpkg.com/typescript-eslint/-/typescript-eslint-8.10.0.tgz#7f7d51577e9b93538cc8801f2cbfdd66098a00e7" | ||||||
|   integrity sha512-PonBkP603E3tt05lDkbOMyaxJjvKqQrXsnow72sVeOFINDE/qNmnnd+f9b4N+U7W6MXnnYyrhtmF2t08QWwUbA== |   integrity sha512-YIu230PeN7z9zpu/EtqCIuRVHPs4iSlqW6TEvjbyDAE3MZsSl2RXBo+5ag+lbABCG8sFM1WVKEXhlQ8Ml8A3Fw== | ||||||
|   dependencies: |   dependencies: | ||||||
|     "@typescript-eslint/eslint-plugin" "7.18.0" |     "@typescript-eslint/eslint-plugin" "8.10.0" | ||||||
|     "@typescript-eslint/parser" "7.18.0" |     "@typescript-eslint/parser" "8.10.0" | ||||||
|     "@typescript-eslint/utils" "7.18.0" |     "@typescript-eslint/utils" "8.10.0" | ||||||
|  |  | ||||||
| typescript@^5.4.5: | typescript@^5.6.3: | ||||||
|   version "5.6.2" |   version "5.6.3" | ||||||
|   resolved "https://registry.npmjs.org/typescript/-/typescript-5.6.2.tgz" |   resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.6.3.tgz#5f3449e31c9d94febb17de03cc081dd56d81db5b" | ||||||
|   integrity sha512-NW8ByodCSNCwZeghjN3o+JX5OFH0Ojg6sadjEKY4huZ52TqbJTJnDo5+Tw98lSy63NZvi4n+ez5m2u5d4PkZyw== |   integrity sha512-hjcS1mhfuyi4WW8IWtjP7brDrG2cuDZukyrYrSauoXGNgx0S7zceP07adYkJycEr56BOUTNPzbInooiN3fn1qw== | ||||||
|  |  | ||||||
| unbox-primitive@^1.0.2: | unbox-primitive@^1.0.2: | ||||||
|   version "1.0.2" |   version "1.0.2" | ||||||
| @@ -5580,10 +5644,10 @@ v8-to-istanbul@^9.0.1: | |||||||
|     "@types/istanbul-lib-coverage" "^2.0.1" |     "@types/istanbul-lib-coverage" "^2.0.1" | ||||||
|     convert-source-map "^2.0.0" |     convert-source-map "^2.0.0" | ||||||
|  |  | ||||||
| vite-plugin-version-mark@^0.1.0: | vite-plugin-version-mark@^0.1.2: | ||||||
|   version "0.1.0" |   version "0.1.2" | ||||||
|   resolved "https://registry.npmjs.org/vite-plugin-version-mark/-/vite-plugin-version-mark-0.1.0.tgz" |   resolved "https://registry.yarnpkg.com/vite-plugin-version-mark/-/vite-plugin-version-mark-0.1.2.tgz#85b2315cffce7ac2d83b6afceb313d0d1c47c4d4" | ||||||
|   integrity sha512-Na9KG6anEwAQVUKm3DOl4AufzvkgWlOogBnGAMR6YEcMMOYMl45/0KqMuvNxGd0WR8aW8cpUiGHDCFlIZ15YeQ== |   integrity sha512-+mPRTwx6SOEfEq3/y1u+O0ju5Lz0Ac27Fx6hE5Xn1fMpIQfo3aERPB2/xxYI8lIa8GPHCX5c8ApqBKWwFNHxwA== | ||||||
|  |  | ||||||
| vite@^5.4.6: | vite@^5.4.6: | ||||||
|   version "5.4.6" |   version "5.4.6" | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user