Compare commits
	
		
			55 Commits
		
	
	
		
			v0.10.3-et
			...
			v0.10.3-et
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 1d5511421d | ||
|   | b925c63171 | ||
|   | 6faebaf9df | ||
|   | c698f57395 | ||
|   | 9adc13e722 | ||
|   | a04b24a5d5 | ||
|   | cd1ca7c039 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 86b4987b7f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a7cf647669 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 80d40d2fb5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b19e961a35 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 08f5f8ebd2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1edf196049 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1f66b4d14a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cec5b0af9a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2b0e1e7c0e | ||
|   | a613a88232 | ||
|   | 7afce71bef | ||
|   | fe4ba22a03 | ||
|   | eb35d9e122 | ||
|   | fa79fecc9d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 61366b3792 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b55033d983 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b1f42988c8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ef05b366c3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b3d52e7d23 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d1c4250b46 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8d44077b24 | ||
|   | 44e8b82412 | ||
|   | 791ae2c869 | ||
|   | ee3753466a | ||
|   | b7dc703157 | ||
|   | 2eca0dcc33 | ||
|   | d2219c1667 | ||
|   | 132ea6f97f | ||
|   | 28ef08de03 | ||
|   | 865e53388e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3a105bb8c7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | edcda7a202 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | edd69273e2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 444bfacbd9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 970e0a550f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b3ef68d66e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 31382a42ee | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1a7748d1ef | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 039b28cc5c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 57eae3edb3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dadc9416c0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | eab2342114 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9cf2f83936 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d823856873 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9b96c7cec8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f211aba873 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c0fc2d8937 | ||
|   | a88b397748 | 
							
								
								
									
										3
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -4,13 +4,16 @@ updates: | ||||
|     directory: "/" | ||||
|     schedule: | ||||
|       interval: "weekly" | ||||
|     open-pull-requests-limit: 10 | ||||
|  | ||||
|   - package-ecosystem: "docker" | ||||
|     directory: "/" | ||||
|     schedule: | ||||
|       interval: "weekly" | ||||
|     open-pull-requests-limit: 10 | ||||
|  | ||||
|   - package-ecosystem: "github-actions" | ||||
|     directory: "/" | ||||
|     schedule: | ||||
|       interval: "weekly" | ||||
|     open-pull-requests-limit: 10 | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/workflows/workflow.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/workflow.yml
									
									
									
									
										vendored
									
									
								
							| @@ -25,10 +25,6 @@ jobs: | ||||
|           cache: yarn | ||||
|       - name: Install dependencies | ||||
|         run: yarn install --immutable --network-timeout=300000 | ||||
|       - name: Set version into manifest.json | ||||
|         run: | | ||||
|           TAG=$(git describe --tags --abbrev=0 || echo "latest") | ||||
|           sed -i "s|\"icons\"|\"version\": \"$TAG\",\\n  \"icons\"|g" public/manifest.json | ||||
|       - name: Build | ||||
|         run: yarn build --base=${{ env.base_path }} | ||||
|       - uses: actions/upload-artifact@v4 | ||||
|   | ||||
							
								
								
									
										96
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										96
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,4 +1,16 @@ | ||||
| # Synapse Admin UI [](https://github.com/Awesome-Technologies/synapse-admin/blob/master/LICENSE) | ||||
| <p align="center"> | ||||
|   <img alt="Synapse Admin Logo" src="./public/images/logo.webp" height="140" /> | ||||
|   <h3 align="center"> | ||||
|     Synapse Admin<br> | ||||
|     <a href="https://matrix.to/#/#synapse-admin:etke.cc"> | ||||
|       <img alt="Community room" src="https://img.shields.io/badge/room-community_room-green?logo=matrix&label=%23synapse-admin%3Aetke.cc"> | ||||
|     </a><br> | ||||
|     <img alt="License" src="https://img.shields.io/github/license/etkecc/synapse-admin"> | ||||
|   </h3> | ||||
|   <p align="center">Manager your Synapse homeserver with ease</p> | ||||
| </p> | ||||
|  | ||||
| --- | ||||
|  | ||||
|  | ||||
|  | ||||
| @@ -23,7 +35,7 @@ This project is built using [react-admin](https://marmelab.com/react-admin/). | ||||
|     * [Steps for 1)](#steps-for-1) | ||||
|     * [Steps for 2)](#steps-for-2) | ||||
|     * [Steps for 3)](#steps-for-3) | ||||
|   * [Serving Synapse-Admin on a different path](#serving-synapse-admin-on-a-different-path) | ||||
|   * [Serving Synapse Admin on a different path](#serving-synapse-admin-on-a-different-path) | ||||
| * [Development](#development-1) | ||||
|  | ||||
| <!-- vim-markdown-toc --> | ||||
| @@ -54,8 +66,9 @@ The following changes are already implemented: | ||||
| * [Fix redirect URL after user creation](https://github.com/etkecc/synapse-admin/pull/16) | ||||
| * [Display actual Synapse errors](https://github.com/etkecc/synapse-admin/pull/17) | ||||
| * [Fix base_url being undefined on unsuccessful login](https://github.com/etkecc/synapse-admin/pull/18) | ||||
| * [Put the version into manifest.json](https://github.com/Awesome-Technologies/synapse-admin/issues/507) (CI only) | ||||
| * [Federation page improvements](https://github.com/Awesome-Technologies/synapse-admin/pull/583) (using theme colors) | ||||
| * [Put the version into manifest.json](https://github.com/Awesome-Technologies/synapse-admin/issues/507) (later replaced | ||||
| with a proper manifest.json generation on build) | ||||
| * [Federation page improvements](https://github.com/Awesome-Technologies/synapse-admin/pull/583) (using icons) | ||||
| * [Add UI option to block deleted rooms from being rejoined](https://github.com/etkecc/synapse-admin/pull/26) | ||||
| * [Fix required fields check on Bulk registration CSV upload](https://github.com/etkecc/synapse-admin/pull/32) | ||||
| * [Fix requests with invalid MXIDs on Bulk registration](https://github.com/etkecc/synapse-admin/pull/33) | ||||
| @@ -71,6 +84,13 @@ The following changes are already implemented: | ||||
| * [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) | ||||
| * [Fix room state events display](https://github.com/etkecc/synapse-admin/pull/100) | ||||
| * [Sanitize CSV on import](https://github.com/etkecc/synapse-admin/pull/101) | ||||
| * Allow setting version using `SYNAPSE_ADMIN_VERSION` environment variable on build (if git is not available) | ||||
| * [Add option to control user's experimental features](https://github.com/etkecc/synapse-admin/pull/111) | ||||
| * [Add random password generation on user create/edit form](https://github.com/etkecc/synapse-admin/pull/123) | ||||
| * [Add option to set user's rate limits](https://github.com/etkecc/synapse-admin/pull/125) | ||||
| * [Support configuration via /.well-known/matrix/client](https://github.com/etkecc/synapse-admin/pull/126) | ||||
|  | ||||
| _the list will be updated as new changes are added_ | ||||
|  | ||||
| @@ -87,7 +107,11 @@ After that open `http://localhost:5173` in your browser, login using the followi | ||||
|  | ||||
| ## Configuration | ||||
|  | ||||
| You can use `config.json` file to configure synapse-admin | ||||
| You can use `config.json` file to configure Synapse Admin instance, | ||||
| and `/.well-known/matrix/client` file to provide Synapse Admin configuration specifically for your homeserver. | ||||
| In the latter case, any instance of Synapse Admin will automatically pick up the configuration from the homeserver. | ||||
| Note that configuration inside the `/.well-known/matrix/client` file should go under the `cc.etke.synapse-admin` key, | ||||
| and it will override the configuration from the `config.json` file. | ||||
|  | ||||
| The `config.json` can be injected into a Docker container using a bind mount. | ||||
|  | ||||
| @@ -112,6 +136,16 @@ Edit `config.json` to restrict either to a single homeserver: | ||||
| } | ||||
| ``` | ||||
|  | ||||
| similar for `/.well-known/matrix/client`: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "cc.etke.synapse-admin": { | ||||
|     "restrictBaseUrl": "https://your-matrixs-erver.example.com" | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| or to a list of homeservers: | ||||
|  | ||||
| ```json | ||||
| @@ -120,6 +154,16 @@ or to a list of homeservers: | ||||
| } | ||||
| ``` | ||||
|  | ||||
| similar for `/.well-known/matrix/client`: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "cc.etke.synapse-admin": { | ||||
|     "restrictBaseUrl": ["https://your-first-matrix-server.example.com", "https://your-second-matrix-server.example.com"] | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### Protecting appservice managed users | ||||
|  | ||||
| To avoid accidental adjustments of appservice-managed users (e.g., puppets created by a bridge) and breaking the bridge, | ||||
| @@ -133,6 +177,16 @@ Example for [mautrix-telegram](https://github.com/mautrix/telegram) | ||||
| } | ||||
| ``` | ||||
|  | ||||
| similar for `/.well-known/matrix/client`: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "cc.etke.synapse-admin": { | ||||
|     "asManagedUsers": ["^@telegram_[a-zA-Z0-9]+:example\\.com$"] | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### Adding custom menu items | ||||
|  | ||||
| You can add custom menu items to the main menu by providing a `menu` array in the `config.json`. | ||||
| @@ -149,13 +203,29 @@ You can add custom menu items to the main menu by providing a `menu` array in th | ||||
| } | ||||
| ``` | ||||
|  | ||||
| similar for `/.well-known/matrix/client`: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "cc.etke.synapse-admin": { | ||||
|     "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 | ||||
|  | ||||
| **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`.~~ | ||||
| ~~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 | ||||
| { | ||||
| @@ -163,6 +233,16 @@ Where `icon` is one of the [preloaded icons](./src/components/icons.ts) | ||||
| } | ||||
| ``` | ||||
|  | ||||
| similar for `/.well-known/matrix/client`: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "cc.etke.synapse-admin": { | ||||
|     "supportURL": "https://example.com/support" | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## Usage | ||||
|  | ||||
| ### Supported Synapse | ||||
| @@ -242,7 +322,7 @@ You have three options: | ||||
|  | ||||
| - browse to http://localhost:8080 | ||||
|  | ||||
| ### Serving Synapse-Admin on a different path | ||||
| ### Serving Synapse Admin on a different path | ||||
|  | ||||
| The path prefix where synapse-admin is served can only be changed during the build step. | ||||
|  | ||||
| @@ -250,7 +330,7 @@ If you downloaded the source code, use `yarn build --base=/my-prefix` to set a p | ||||
|  | ||||
| If you want to build your own Docker container, use the `BASE_PATH` argument. | ||||
|  | ||||
| We do not support directly changing the path where Synapse-Admin is served in the pre-built Docker container. Instead please use a reverse proxy if you need to move Synapse-Admin to a different base path. If you want to serve multiple applications with different paths on the same domain, you need a reverse proxy anyway. | ||||
| We do not support directly changing the path where Synapse Admin is served in the pre-built Docker container. Instead please use a reverse proxy if you need to move Synapse Admin to a different base path. If you want to serve multiple applications with different paths on the same domain, you need a reverse proxy anyway. | ||||
|  | ||||
| Example for Traefik: | ||||
|  | ||||
|   | ||||
| @@ -4,17 +4,14 @@ | ||||
|     <meta charset="utf-8" /> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1" /> | ||||
|     <meta name="theme-color" content="#000000" /> | ||||
|     <meta | ||||
|       name="description" | ||||
|       content="Synapse-Admin" | ||||
|     /> | ||||
|     <meta name="description" content="Synapse Admin" /> | ||||
|     <!-- | ||||
|       manifest.json provides metadata used when your web app is installed on a | ||||
|       user's mobile device or desktop. See https://developers.google.com/web/fundamentals/web-app-manifest/ | ||||
|     --> | ||||
|     <link rel="manifest" href="./manifest.json" /> | ||||
|     <link rel="shortcut icon" href="./favicon.ico" /> | ||||
|     <title>Synapse-Admin</title> | ||||
|     <title>Synapse Admin</title> | ||||
|     <style> | ||||
|       body { | ||||
|         margin: 0; | ||||
|   | ||||
							
								
								
									
										46
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										46
									
								
								package.json
									
									
									
									
									
								
							| @@ -11,25 +11,25 @@ | ||||
|     "url": "https://github.com/etkecc/synapse-admin" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@eslint/js": "^9.7.0", | ||||
|     "@eslint/js": "^9.13.0", | ||||
|     "@testing-library/dom": "^10.0.0", | ||||
|     "@testing-library/jest-dom": "^6.6.2", | ||||
|     "@testing-library/jest-dom": "^6.6.3", | ||||
|     "@testing-library/react": "^16.0.0", | ||||
|     "@testing-library/user-event": "^14.5.2", | ||||
|     "@types/jest": "^29.5.13", | ||||
|     "@types/lodash": "^4.17.7", | ||||
|     "@types/node": "^22.7.7", | ||||
|     "@types/jest": "^29.5.14", | ||||
|     "@types/lodash": "^4.17.13", | ||||
|     "@types/node": "^22.8.7", | ||||
|     "@types/papaparse": "^5.3.15", | ||||
|     "@types/react": "^18.3.3", | ||||
|     "@typescript-eslint/eslint-plugin": "^8.10.0", | ||||
|     "@typescript-eslint/parser": "^8.10.0", | ||||
|     "@vitejs/plugin-react": "^4.3.2", | ||||
|     "eslint": "^8.57.0", | ||||
|     "@types/react": "^18.3.12", | ||||
|     "@typescript-eslint/eslint-plugin": "^8.11.0", | ||||
|     "@typescript-eslint/parser": "^8.11.0", | ||||
|     "@vitejs/plugin-react": "^4.3.3", | ||||
|     "eslint": "^9.13.0", | ||||
|     "eslint-config-prettier": "^9.1.0", | ||||
|     "eslint-plugin-import": "^2.31.0", | ||||
|     "eslint-plugin-jsx-a11y": "^6.9.0", | ||||
|     "eslint-plugin-jsx-a11y": "^6.10.2", | ||||
|     "eslint-plugin-prettier": "^5.2.1", | ||||
|     "eslint-plugin-unused-imports": "^3.2.0", | ||||
|     "eslint-plugin-unused-imports": "^4.1.4", | ||||
|     "eslint-plugin-yaml": "^1.0.3", | ||||
|     "jest": "^29.7.0", | ||||
|     "jest-environment-jsdom": "^29.7.0", | ||||
| @@ -39,8 +39,8 @@ | ||||
|     "ts-jest": "^29.2.5", | ||||
|     "ts-node": "^10.9.2", | ||||
|     "typescript": "^5.6.3", | ||||
|     "typescript-eslint": "^8.10.0", | ||||
|     "vite": "^5.4.6", | ||||
|     "typescript-eslint": "^8.12.2", | ||||
|     "vite": "^5.4.10", | ||||
|     "vite-plugin-version-mark": "^0.1.2" | ||||
|   }, | ||||
|   "dependencies": { | ||||
| @@ -48,22 +48,22 @@ | ||||
|     "@emotion/styled": "^11.13.0", | ||||
|     "@haleos/ra-language-german": "^1.0.0", | ||||
|     "@haxqer/ra-language-chinese": "^4.16.2", | ||||
|     "@mui/icons-material": "^6.1.1", | ||||
|     "@mui/material": "^6.1.1", | ||||
|     "@mui/icons-material": "^6.1.5", | ||||
|     "@mui/material": "^6.1.6", | ||||
|     "@mui/utils": "^5.16.6", | ||||
|     "@tanstack/react-query": "^5.56.2", | ||||
|     "@tanstack/react-query": "^5.59.19", | ||||
|     "history": "^5.3.0", | ||||
|     "lodash": "^4.17.21", | ||||
|     "papaparse": "^5.4.1", | ||||
|     "ra-core": "^5.3.0", | ||||
|     "ra-i18n-polyglot": "^5.3.0", | ||||
|     "ra-language-english": "^5.3.0", | ||||
|     "ra-core": "^5.3.2", | ||||
|     "ra-i18n-polyglot": "^5.3.2", | ||||
|     "ra-language-english": "^5.3.2", | ||||
|     "ra-language-farsi": "^5.0.0", | ||||
|     "ra-language-french": "^5.2.0", | ||||
|     "ra-language-french": "^5.3.1", | ||||
|     "ra-language-italian": "^3.13.1", | ||||
|     "ra-language-russian": "^4.14.2", | ||||
|     "react": "^18.3.1", | ||||
|     "react-admin": "^5.3.0", | ||||
|     "react-admin": "^5.3.1", | ||||
|     "react-dom": "^18.3.1", | ||||
|     "react-hook-form": "^7.53.1", | ||||
|     "react-is": "^18.3.1", | ||||
| @@ -73,7 +73,7 @@ | ||||
|   "scripts": { | ||||
|     "start": "vite serve", | ||||
|     "build": "vite build", | ||||
|     "lint": "eslint --ignore-path .gitignore --ext .ts,.tsx,.yml,.yaml .", | ||||
|     "lint": "ESLINT_USE_FLAT_CONFIG=false eslint --ignore-path .gitignore --ignore-pattern testdata/ --ext .ts,.tsx,.yml,.yaml .", | ||||
|     "fix": "yarn lint --fix", | ||||
|     "test": "yarn jest", | ||||
|     "test:watch": "yarn jest --watch" | ||||
|   | ||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 1.1 KiB After Width: | Height: | Size: 4.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								public/images/logo.webp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								public/images/logo.webp
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 16 KiB | 
| @@ -1,15 +0,0 @@ | ||||
| { | ||||
|   "short_name": "Synapse-Admin", | ||||
|   "name": "Synapse-Admin", | ||||
|   "icons": [ | ||||
|     { | ||||
|       "src": "favicon.ico", | ||||
|       "sizes": "64x64 32x32 24x24 16x16", | ||||
|       "type": "image/x-icon" | ||||
|     } | ||||
|   ], | ||||
|   "start_url": ".", | ||||
|   "display": "standalone", | ||||
|   "theme_color": "#000000", | ||||
|   "background_color": "#ffffff" | ||||
| } | ||||
| @@ -1,18 +1,6 @@ | ||||
| import { createContext, useContext } from "react"; | ||||
|  | ||||
| interface AppContextType { | ||||
|   restrictBaseUrl: string | string[]; | ||||
|   asManagedUsers: string[]; | ||||
|   supportURL: string; | ||||
|   menu: MenuItem[]; | ||||
| } | ||||
|  | ||||
| interface MenuItem { | ||||
|   label: string; | ||||
|   icon: string; | ||||
|   url: string; | ||||
| } | ||||
| import { Config } from "./components/config"; | ||||
|  | ||||
| export const AppContext = createContext({}); | ||||
|  | ||||
| export const useAppContext = () => useContext(AppContext) as AppContextType; | ||||
| export const useAppContext = () => useContext(AppContext) as Config; | ||||
|   | ||||
| @@ -2,6 +2,7 @@ import { AppBar, TitlePortal, InspectorButton, Confirm, Layout, Logout, Menu, us | ||||
| import { LoginMethod } from "../pages/LoginPage"; | ||||
| import { useEffect, useState, Suspense } from "react"; | ||||
| import { Icons, DefaultIcon } from "./icons"; | ||||
| import { ClearConfig } from "./config"; | ||||
|  | ||||
| const AdminUserMenu = () => { | ||||
|   const [open, setOpen] = useState(false); | ||||
| @@ -21,8 +22,7 @@ const AdminUserMenu = () => { | ||||
|  | ||||
|   const handleDialogClose = () => { | ||||
|     setOpen(false); | ||||
|     localStorage.removeItem("access_token"); | ||||
|     localStorage.removeItem("login_type"); | ||||
|     ClearConfig(); | ||||
|     window.location.reload(); | ||||
|   }; | ||||
|  | ||||
|   | ||||
							
								
								
									
										95
									
								
								src/components/ExperimentalFeatures.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								src/components/ExperimentalFeatures.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,95 @@ | ||||
| import { useRecordContext } from "react-admin"; | ||||
| import { useNotify } from "react-admin"; | ||||
| import { useDataProvider } from "react-admin"; | ||||
| import { useState, useEffect } from "react"; | ||||
| import { Stack, Switch, Typography } from "@mui/material"; | ||||
| import { ExperimentalFeaturesModel, SynapseDataProvider } from "../synapse/dataProvider"; | ||||
|  | ||||
| const experimentalFeaturesMap = { | ||||
|     msc3881: "enable remotely toggling push notifications for another client", | ||||
|     msc3575: "enable experimental sliding sync support", | ||||
| }; | ||||
| const ExperimentalFeatureRow = (props: { featureKey: string, featureValue: boolean, updateFeature: (feature_name: string, feature_value: boolean) => void}) => { | ||||
|   const featureKey = props.featureKey; | ||||
|   const featureValue = props.featureValue; | ||||
|   const featureDescription = experimentalFeaturesMap[featureKey] ?? ""; | ||||
|   const [checked, setChecked] = useState(featureValue); | ||||
|  | ||||
|   const handleChange = (event: React.ChangeEvent<HTMLInputElement>) => { | ||||
|     setChecked(event.target.checked); | ||||
|     props.updateFeature(featureKey, event.target.checked); | ||||
|   }; | ||||
|  | ||||
|   return <Stack | ||||
|       direction="row" | ||||
|       spacing={2} | ||||
|       alignItems="start" | ||||
|       sx={{ | ||||
|           padding: 2, | ||||
|       }} | ||||
|   > | ||||
|     <Switch checked={checked} onChange={handleChange} /> | ||||
|     <Stack> | ||||
|       <Typography | ||||
|           variant="subtitle1" | ||||
|           sx={{ | ||||
|               fontWeight: "medium", | ||||
|               color: "text.primary" | ||||
|           }} | ||||
|       > | ||||
|           {featureKey} | ||||
|       </Typography> | ||||
|       <Typography | ||||
|           variant="body2" | ||||
|           color="text.secondary" | ||||
|       > | ||||
|           {featureDescription} | ||||
|       </Typography> | ||||
|     </Stack> | ||||
|   </Stack> | ||||
| } | ||||
|  | ||||
| export const ExperimentalFeaturesList = () => { | ||||
|   const record = useRecordContext(); | ||||
|   const notify = useNotify(); | ||||
|   const dataProvider = useDataProvider() as SynapseDataProvider; | ||||
|   const [features, setFeatures] = useState({}); | ||||
|   if (!record) { | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   useEffect(() => { | ||||
|     const fetchFeatures = async () => { | ||||
|       const features = await dataProvider.getFeatures(record.id); | ||||
|       setFeatures(features); | ||||
|     } | ||||
|  | ||||
|     fetchFeatures(); | ||||
|   }, []); | ||||
|  | ||||
|   const updateFeature = async (feature_name: string, feature_value: boolean) => { | ||||
|     const updatedFeatures = {...features, [feature_name]: feature_value} as ExperimentalFeaturesModel; | ||||
|     setFeatures(updatedFeatures); | ||||
|     const reponse = await dataProvider.updateFeatures(record.id, updatedFeatures); | ||||
|     notify("ra.notification.updated", { | ||||
|         messageArgs: { smart_count: 1 }, | ||||
|         type: "success", | ||||
|     }); | ||||
|   }; | ||||
|  | ||||
|   return <> | ||||
|     <Stack | ||||
|       direction="column" | ||||
|       spacing={1} | ||||
|     > | ||||
|       {Object.keys(features).map((featureKey: string) => | ||||
|         <ExperimentalFeatureRow | ||||
|           key={featureKey} | ||||
|           featureKey={featureKey} | ||||
|           featureValue={features[featureKey]} | ||||
|           updateFeature={updateFeature} | ||||
|         /> | ||||
|       )} | ||||
|     </Stack> | ||||
|   </> | ||||
| } | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { Box, Link, Typography } from "@mui/material"; | ||||
| import { Avatar, Box, Link, Typography } from "@mui/material"; | ||||
| import { useEffect, useState } from "react"; | ||||
|  | ||||
| const Footer = () => { | ||||
| @@ -22,16 +22,17 @@ const Footer = () => { | ||||
|       borderColor: '#ddd', | ||||
|       p: 1, | ||||
|     }}> | ||||
|     <Typography variant="body2"> | ||||
|     <Typography variant="body2" component="div"> | ||||
|       <Avatar src="./images/logo.webp" sx={{ width: "1rem", height: "1rem", display: "inline-block", verticalAlign: "sub" }} /> | ||||
|       <Link sx={{ color: "#888", textDecoration: 'none' }} href="https://github.com/etkecc/synapse-admin" target="_blank"> | ||||
|         Synapse-Admin | ||||
|         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> | ||||
|         (originally developed by Awesome Technologies Innovationslabor GmbH). | ||||
|       </Link> <Link sx={{ fontWeight: 'bold', color: "#000", textDecoration: 'none' }} href="https://matrix.to/#/#synapse-admin:etke.cc" target="_blank">#synapse-admin:etke.cc</Link> | ||||
|     </Typography> | ||||
|   </Box> | ||||
|   ); | ||||
|   | ||||
| @@ -121,6 +121,7 @@ const FilePicker = () => { | ||||
|  | ||||
|   const verifyCsv = ({ data, meta, errors }: ParseResult<ImportLine>, { setValues, setStats, setError }) => { | ||||
|     /* First, verify the presence of required fields */ | ||||
|     meta.fields = meta.fields?.map(f => f.trim().toLowerCase()); | ||||
|     const missingFields = expectedFields.filter(eF => !meta.fields?.find(mF => eF === mF)); | ||||
|  | ||||
|     if (missingFields.length > 0) { | ||||
| @@ -147,6 +148,15 @@ const FilePicker = () => { | ||||
|     }; | ||||
|  | ||||
|     const errorMessages = errors.map(e => e.message); | ||||
|     // sanitize the data first | ||||
|     data = data.map(line => { | ||||
|       const newLine = {} as ImportLine; | ||||
|       for (const [key, value] of Object.entries(line)) { | ||||
|         newLine[key.trim().toLowerCase()] = value; | ||||
|       } | ||||
|       return newLine; | ||||
|     }); | ||||
|     // process the data | ||||
|     data.forEach((line, idx) => { | ||||
|       if (line.user_type === undefined || line.user_type === "") { | ||||
|         stats.user_types.default++; | ||||
| @@ -173,6 +183,7 @@ const FilePicker = () => { | ||||
|           line[f] = true; // we need true booleans instead of strings | ||||
|         } else { | ||||
|           if (line[f] !== "false" && line[f] !== "") { | ||||
|             console.log("invalid value", line[f], "for field " + f + " in row " + idx); | ||||
|             errorMessages.push( | ||||
|               translate("import_users.error.invalid_value", { | ||||
|                 field: f, | ||||
|   | ||||
| @@ -13,7 +13,7 @@ const LoginFormBox = styled(Box)(({ theme }) => ({ | ||||
|   backgroundSize: "cover", | ||||
|  | ||||
|   [`& .card`]: { | ||||
|     maxWidth: "30rem", | ||||
|     width: "30rem", | ||||
|     marginTop: "6rem", | ||||
|     marginBottom: "6rem", | ||||
|   }, | ||||
|   | ||||
							
								
								
									
										95
									
								
								src/components/UserRateLimits.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								src/components/UserRateLimits.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,95 @@ | ||||
| import { Stack, Typography } from "@mui/material"; | ||||
| import { useEffect, useState } from "react"; | ||||
| import { useDataProvider, useNotify, useRecordContext, useTranslate } from "react-admin"; | ||||
| import { TextField } from "@mui/material"; | ||||
| import { useFormContext } from "react-hook-form"; | ||||
|  | ||||
| const RateLimitRow = ({ limit, value, updateRateLimit }: { limit: string, value: any, updateRateLimit: (limit: string, value: any) => void }) => { | ||||
|   const translate = useTranslate(); | ||||
|  | ||||
|   const handleChange = (event: React.ChangeEvent<HTMLInputElement>) => { | ||||
|     const value = parseInt(event.target.value); | ||||
|     if (isNaN(value)) { | ||||
|       updateRateLimit(limit, null); | ||||
|       return; | ||||
|     } | ||||
|     updateRateLimit(limit, value); | ||||
|   }; | ||||
|  | ||||
|   return <Stack | ||||
|     spacing={1} | ||||
|     alignItems="start" | ||||
|     sx={{ | ||||
|         padding: 2, | ||||
|     }} | ||||
|   > | ||||
|     <TextField | ||||
|       id="outlined-number" | ||||
|       type="number" | ||||
|       value={value} | ||||
|       onChange={handleChange} | ||||
|       slotProps={{ | ||||
|         inputLabel: { | ||||
|           shrink: true, | ||||
|         }, | ||||
|       }} | ||||
|       label={translate(`resources.users.limits.${limit}`)} | ||||
|     /> | ||||
|     <Stack> | ||||
|       <Typography | ||||
|         variant="body2" | ||||
|         color="text.secondary" | ||||
|       > | ||||
|         {translate(`resources.users.limits.${limit}_text`)} | ||||
|       </Typography> | ||||
|     </Stack> | ||||
|   </Stack> | ||||
| } | ||||
|  | ||||
| export const UserRateLimits = () => { | ||||
|   const translate = useTranslate(); | ||||
|   const notify = useNotify(); | ||||
|   const record = useRecordContext(); | ||||
|   const form = useFormContext(); | ||||
|   const dataProvider = useDataProvider(); | ||||
|   const [rateLimits, setRateLimits] = useState({ | ||||
|     messages_per_second: "", // we are setting string here to make the number field empty by default, null is prohibited by the field validation | ||||
|     burst_count: "", | ||||
|   }); | ||||
|  | ||||
|   if (!record) { | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   useEffect(() => { | ||||
|       const fetchRateLimits = async () => { | ||||
|           const rateLimits = await dataProvider.getRateLimits(record.id); | ||||
|           if (Object.keys(rateLimits).length > 0) { | ||||
|             setRateLimits(rateLimits); | ||||
|           } | ||||
|       } | ||||
|  | ||||
|       fetchRateLimits(); | ||||
|   }, []); | ||||
|  | ||||
|   const updateRateLimit = async (limit: string, value: any) => { | ||||
|     let updatedRateLimits = { ...rateLimits, [limit]: value }; | ||||
|     setRateLimits(updatedRateLimits); | ||||
|     form.setValue(`rates.${limit}`, value, { shouldDirty: true }); | ||||
|   }; | ||||
|  | ||||
|   return <> | ||||
|     <Stack | ||||
|       direction="column" | ||||
|     > | ||||
|       {Object.keys(rateLimits).map((limit: string) => | ||||
|         <RateLimitRow | ||||
|           key={limit} | ||||
|           limit={limit} | ||||
|           value={rateLimits[limit]} | ||||
|           updateRateLimit={updateRateLimit} | ||||
|         /> | ||||
|       )} | ||||
|     </Stack> | ||||
|   </> | ||||
| }; | ||||
							
								
								
									
										87
									
								
								src/components/config.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								src/components/config.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | ||||
| import storage from "../storage"; | ||||
|  | ||||
| export interface Config { | ||||
|   restrictBaseUrl: string | string[]; | ||||
|   asManagedUsers: string[]; | ||||
|   supportURL: string; | ||||
|   menu: MenuItem[]; | ||||
| } | ||||
|  | ||||
| export interface MenuItem { | ||||
|   label: string; | ||||
|   icon: string; | ||||
|   url: string; | ||||
| } | ||||
|  | ||||
| export const WellKnownKey = "cc.etke.synapse-admin"; | ||||
|  | ||||
| export const LoadConfig = (context: Config): Config => { | ||||
|   if (context.restrictBaseUrl) { | ||||
|     storage.setItem("restrict_base_url", JSON.stringify(context.restrictBaseUrl)); | ||||
|   } | ||||
|  | ||||
|   if (context.asManagedUsers) { | ||||
|     storage.setItem("as_managed_users", JSON.stringify(context.asManagedUsers)); | ||||
|   } | ||||
|  | ||||
|   let menu: MenuItem[] = []; | ||||
|   if (context.menu) { | ||||
|     menu = context.menu; | ||||
|   } | ||||
|   if (context.supportURL) { | ||||
|     const migratedSupportURL = { | ||||
|       label: "Contact support", | ||||
|       icon: "SupportAgent", | ||||
|       url: context.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)); | ||||
|   } | ||||
|  | ||||
|   // below we try to calculate "final" config, which will contain values from context and already set values in storage | ||||
|   // because LoadConfig could be called multiple times to get config from different sources | ||||
|   let finalRestrictBaseUrl: string | string[] = ""; | ||||
|   try { | ||||
|     finalRestrictBaseUrl = JSON.parse(storage.getItem("restrict_base_url") || ""); | ||||
|     if (Array.isArray(finalRestrictBaseUrl) && finalRestrictBaseUrl.length == 1) { | ||||
|       finalRestrictBaseUrl = finalRestrictBaseUrl[0]; | ||||
|     } | ||||
|   } catch (e) {} | ||||
|   let finalAsManagedUsers: string[] = []; | ||||
|   try { | ||||
|     finalAsManagedUsers = JSON.parse(storage.getItem("as_managed_users") || ""); | ||||
|   } catch (e) {} | ||||
|  | ||||
|   let finalMenu: MenuItem[] = []; | ||||
|   try { | ||||
|     finalMenu = JSON.parse(storage.getItem("menu") || ""); | ||||
|   } catch (e) {} | ||||
|  | ||||
|   return { | ||||
|     restrictBaseUrl: finalRestrictBaseUrl, | ||||
|     asManagedUsers: finalAsManagedUsers, | ||||
|     supportURL: storage.getItem("support_url") || "", | ||||
|     menu: finalMenu, | ||||
|   } as Config; | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ClearConfig = () => { | ||||
|   // config.json | ||||
|   storage.removeItem("restrict_base_url"); | ||||
|   storage.removeItem("as_managed_users"); | ||||
|   storage.removeItem("support_url"); | ||||
|   storage.removeItem("menu"); | ||||
|  | ||||
|   // session | ||||
|   storage.removeItem("home_server"); | ||||
|   storage.removeItem("base_url"); | ||||
|   storage.removeItem("user_id"); | ||||
|   storage.removeItem("device_id"); | ||||
|   storage.removeItem("access_token"); | ||||
|   storage.removeItem("login_type"); | ||||
| } | ||||
| @@ -55,7 +55,7 @@ const de: SynapseTranslationMessages = { | ||||
|     }, | ||||
|     users: { | ||||
|       invalid_user_id: "Lokaler Anteil der Matrix Benutzer-ID ohne Homeserver.", | ||||
|       tabs: { sso: "SSO" }, | ||||
|       tabs: { sso: "SSO", experimental: "Experimentell", limits: "Rate Limits" }, | ||||
|     }, | ||||
|     rooms: { | ||||
|       details: "Raumdetails", | ||||
| @@ -190,7 +190,14 @@ const de: SynapseTranslationMessages = { | ||||
|         erase_avatar: "Avatar löschen", | ||||
|         delete_media: "Alle von dem/den Benutzer(n) hochgeladenen Medien löschen", | ||||
|         redact_events: "Schwärzen aller vom Benutzer gesendeten Ereignisse (-s)", | ||||
|         generate_password: "Passwort generieren", | ||||
|       }, | ||||
|       limits: { | ||||
|         messages_per_second: "Nachrichten pro Sekunde", | ||||
|         messages_per_second_text: "Die Anzahl der Aktionen, die in einer Sekunde durchgeführt werden können.", | ||||
|         burst_count: "Burst-Anzahl", | ||||
|         burst_count_text: "Die Anzahl der Aktionen, die vor der Begrenzung durchgeführt werden können.", | ||||
|       } | ||||
|     }, | ||||
|     rooms: { | ||||
|       name: "Raum |||| Räume", | ||||
|   | ||||
| @@ -25,7 +25,11 @@ const en: SynapseTranslationMessages = { | ||||
|     }, | ||||
|     users: { | ||||
|       invalid_user_id: "Localpart of a Matrix user-id without homeserver.", | ||||
|       tabs: { sso: "SSO" }, | ||||
|       tabs: { | ||||
|         sso: "SSO", | ||||
|         experimental: "Experimental", | ||||
|         limits: "Rate Limits", | ||||
|       }, | ||||
|     }, | ||||
|     rooms: { | ||||
|       details: "Room details", | ||||
| @@ -159,7 +163,14 @@ const en: SynapseTranslationMessages = { | ||||
|         erase_avatar: "Erase avatar", | ||||
|         delete_media: "Delete all media uploaded by the user(-s)", | ||||
|         redact_events: "Redact all events sent by the user(-s)", | ||||
|         generate_password: "Generate password", | ||||
|       }, | ||||
|       limits: { | ||||
|         messages_per_second: "Messages per second", | ||||
|         messages_per_second_text: "The number of actions that can be performed in a second.", | ||||
|         burst_count: "Burst count", | ||||
|         burst_count_text: "How many actions that can be performed before being limited.", | ||||
|       } | ||||
|     }, | ||||
|     rooms: { | ||||
|       name: "Room |||| Rooms", | ||||
|   | ||||
| @@ -24,7 +24,7 @@ const fa: SynapseTranslationMessages = { | ||||
|     }, | ||||
|     users: { | ||||
|       invalid_user_id: "بخش محلی یک شناسه کاربری ماتریکس بدون سرور خانگی.", | ||||
|       tabs: { sso: "SSO" }, | ||||
|       tabs: { sso: "SSO", experimental: "تجربی", limits: "محدودیت ها" }, | ||||
|     }, | ||||
|     rooms: { | ||||
|       tabs: { | ||||
| @@ -155,7 +155,14 @@ const fa: SynapseTranslationMessages = { | ||||
|         erase_avatar: "محو الصورة الرمزية", | ||||
|         delete_media: "حذف جميع الوسائط التي تم تحميلها بواسطة المستخدم (المستخدمين)", | ||||
|         redact_events: "تنقيح جميع الأحداث المرسلة من قبل المستخدم (-s)", | ||||
|         generate_password: "توليد رمز عبور", | ||||
|       }, | ||||
|       limits: { | ||||
|         messages_per_second: "پیام در ثانیه", | ||||
|         messages_per_second_text: "تعداد عملیاتی که می تواند در یک ثانیه انجام شود.", | ||||
|         burst_count: "تعداد پیچیدگی", | ||||
|         burst_count_text: "تعداد عملیاتی که می تواند قبل از محدودیت انجام شود.", | ||||
|       } | ||||
|     }, | ||||
|     rooms: { | ||||
|       name: "اتاق |||| اتاق ها", | ||||
| @@ -207,6 +214,11 @@ const fa: SynapseTranslationMessages = { | ||||
|           title: "حذف اتاق", | ||||
|           content: | ||||
|             "آیا مطمئن هستید که می خواهید اتاق را حذف کنید؟ این قابل بازگشت نیست. همه پیام ها و رسانه های مشترک در اتاق از سرور حذف می شوند!", | ||||
|           fields: { | ||||
|             block: "حذف", | ||||
|           }, | ||||
|           success: "اتاق با موفقیت حذف شد.", | ||||
|           failure: "خطایی رخ داده است.", | ||||
|         }, | ||||
|       }, | ||||
|     }, | ||||
|   | ||||
| @@ -24,7 +24,7 @@ const fr: SynapseTranslationMessages = { | ||||
|     }, | ||||
|     users: { | ||||
|       invalid_user_id: "Partie locale d'un identifiant utilisateur Matrix sans le nom du serveur d’accueil.", | ||||
|       tabs: { sso: "Authentification unique" }, | ||||
|       tabs: { sso: "Authentification unique", experimental: "Expérimental", limits: "Limites" }, | ||||
|     }, | ||||
|     rooms: { | ||||
|       tabs: { | ||||
| @@ -157,7 +157,14 @@ const fr: SynapseTranslationMessages = { | ||||
|         erase_avatar: "Effacer l'avatar", | ||||
|         delete_media: "Supprimer tous les médias téléchargés par le(s) utilisateur(s)", | ||||
|         redact_events: "Expurger tous les événements envoyés par l'utilisateur(-s)", | ||||
|         generate_password: "Générer un mot de passe", | ||||
|       }, | ||||
|       limits: { | ||||
|         messages_per_second: "Messages par seconde", | ||||
|         messages_per_second_text: "Le nombre d'actions que l'utilisateur peut effectuer par seconde.", | ||||
|         burst_count: "Compteur de pics", | ||||
|         burst_count_text: "Le nombre d'actions que l'utilisateur peut effectuer avant d'être limité.", | ||||
|       } | ||||
|     }, | ||||
|     rooms: { | ||||
|       name: "Salon |||| Salons", | ||||
|   | ||||
							
								
								
									
										9
									
								
								src/i18n/index.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								src/i18n/index.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -22,7 +22,7 @@ interface SynapseTranslationMessages extends TranslationMessages { | ||||
|     }; | ||||
|     users: { | ||||
|       invalid_user_id: string; | ||||
|       tabs: { sso: string }; | ||||
|       tabs: { sso: string; experimental: string; limits: string; }; | ||||
|     }; | ||||
|     rooms: { | ||||
|       details?: string; // TODO: fa, fr, it, zh | ||||
| @@ -155,6 +155,13 @@ interface SynapseTranslationMessages extends TranslationMessages { | ||||
|         erase_avatar: string; | ||||
|         delete_media: string; | ||||
|         redact_events: string; | ||||
|         generate_password: string; | ||||
|       }; | ||||
|       limits: { | ||||
|         messages_per_second: string; | ||||
|         messages_per_second_text: string; | ||||
|         burst_count: string; | ||||
|         burst_count_text: string; | ||||
|       }; | ||||
|     }; | ||||
|     rooms: { | ||||
|   | ||||
| @@ -24,7 +24,7 @@ const it: SynapseTranslationMessages = { | ||||
|     }, | ||||
|     users: { | ||||
|       invalid_user_id: "ID utente non valido su questo homeserver.", | ||||
|       tabs: { sso: "SSO" }, | ||||
|       tabs: { sso: "SSO", experimental: "Sperimentale", limits: "Limiti" }, | ||||
|     }, | ||||
|     rooms: { | ||||
|       tabs: { | ||||
| @@ -156,7 +156,14 @@ const it: SynapseTranslationMessages = { | ||||
|         erase_avatar: "Cancella l'avatar dell'utente", | ||||
|         delete_media: "Elimina tutti i media caricati dall'utente(-s)", | ||||
|         redact_events: "Ridurre tutti gli eventi inviati dall'utente(-s)", | ||||
|         generate_password: "Genera password", | ||||
|       }, | ||||
|       limits: { | ||||
|         messages_per_second: "Messaggi al secondo", | ||||
|         messages_per_second_text: "Il numero di azioni che l'utente può eseguire al secondo.", | ||||
|         burst_count: "Burst-conteggio", | ||||
|         burst_count_text: "Il numero di azioni che l'utente può eseguire prima di essere limitato.", | ||||
|       } | ||||
|     }, | ||||
|     rooms: { | ||||
|       name: "Stanza |||| Stanze", | ||||
|   | ||||
| @@ -50,7 +50,7 @@ const ru: SynapseTranslationMessages = { | ||||
|     }, | ||||
|     users: { | ||||
|       invalid_user_id: "Локальная часть ID пользователя Matrix без адреса домашнего сервера.", | ||||
|       tabs: { sso: "SSO" }, | ||||
|       tabs: { sso: "SSO", experimental: "Экспериментальные", limits: "Ограничения" }, | ||||
|     }, | ||||
|     rooms: { | ||||
|       details: "Данные комнаты", | ||||
| @@ -193,8 +193,15 @@ const ru: SynapseTranslationMessages = { | ||||
|         erase_avatar: "Удалить аватар", | ||||
|         delete_media: "Удаление всех медиафайлов, загруженных пользователем (-ами)", | ||||
|         redact_events: "Удаление всех событий, отправленных пользователем (-ами)", | ||||
|         generate_password: "Сгенерировать пароль", | ||||
|       }, | ||||
|       }, | ||||
|       limits: { | ||||
|         messages_per_second: "Сообщений в секунду", | ||||
|         messages_per_second_text: "Количество действий, которые могут быть выполнены в секунду.", | ||||
|         burst_count: "Burst-счётчик", | ||||
|         burst_count_text: "Количество действий, которые могут быть выполнены до ограничения.", | ||||
|       } | ||||
|     }, | ||||
|       rooms: { | ||||
|         name: "Комната |||| Комнаты", | ||||
|         fields: { | ||||
|   | ||||
| @@ -52,7 +52,7 @@ const zh: SynapseTranslationMessages = { | ||||
|     }, | ||||
|     users: { | ||||
|       invalid_user_id: "必须要是一个有效的 Matrix 用户 ID ,例如 @user_id:homeserver", | ||||
|       tabs: { sso: "SSO" }, | ||||
|       tabs: { sso: "SSO", experimental: "实验性", limits: "限制" }, | ||||
|     }, | ||||
|     rooms: { | ||||
|       tabs: { | ||||
| @@ -180,7 +180,14 @@ const zh: SynapseTranslationMessages = { | ||||
|         erase_avatar: "抹掉头像", | ||||
|         delete_media: "删除用户上传的所有媒体", | ||||
|         redact_events: "重新编辑用户(-s)发送的所有事件", | ||||
|         generate_password: "生成密码", | ||||
|       }, | ||||
|       limits: { | ||||
|         messages_per_second: "每秒消息数", | ||||
|         messages_per_second_text: "每秒可以执行的操作数。", | ||||
|         burst_count: "Burst-计数", | ||||
|         burst_count_text: "在限制之前可以执行的操作数。", | ||||
|       } | ||||
|     }, | ||||
|     rooms: { | ||||
|       name: "房间", | ||||
|   | ||||
| @@ -3,38 +3,42 @@ import React from "react"; | ||||
| import { createRoot } from "react-dom/client"; | ||||
|  | ||||
| import App from "./App"; | ||||
| import { AppContext, MenuItem } from "./AppContext"; | ||||
| import { Config, WellKnownKey, LoadConfig } from "./components/config"; | ||||
| import { AppContext } from "./AppContext"; | ||||
| import storage from "./storage"; | ||||
|  | ||||
| fetch("config.json") | ||||
|   .then(res => res.json()) | ||||
|   .then(props => { | ||||
|     if (props.asManagedUsers) { | ||||
|       storage.setItem("as_managed_users", JSON.stringify(props.asManagedUsers)); | ||||
|     } | ||||
| // load config.json | ||||
| let props: Config = {}; | ||||
| try { | ||||
|   const resp = await fetch("config.json"); | ||||
|   const configJSON = await resp.json(); | ||||
|   console.log("Loaded config.json", configJSON); | ||||
|   props = LoadConfig(configJSON as Config); | ||||
| } catch (e) { | ||||
|   console.error(e); | ||||
| } | ||||
|  | ||||
|     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)); | ||||
| // if home_server is set, try to load https://home_server/.well-known/matrix/client | ||||
| const homeserver = storage.getItem("home_server"); | ||||
| if (homeserver) { | ||||
|   try { | ||||
|     const resp = await fetch(`https://${homeserver}/.well-known/matrix/client`); | ||||
|     const configWK = await resp.json(); | ||||
|     if (!configWK[WellKnownKey]) { | ||||
|       console.log(`Loaded https://${homeserver}.well-known/matrix/client, but it doesn't contain ${WellKnownKey} key, skipping`, configWK); | ||||
|     } else { | ||||
|       console.log(`Loaded https://${homeserver}.well-known/matrix/client`, configWK); | ||||
|       props = LoadConfig(configWK[WellKnownKey] as Config); | ||||
|     } | ||||
|   } catch (e) { | ||||
|     console.log(`https://${homeserver}/.well-known/matrix/client not found, skipping`, e); | ||||
|   } | ||||
| } | ||||
|  | ||||
|     return createRoot(document.getElementById("root")).render( | ||||
| createRoot(document.getElementById("root")).render( | ||||
|       <React.StrictMode> | ||||
|         <AppContext.Provider value={props}> | ||||
|           <App /> | ||||
|         </AppContext.Provider> | ||||
|       </React.StrictMode> | ||||
|     ) | ||||
|   }); | ||||
| ); | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| import { useState, useEffect } from "react"; | ||||
|  | ||||
| import LockIcon from "@mui/icons-material/Lock"; | ||||
| import { Avatar, Box, Button, Card, CardActions, CircularProgress, MenuItem, Select, Tab, Tabs, Typography } from "@mui/material"; | ||||
| import { | ||||
|   Form, | ||||
| @@ -34,7 +33,7 @@ const LoginPage = () => { | ||||
|   const login = useLogin(); | ||||
|   const notify = useNotify(); | ||||
|   const { restrictBaseUrl } = useAppContext(); | ||||
|   const allowSingleBaseUrl = typeof restrictBaseUrl === "string"; | ||||
|   const allowSingleBaseUrl = typeof restrictBaseUrl === "string" && restrictBaseUrl !== ""; | ||||
|   const allowMultipleBaseUrls = | ||||
|     Array.isArray(restrictBaseUrl) && | ||||
|     restrictBaseUrl.length > 0 && | ||||
| @@ -249,9 +248,7 @@ const LoginPage = () => { | ||||
|             {loading ? ( | ||||
|               <CircularProgress size={25} thickness={2} /> | ||||
|             ) : ( | ||||
|               <Avatar className="icon"> | ||||
|                 <LockIcon /> | ||||
|               </Avatar> | ||||
|               <Avatar sx={{ width: "120px", height: "120px" }} src="./images/logo.webp"/> | ||||
|             )} | ||||
|           </Box> | ||||
|           <Box className="hint">{translate("synapseadmin.auth.welcome")}</Box> | ||||
|   | ||||
| @@ -187,7 +187,7 @@ export const RoomShow = (props: ShowProps) => { | ||||
|             <Datagrid style={{ width: "100%" }} bulkActionButtons={false}> | ||||
|               <TextField source="type" sortable={false} /> | ||||
|               <DateField source="origin_server_ts" showTime options={DATE_FORMAT} sortable={false} /> | ||||
|               <TextField source="content" sortable={false} /> | ||||
|               <FunctionField source="content" sortable={false} render={record => `${JSON.stringify(record.content, null, 2)}`} /> | ||||
|               <ReferenceField source="sender" reference="users" sortable={false}> | ||||
|                 <TextField source="id" /> | ||||
|               </ReferenceField> | ||||
|   | ||||
| @@ -7,9 +7,11 @@ import NotificationsIcon from "@mui/icons-material/Notifications"; | ||||
| import PermMediaIcon from "@mui/icons-material/PermMedia"; | ||||
| import PersonPinIcon from "@mui/icons-material/PersonPin"; | ||||
| import SettingsInputComponentIcon from "@mui/icons-material/SettingsInputComponent"; | ||||
| import ScienceIcon from "@mui/icons-material/Science"; | ||||
| import LockClockIcon from '@mui/icons-material/LockClock'; | ||||
| import ViewListIcon from "@mui/icons-material/ViewList"; | ||||
| import { useEffect, useState } from "react"; | ||||
| import { Alert, ownerDocument } from "@mui/material"; | ||||
| import { Alert } from "@mui/material"; | ||||
| import { | ||||
|   ArrayInput, | ||||
|   ArrayField, | ||||
| @@ -54,10 +56,10 @@ import { | ||||
|   useNotify, | ||||
|   Identifier, | ||||
|   ToolbarClasses, | ||||
|   RaRecord, | ||||
|   ImageInput, | ||||
|   ImageField, | ||||
|   FunctionField, | ||||
|   useDataProvider, | ||||
| } from "react-admin"; | ||||
| import { Link } from "react-router-dom"; | ||||
|  | ||||
| @@ -68,6 +70,10 @@ import { ServerNoticeButton, ServerNoticeBulkButton } from "../components/Server | ||||
| import { DATE_FORMAT } from "../components/date"; | ||||
| import { DeviceRemoveButton } from "../components/devices"; | ||||
| import { MediaIDField, ProtectMediaButton, QuarantineMediaButton } from "../components/media"; | ||||
| import { generateRandomPassword } from "../synapse/synapse"; | ||||
| import { useFormContext } from "react-hook-form"; | ||||
| import { ExperimentalFeaturesList } from "../components/ExperimentalFeatures"; | ||||
| import { UserRateLimits } from "../components/UserRateLimits"; | ||||
|  | ||||
| const choices_medium = [ | ||||
|   { id: "email", name: "resources.users.email" }, | ||||
| @@ -126,8 +132,6 @@ const UserBulkActionButtons = () => { | ||||
|   const [asManagedUserIsSelected, setAsManagedUserIsSelected] = useState(false); | ||||
|   const selectedIds = record.selectedIds; | ||||
|   const ownUserId = localStorage.getItem("user_id"); | ||||
|   const notify = useNotify(); | ||||
|   const translate = useTranslate(); | ||||
|  | ||||
|   useEffect(() => { | ||||
|     setOwnUserIsSelected(selectedIds.includes(ownUserId)); | ||||
| @@ -238,11 +242,11 @@ export const UserCreate = (props: CreateProps) => ( | ||||
|  | ||||
| const UserTitle = () => { | ||||
|   const record = useRecordContext(); | ||||
|   const translate = useTranslate(); | ||||
|   if (!record) { | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   const translate = useTranslate(); | ||||
|   let username = record ? (record.displayname ? `"${record.displayname}"` : `"${record.name}"`) : "" | ||||
|   if (isASManaged(record?.id)) { | ||||
|     username += " 🤖"; | ||||
| @@ -301,12 +305,33 @@ const UserBooleanInput = props => { | ||||
| const UserPasswordInput = props => { | ||||
|   const record = useRecordContext(); | ||||
|   let asManagedUserIsSelected = false; | ||||
|  | ||||
|   // Get form context to update field value | ||||
|   const form = useFormContext(); | ||||
|   if (record) { | ||||
|     asManagedUserIsSelected = isASManaged(record.id); | ||||
|   } | ||||
|  | ||||
|   const generatePassword = () => { | ||||
|     const password = generateRandomPassword(); | ||||
|     if (record) { | ||||
|       form.setValue("password", password, { shouldDirty: true }); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   return ( | ||||
|       <PasswordInput {...props} helperText="resources.users.helper.modify_managed_user_error" disabled={asManagedUserIsSelected} /> | ||||
|     <> | ||||
|       <PasswordInput {...props} helperText="resources.users.helper.modify_managed_user_error" | ||||
|         {...(asManagedUserIsSelected ? { disabled: true } : {})} | ||||
|        /> | ||||
|        <Button | ||||
|         variant="outlined" | ||||
|         label="Generate Password" | ||||
|         onClick={generatePassword} | ||||
|         sx={{ marginBottom: "10px" }} | ||||
|         disabled={asManagedUserIsSelected} | ||||
|       /> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| @@ -314,7 +339,11 @@ export const UserEdit = (props: EditProps) => { | ||||
|   const translate = useTranslate(); | ||||
|  | ||||
|   return ( | ||||
|     <Edit {...props} title={<UserTitle />} actions={<UserEditActions />} mutationMode="pessimistic"> | ||||
|     <Edit {...props} title={<UserTitle />} actions={<UserEditActions />} mutationMode="pessimistic" queryOptions={{ | ||||
|       meta: { | ||||
|         include: ["features"] // Tell your dataProvider to include features | ||||
|       } | ||||
|     }}> | ||||
|       <TabbedForm toolbar={<UserEditToolbar />}> | ||||
|         <FormTab label={translate("resources.users.name", { smart_count: 1 })} icon={<PersonPinIcon />}> | ||||
|           <AvatarField source="avatar_src" sx={{ height: "120px", width: "120px" }} /> | ||||
| @@ -448,6 +477,14 @@ export const UserEdit = (props: EditProps) => { | ||||
|             </Datagrid> | ||||
|           </ReferenceManyField> | ||||
|         </FormTab> | ||||
|  | ||||
|         <FormTab label="synapseadmin.users.tabs.experimental" icon={<ScienceIcon />} path="experimental"> | ||||
|           <ExperimentalFeaturesList /> | ||||
|         </FormTab> | ||||
|  | ||||
|         <FormTab label="synapseadmin.users.tabs.limits" icon={<LockClockIcon />} path="limits"> | ||||
|           <UserRateLimits /> | ||||
|         </FormTab> | ||||
|       </TabbedForm> | ||||
|     </Edit> | ||||
|   ); | ||||
|   | ||||
| @@ -3,6 +3,7 @@ import { AuthProvider, HttpError, Options, fetchUtils } from "react-admin"; | ||||
| import storage from "../storage"; | ||||
| import { MatrixError, displayError } from "../components/error"; | ||||
| import { fetchAuthenticatedMedia } from "../utils/fetchMedia"; | ||||
| import { ClearConfig } from "../components/config"; | ||||
|  | ||||
| const authProvider: AuthProvider = { | ||||
|   // called when the user attempts to log in | ||||
| @@ -75,7 +76,7 @@ const authProvider: AuthProvider = { | ||||
|  | ||||
|       response = await fetchUtils.fetchJson(login_api_url, options); | ||||
|       const json = response.json; | ||||
|       storage.setItem("home_server", accessToken ? base_url : json.home_server); | ||||
|       storage.setItem("home_server", accessToken ? json.user_id.split(":")[1] : 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); | ||||
| @@ -154,8 +155,7 @@ const authProvider: AuthProvider = { | ||||
|       } catch (err) { | ||||
|         console.log("Error logging out", err); | ||||
|       } finally { | ||||
|         storage.removeItem("access_token"); | ||||
|         storage.removeItem("login_type"); | ||||
|         ClearConfig(); | ||||
|       } | ||||
|     } | ||||
|   }, | ||||
|   | ||||
| @@ -248,9 +248,23 @@ export interface UploadMediaResult { | ||||
|   content_uri: string; | ||||
| } | ||||
|  | ||||
| export interface ExperimentalFeaturesModel { | ||||
|   features: { | ||||
|     [key: string]: boolean; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface RateLimitsModel { | ||||
|   messages_per_second?: number; | ||||
|   burst_count?: number; | ||||
| } | ||||
|  | ||||
| export interface SynapseDataProvider extends DataProvider { | ||||
|   deleteMedia: (params: DeleteMediaParams) => Promise<DeleteMediaResult>; | ||||
|   uploadMedia: (params: UploadMediaParams) => Promise<UploadMediaResult>; | ||||
|   updateFeatures: (id: Identifier, features: ExperimentalFeaturesModel) => Promise<void>; | ||||
|   getRateLimits: (id: Identifier) => Promise<RateLimitsModel>; | ||||
|   setRateLimits: (id: Identifier, rateLimits: RateLimitsModel) => Promise<void>; | ||||
| } | ||||
|  | ||||
| const resourceMap = { | ||||
| @@ -575,13 +589,28 @@ const baseDataProvider: SynapseDataProvider = { | ||||
|  | ||||
|   getMany: async (resource, params) => { | ||||
|     console.log("getMany " + resource); | ||||
|     const homeserver = storage.getItem("base_url"); | ||||
|     if (!homeserver || !(resource in resourceMap)) throw Error("Homerserver not set"); | ||||
|     const base_url = storage.getItem("base_url"); | ||||
|     const homeserver = storage.getItem("home_server"); | ||||
|     if (!base_url || !(resource in resourceMap)) throw Error("base_url not set"); | ||||
|  | ||||
|     const res = resourceMap[resource]; | ||||
|  | ||||
|     const endpoint_url = homeserver + res.path; | ||||
|     const responses = await Promise.all(params.ids.map(id => jsonClient(`${endpoint_url}/${encodeURIComponent(id)}`))); | ||||
|     const endpoint_url = base_url + res.path; | ||||
|     const responses = await Promise.all(params.ids.map(id => { | ||||
|       // edge case: when user is external / federated, homeserver will return error, as querying external users via | ||||
|       // /_synapse/admin/v2/users is not allowed. | ||||
|       // That leads to an issue when a user is referenced (e.g., in room state datagrid) - the user cell is just empty. | ||||
|       // To avoid that, we fake the response with one specific field (name) which is used in the datagrid. | ||||
|       if (homeserver && resource === "users") { | ||||
|         if (!(<string>id).endsWith(homeserver)) { | ||||
|           const json = { | ||||
|               name: id, | ||||
|           }; | ||||
|           return Promise.resolve({ json }); | ||||
|         } | ||||
|       } | ||||
|       return jsonClient(`${endpoint_url}/${encodeURIComponent(id)}`); | ||||
|     })); | ||||
|     return { | ||||
|       data: responses.map(({ json }) => res.map(json)), | ||||
|       total: responses.length, | ||||
| @@ -783,6 +812,40 @@ const baseDataProvider: SynapseDataProvider = { | ||||
|     }); | ||||
|     return json as UploadMediaResult; | ||||
|   }, | ||||
|   getFeatures: async (id: Identifier) => { | ||||
|     const base_url = storage.getItem("base_url"); | ||||
|     const endpoint_url = `${base_url}/_synapse/admin/v1/experimental_features/${encodeURIComponent(returnMXID(id))}`; | ||||
|     const { json } = await jsonClient(endpoint_url); | ||||
|     return json.features as ExperimentalFeaturesModel; | ||||
|   }, | ||||
|   updateFeatures: async (id: Identifier, features: ExperimentalFeaturesModel) => { | ||||
|     const base_url = storage.getItem("base_url"); | ||||
|     const endpoint_url = `${base_url}/_synapse/admin/v1/experimental_features/${encodeURIComponent(returnMXID(id))}`; | ||||
|     await jsonClient(endpoint_url, { method: "PUT", body: JSON.stringify({ features }) }); | ||||
|   }, | ||||
|   getRateLimits: async (id: Identifier) => { | ||||
|     const base_url = storage.getItem("base_url"); | ||||
|     const endpoint_url = `${base_url}/_synapse/admin/v1/users/${encodeURIComponent(returnMXID(id))}/override_ratelimit`; | ||||
|     const { json } = await jsonClient(endpoint_url); | ||||
|     return json as RateLimitsModel; | ||||
|   }, | ||||
|   setRateLimits: async (id: Identifier, rateLimits: RateLimitsModel) => { | ||||
|     const filtered = Object.entries(rateLimits). | ||||
|       filter(([key, value]) => value !== null && value !== undefined). | ||||
|       reduce((obj, [key, value]) => { | ||||
|         obj[key] = value; | ||||
|         return obj; | ||||
|       }, {}); | ||||
|  | ||||
|     const base_url = storage.getItem("base_url"); | ||||
|     const endpoint_url = `${base_url}/_synapse/admin/v1/users/${encodeURIComponent(returnMXID(id))}/override_ratelimit`; | ||||
|     if (Object.keys(filtered).length === 0) { | ||||
|       await jsonClient(endpoint_url, { method: "DELETE" }); | ||||
|       return | ||||
|     } | ||||
|  | ||||
|     await jsonClient(endpoint_url, { method: "POST", body: JSON.stringify(filtered) }); | ||||
|   }, | ||||
| }; | ||||
|  | ||||
| const dataProvider = withLifecycleCallbacks(baseDataProvider, [ | ||||
| @@ -791,6 +854,12 @@ const dataProvider = withLifecycleCallbacks(baseDataProvider, [ | ||||
|     beforeUpdate: async (params: UpdateParams<any>, dataProvider: DataProvider) => { | ||||
|       const avatarFile = params.data.avatar_file?.rawFile; | ||||
|       const avatarErase = params.data.avatar_erase; | ||||
|       const rates = params.data.rates; | ||||
|  | ||||
|       if (rates) { | ||||
|         await dataProvider.setRateLimits(params.id, rates); | ||||
|         delete params.data.rates; | ||||
|       } | ||||
|  | ||||
|       if (avatarErase) { | ||||
|         params.data.avatar_url = ""; | ||||
|   | ||||
| @@ -91,8 +91,8 @@ export function returnMXID(input: string | Identifier): string { | ||||
|  * Generate a random user password | ||||
|  * @returns a new random password as string | ||||
|  */ | ||||
| export function generateRandomPassword(length = 20): string { | ||||
|   const characters = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz~!@-#$"; | ||||
| export function generateRandomPassword(length = 64): string { | ||||
|   const characters = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz~`!@#$%^&*()_-+={[}]|:;'.?/<>,"; | ||||
|   return Array.from(crypto.getRandomValues(new Uint32Array(length))) | ||||
|     .map(x => characters[x % characters.length]) | ||||
|     .join(""); | ||||
|   | ||||
| @@ -5,13 +5,41 @@ import { defineConfig } from "vite"; | ||||
|  | ||||
| export default defineConfig({ | ||||
|   base: "./", | ||||
|   build: { | ||||
|     target: "esnext", | ||||
|   }, | ||||
|   plugins: [ | ||||
|     react(), | ||||
|     vitePluginVersionMark({ | ||||
|       command: "git describe --tags || git rev-parse --short HEAD", | ||||
|       ifMeta: true, | ||||
|       ifLog: true, | ||||
|       name: "Synapse Admin", | ||||
|       command: 'git describe --tags || git rev-parse --short HEAD || echo "${SYNAPSE_ADMIN_VERSION:-unknown}"', | ||||
|       ifMeta: false, | ||||
|       ifLog: false, | ||||
|       ifGlobal: true, | ||||
|       outputFile: (version) => ({ | ||||
|         path: "manifest.json", | ||||
|         content: JSON.stringify({ | ||||
|           name: "Synapse Admin", | ||||
|           version: version, | ||||
|           description: "Synapse Admin is an admin console for synapse Matrix homeserver with additional features.", | ||||
|           categories: ["productivity", "utilities"], | ||||
|           orientation: "landscape", | ||||
|           icons: [{ | ||||
|             src: "favicon.ico", | ||||
|             sizes: "32x32", | ||||
|             type: "image/x-icon" | ||||
|           },{ | ||||
|             src: "images/logo.webp", | ||||
|             sizes: "512x512", | ||||
|             type: "image/webp", | ||||
|             purpose: "any maskable" | ||||
|           }], | ||||
|           start_url: ".", | ||||
|           display: "standalone", | ||||
|           theme_color: "#000000", | ||||
|           background_color: "#ffffff" | ||||
|         }), | ||||
|       }), | ||||
|     }), | ||||
|   ], | ||||
| }); | ||||
|   | ||||
		Reference in New Issue
	
	Block a user