Compare commits
	
		
			66 Commits
		
	
	
		
			master
			...
			v0.10.3-et
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 02bee92806 | ||
|   | b7f8e03894 | ||
|   | 59bb4b4183 | ||
|   | bb53d53692 | ||
|   | 52a2f1c936 | ||
|   | e328380c77 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a277ded227 | ||
|   | 48d933e028 | ||
|   | 24cf0a60bf | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d5113aad72 | ||
|   | 6957cb1f7d | ||
|   | 66c706532a | ||
|   | 332e98a095 | ||
|   | f639a1c9ff | ||
|   | 8282bb0926 | ||
|   | 264d2ff59d | ||
|   | 90a6c7d8c2 | ||
|   | 7de9166648 | ||
|   | 0bf3440fc8 | ||
|   | cceae77529 | ||
|   | 1474b46ff0 | ||
|   | 01e3947b22 | ||
|   | e093bd8625 | ||
|   | 390aab5ce7 | ||
|   | fb1a04971b | ||
|   | eff17a0929 | ||
|   | 8eaaaa50ec | ||
|   | 85b305d117 | ||
|   | 4cbea6ffb6 | ||
|   | 1967546ae4 | ||
|   | c9a3294852 | ||
|   | f1839387e2 | ||
|   | c8246a1d19 | ||
|   | 3c6259cc88 | ||
|   | 317df5af0f | ||
|   | 2142770a5b | ||
|   | 51297b49fc | ||
|   | 311cc2a1f4 | ||
|   | 6bc760a6fa | ||
|   | 50c96cfd77 | ||
|   | 7747dc7f28 | ||
|   | 678867e981 | ||
|   | 4a4fae104e | ||
|   | 265b5157af | ||
|   | b5b593945d | ||
|   | 15c8a30c92 | ||
|   | 40c6eb0b95 | ||
|   | 4c4e3a07f6 | ||
|   | 35322fa01d | ||
|   | 88e88fb6b9 | ||
|   | 7016e5a349 | ||
|   | c7e275d4ec | ||
|   | b1d3340fce | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2ae5930578 | ||
|   | 9dd6940383 | ||
|   | e2194f5c49 | ||
|   | e6dcbcb052 | ||
|   | 5acf3042c3 | ||
|   | 7286abaaae | ||
|   | 0d5e95fa7c | ||
|   | c4b54c40fb | ||
|   | ec0b980e06 | ||
|   | 94ccd3ad36 | ||
|   | 49d67f9130 | ||
|   | 2bb846734e | ||
|   | 056d9c6b4c | 
| @@ -1,13 +0,0 @@ | |||||||
| # Exclude a bunch of stuff which can make the build context a larger than it needs to be |  | ||||||
| tests/ |  | ||||||
| build/ |  | ||||||
| dist/ |  | ||||||
| lib/ |  | ||||||
| node_modules/ |  | ||||||
| electron_app/ |  | ||||||
| karma-reports/ |  | ||||||
| .pnp.cjs |  | ||||||
| .pnp.loader.mjs |  | ||||||
| .idea/ |  | ||||||
| .tmp/ |  | ||||||
| config.json* |  | ||||||
							
								
								
									
										1
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | liberapay: etkecc | ||||||
							
								
								
									
										20
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										20
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,20 +0,0 @@ | |||||||
| version: 2 |  | ||||||
| updates: |  | ||||||
|   - package-ecosystem: "npm" |  | ||||||
|     directory: "/" |  | ||||||
|     schedule: |  | ||||||
|       interval: "weekly" |  | ||||||
|     ignore: |  | ||||||
|       # Major updates for react-admin have breaking changes |  | ||||||
|       - dependency-name: "react-admin" |  | ||||||
|         update-types: ["version-update:semver-major"] |  | ||||||
|  |  | ||||||
|   - package-ecosystem: "docker" |  | ||||||
|     directory: "/" |  | ||||||
|     schedule: |  | ||||||
|       interval: "weekly" |  | ||||||
|  |  | ||||||
|   - package-ecosystem: "github-actions" |  | ||||||
|     directory: "/" |  | ||||||
|     schedule: |  | ||||||
|       interval: "weekly" |  | ||||||
							
								
								
									
										23
									
								
								.github/workflows/build-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										23
									
								
								.github/workflows/build-test.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,23 +0,0 @@ | |||||||
| name: build-test |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   push: |  | ||||||
|     branches: ["master"] |  | ||||||
|   pull_request: |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   check: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|       - name: Checkout |  | ||||||
|         uses: actions/checkout@v4 |  | ||||||
|       - name: Setup node |  | ||||||
|         uses: actions/setup-node@v4 |  | ||||||
|         with: |  | ||||||
|           node-version: "18" |  | ||||||
|       - name: Install dependencies |  | ||||||
|         run: yarn --immutable |  | ||||||
|       - name: Run checks |  | ||||||
|         run: yarn lint |  | ||||||
|       - name: Run tests |  | ||||||
|         run: yarn test |  | ||||||
							
								
								
									
										63
									
								
								.github/workflows/docker-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										63
									
								
								.github/workflows/docker-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,63 +0,0 @@ | |||||||
| name: Create docker image(s) and push to docker hub and ghcr.io |  | ||||||
| # see https://docs.github.com/en/actions/publishing-packages/publishing-docker-images#publishing-images-to-docker-hub-and-github-packages |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   push: |  | ||||||
|     # Sequence of patterns matched against refs/heads |  | ||||||
|     # prettier-ignore |  | ||||||
|     branches: |  | ||||||
|       # Push events on master branch |  | ||||||
|       - master |  | ||||||
|     # Sequence of patterns matched against refs/tags |  | ||||||
|     tags: |  | ||||||
|       - '[0-9]+\.[0-9]+\.[0-9]+'             # Push events to 0.X.X tag |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   docker: |  | ||||||
|     name: Push Docker image to multiple registries |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     permissions: |  | ||||||
|       packages: write |  | ||||||
|       contents: read |  | ||||||
|  |  | ||||||
|     steps: |  | ||||||
|       - name: Checkout |  | ||||||
|         uses: actions/checkout@v4 |  | ||||||
|         with: |  | ||||||
|           fetch-depth: 0 |  | ||||||
|  |  | ||||||
|       - name: Set up QEMU |  | ||||||
|         uses: docker/setup-qemu-action@v3 |  | ||||||
|  |  | ||||||
|       - name: Set up Docker Buildx |  | ||||||
|         uses: docker/setup-buildx-action@v3 |  | ||||||
|  |  | ||||||
|       - name: Login to DockerHub |  | ||||||
|         uses: docker/login-action@v3 |  | ||||||
|         with: |  | ||||||
|           username: ${{ secrets.DOCKERHUB_USERNAME }} |  | ||||||
|           password: ${{ secrets.DOCKERHUB_TOKEN }} |  | ||||||
|  |  | ||||||
|       - name: Login to GHCR |  | ||||||
|         uses: docker/login-action@v3 |  | ||||||
|         with: |  | ||||||
|           registry: ghcr.io |  | ||||||
|           username: ${{ github.repository_owner }} |  | ||||||
|           password: ${{ secrets.GITHUB_TOKEN }} |  | ||||||
|  |  | ||||||
|       - name: Extract metadata (tags, labels) for Docker |  | ||||||
|         id: meta |  | ||||||
|         uses: docker/metadata-action@v5 |  | ||||||
|         with: |  | ||||||
|           images: | |  | ||||||
|             awesometechnologies/synapse-admin |  | ||||||
|             ghcr.io/${{ github.repository }} |  | ||||||
|  |  | ||||||
|       - name: Build and Push Tag |  | ||||||
|         uses: docker/build-push-action@v6 |  | ||||||
|         with: |  | ||||||
|           context: . |  | ||||||
|           push: true |  | ||||||
|           tags: ${{ steps.meta.outputs.tags }} |  | ||||||
|           labels: ${{ steps.meta.outputs.labels }} |  | ||||||
|           platforms: linux/amd64,linux/arm64 |  | ||||||
							
								
								
									
										29
									
								
								.github/workflows/edge_ghpage.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										29
									
								
								.github/workflows/edge_ghpage.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,29 +0,0 @@ | |||||||
| name: Build and Deploy Edge version to GH Pages |  | ||||||
| on: |  | ||||||
|   workflow_dispatch: |  | ||||||
|   push: |  | ||||||
|     branches: |  | ||||||
|       - main |  | ||||||
|       - master |  | ||||||
| jobs: |  | ||||||
|   build-and-deploy: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|       - name: Checkout 🛎️ |  | ||||||
|         uses: actions/checkout@v4 |  | ||||||
|         with: |  | ||||||
|           fetch-depth: 100 |  | ||||||
|           fetch-tags: true |  | ||||||
|       - uses: actions/setup-node@v4 |  | ||||||
|         with: |  | ||||||
|           node-version: "20" |  | ||||||
|       - name: Install and Build 🔧 |  | ||||||
|         run: | |  | ||||||
|           yarn install --immutable |  | ||||||
|           yarn build --base=/synapse-admin |  | ||||||
|  |  | ||||||
|       - name: Deploy 🚀 |  | ||||||
|         uses: JamesIves/github-pages-deploy-action@v4.6.3 |  | ||||||
|         with: |  | ||||||
|           branch: gh-pages |  | ||||||
|           folder: dist |  | ||||||
							
								
								
									
										30
									
								
								.github/workflows/github-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										30
									
								
								.github/workflows/github-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,30 +0,0 @@ | |||||||
| name: Create release tarball and attach to tag |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   push: |  | ||||||
|     tags: |  | ||||||
|       - "*" |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   build: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     permissions: |  | ||||||
|       contents: write |  | ||||||
|       packages: write |  | ||||||
|  |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - uses: actions/setup-node@v4 |  | ||||||
|         with: |  | ||||||
|           node-version: "20" |  | ||||||
|       - run: yarn install --immutable |  | ||||||
|       - run: yarn build |  | ||||||
|       - run: | |  | ||||||
|           version=`git describe --dirty --tags || echo unknown` |  | ||||||
|           cp -r dist synapse-admin-$version |  | ||||||
|           tar chvzf dist/synapse-admin-$version.tar.gz synapse-admin-$version |  | ||||||
|       - uses: softprops/action-gh-release@c062e08bd532815e2082a85e87e3ef29c3e6d191 |  | ||||||
|         with: |  | ||||||
|           files: dist/*.tar.gz |  | ||||||
|         env: |  | ||||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} |  | ||||||
							
								
								
									
										119
									
								
								.github/workflows/workflow.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										119
									
								
								.github/workflows/workflow.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,119 @@ | |||||||
|  | name: CI | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: [ "main" ] | ||||||
|  |     tags: [ "v*" ] | ||||||
|  | env: | ||||||
|  |   bunny_version: v0.1.0 | ||||||
|  |   base_path: ./ | ||||||
|  | permissions: | ||||||
|  |   checks: write | ||||||
|  |   contents: write | ||||||
|  |   packages: write | ||||||
|  |   pull-requests: read | ||||||
|  | jobs: | ||||||
|  |   build: | ||||||
|  |     name: Build | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@v4 | ||||||
|  |         with: | ||||||
|  |           fetch-depth: 0 | ||||||
|  |       - uses: actions/setup-node@v4 | ||||||
|  |         with: | ||||||
|  |           node-version: lts/* | ||||||
|  |           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 | ||||||
|  |         with: | ||||||
|  |           path: dist/ | ||||||
|  |           name: dist | ||||||
|  |           if-no-files-found: error | ||||||
|  |           retention-days: 1 | ||||||
|  |           compression-level: 0 | ||||||
|  |           overwrite: true | ||||||
|  |           include-hidden-files: true | ||||||
|  |  | ||||||
|  |   docker: | ||||||
|  |     name: Docker | ||||||
|  |     needs: build | ||||||
|  |     runs-on: self-hosted | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@v4 | ||||||
|  |       - uses: actions/download-artifact@v4 | ||||||
|  |         with: | ||||||
|  |           name: dist | ||||||
|  |           path: dist/ | ||||||
|  |       - name: Set up Docker Buildx | ||||||
|  |         uses: docker/setup-buildx-action@v1 | ||||||
|  |       - name: Login to ghcr.io | ||||||
|  |         uses: docker/login-action@v3 | ||||||
|  |         with: | ||||||
|  |           registry: ghcr.io | ||||||
|  |           username: ${{ github.actor }} | ||||||
|  |           password: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |       - name: Extract metadata (tags, labels) for Docker | ||||||
|  |         id: meta | ||||||
|  |         uses: docker/metadata-action@v5 | ||||||
|  |         with: | ||||||
|  |           images: | | ||||||
|  |             ghcr.io/${{ github.repository }} | ||||||
|  |             registry.etke.cc/${{ github.repository }} | ||||||
|  |           tags: | | ||||||
|  |             type=raw,value=latest,enable=${{ github.ref_name == 'main' }} | ||||||
|  |             type=semver,pattern={{raw}} | ||||||
|  |       - name: Build and push | ||||||
|  |         uses: docker/build-push-action@v6 | ||||||
|  |         with: | ||||||
|  |           platforms: linux/amd64,linux/arm64 | ||||||
|  |           context: . | ||||||
|  |           push: true | ||||||
|  |           tags: ${{ steps.meta.outputs.tags }} | ||||||
|  |           labels: ${{ steps.meta.outputs.labels }} | ||||||
|  |  | ||||||
|  |   cdn: | ||||||
|  |     name: CDN | ||||||
|  |     needs: build | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@v4 | ||||||
|  |       - uses: actions/download-artifact@v4 | ||||||
|  |         with: | ||||||
|  |           name: dist | ||||||
|  |           path: dist/ | ||||||
|  |       - name: Upload | ||||||
|  |         run: | | ||||||
|  |           wget -O bunny-upload.tar.gz https://github.com/etkecc/bunny-upload/releases/download/${{ env.bunny_version }}/bunny-upload_Linux_x86_64.tar.gz | ||||||
|  |           tar -xzf bunny-upload.tar.gz | ||||||
|  |           echo "${{ secrets.BUNNY_CONFIG }}" > bunny-config.yaml | ||||||
|  |           ./bunny-upload -c bunny-config.yaml | ||||||
|  |  | ||||||
|  |   github-release: | ||||||
|  |     name: Github Release | ||||||
|  |     needs: build | ||||||
|  |     if: ${{ startsWith(github.ref, 'refs/tags/') }} | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@v4 | ||||||
|  |       - uses: actions/download-artifact@v4 | ||||||
|  |         with: | ||||||
|  |           name: dist | ||||||
|  |           path: dist/ | ||||||
|  |       - name: Prepare release | ||||||
|  |         run: | | ||||||
|  |           mv dist synapse-admin | ||||||
|  |           tar chvzf synapse-admin.tar.gz synapse-admin | ||||||
|  |       - uses: softprops/action-gh-release@v2 | ||||||
|  |         with: | ||||||
|  |           files: synapse-admin.tar.gz | ||||||
|  |           generate_release_notes: true | ||||||
|  |           make_latest: "true" | ||||||
|  |           draft: false | ||||||
|  |           prerelease: false | ||||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -191,3 +191,6 @@ sketch | |||||||
| # .pnp.* | # .pnp.* | ||||||
|  |  | ||||||
| # End of https://www.toptal.com/developers/gitignore/api/node,yarn,react,visualstudiocode | # End of https://www.toptal.com/developers/gitignore/api/node,yarn,react,visualstudiocode | ||||||
|  |  | ||||||
|  | /testdata/synapse.data | ||||||
|  | /testdata/postgres.data | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @@ -5,6 +5,6 @@ | |||||||
|   }, |   }, | ||||||
|   "eslint.nodePath": ".yarn/sdks", |   "eslint.nodePath": ".yarn/sdks", | ||||||
|   "prettier.prettierPath": ".yarn/sdks/prettier/index.cjs", |   "prettier.prettierPath": ".yarn/sdks/prettier/index.cjs", | ||||||
|   "typescript.tsdk": ".yarn/sdks/typescript/lib", |   "typescript.tsdk": "node_modules/typescript/lib", | ||||||
|   "typescript.enablePromptUseWorkspaceTsdk": true |   "typescript.enablePromptUseWorkspaceTsdk": true | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1 +0,0 @@ | |||||||
| yarnPath: .yarn/releases/yarn-4.1.1.cjs |  | ||||||
							
								
								
									
										27
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,26 +1,5 @@ | |||||||
| # Builder | FROM ghcr.io/static-web-server/static-web-server:2 | ||||||
| FROM node:lts as builder |  | ||||||
| LABEL org.opencontainers.image.url=https://github.com/Awesome-Technologies/synapse-admin org.opencontainers.image.source=https://github.com/Awesome-Technologies/synapse-admin |  | ||||||
| # Base path for synapse admin |  | ||||||
| ARG BASE_PATH=./ |  | ||||||
|  |  | ||||||
| WORKDIR /src | ENV SERVER_ROOT=/app | ||||||
|  |  | ||||||
| # Copy .yarn directory to the working directory (must be on a separate line!) | COPY ./dist /app | ||||||
| # Use https://docs.docker.com/engine/reference/builder/#copy---parents when available |  | ||||||
| COPY .yarn .yarn |  | ||||||
| COPY package.json .yarnrc.yml yarn.lock ./ |  | ||||||
|  |  | ||||||
| # Disable telemetry and install packages |  | ||||||
| RUN yarn config set enableTelemetry 0 && yarn install --immutable --network-timeout=300000 |  | ||||||
|  |  | ||||||
| COPY . /src |  | ||||||
| RUN yarn build --base=$BASE_PATH |  | ||||||
|  |  | ||||||
| # App |  | ||||||
| FROM nginx:stable-alpine |  | ||||||
|  |  | ||||||
| COPY --from=builder /src/dist /app |  | ||||||
|  |  | ||||||
| RUN rm -rf /usr/share/nginx/html \ |  | ||||||
|  && ln -s /app /usr/share/nginx/html |  | ||||||
|   | |||||||
							
								
								
									
										186
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										186
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,14 +1,137 @@ | |||||||
| [](https://github.com/Awesome-Technologies/synapse-admin/blob/master/LICENSE) | # Synapse Admin UI [](https://github.com/Awesome-Technologies/synapse-admin/blob/master/LICENSE) | ||||||
| [](https://app.travis-ci.com/github/Awesome-Technologies/synapse-admin) |  | ||||||
| [](https://github.com/Awesome-Technologies/synapse-admin/actions/workflows/build-test.yml) |  | ||||||
| [](https://awesome-technologies.github.io/synapse-admin/) |  | ||||||
| [](https://hub.docker.com/r/awesometechnologies/synapse-admin) |  | ||||||
| [](https://github.com/Awesome-Technologies/synapse-admin/releases) |  | ||||||
|  |  | ||||||
| # Synapse admin ui |  | ||||||
|  |  | ||||||
| This project is built using [react-admin](https://marmelab.com/react-admin/). | This project is built using [react-admin](https://marmelab.com/react-admin/). | ||||||
|  |  | ||||||
|  | <!-- vim-markdown-toc GFM --> | ||||||
|  |  | ||||||
|  | * [Fork differences](#fork-differences) | ||||||
|  |   * [Available via CDN](#available-via-cdn) | ||||||
|  |   * [Changes](#changes) | ||||||
|  |   * [Development](#development) | ||||||
|  | * [Configuration](#configuration) | ||||||
|  |   * [Restricting available homeserver](#restricting-available-homeserver) | ||||||
|  |   * [Protecting appservice managed users](#protecting-appservice-managed-users) | ||||||
|  |   * [Providing support URL](#providing-support-url) | ||||||
|  | * [Usage](#usage) | ||||||
|  |   * [Supported Synapse](#supported-synapse) | ||||||
|  |   * [Prerequisites](#prerequisites) | ||||||
|  |   * [Use without install](#use-without-install) | ||||||
|  |   * [Step-By-Step install](#step-by-step-install) | ||||||
|  |     * [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) | ||||||
|  | * [Development](#development-1) | ||||||
|  |  | ||||||
|  | <!-- vim-markdown-toc --> | ||||||
|  |  | ||||||
|  | ## Fork differences | ||||||
|  |  | ||||||
|  | With [Awesome-Technologies/synapse-admin](https://github.com/Awesome-Technologies/synapse-admin) as the upstream, this | ||||||
|  | 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. | ||||||
|  |  | ||||||
|  | ### Available via CDN | ||||||
|  |  | ||||||
|  | On [admin.etke.cc](https://admin.etke.cc) you can find the latest version of this fork. | ||||||
|  |  | ||||||
|  | ### Changes | ||||||
|  |  | ||||||
|  | The following changes are already implemented: | ||||||
|  |  | ||||||
|  | * [Prevent admins from deleting themselves](https://github.com/etkecc/synapse-admin/pull/1) | ||||||
|  | * [Fix user's default tab not being shown](https://github.com/etkecc/synapse-admin/pull/8) | ||||||
|  | * [Add identifier when authorizing with password](https://github.com/Awesome-Technologies/synapse-admin/pull/601) | ||||||
|  | * [Add ability to toggle whether to show locked users](https://github.com/Awesome-Technologies/synapse-admin/pull/573) | ||||||
|  | * [Fix user's display name in header on user's page](https://github.com/etkecc/synapse-admin/pull/9) | ||||||
|  | * [Fix footer overlapping content](https://github.com/Awesome-Technologies/synapse-admin/issues/574) | ||||||
|  | * Switch from nginx to [SWS](https://static-web-server.net/) for serving the app, reducing the size of the Docker image | ||||||
|  | * [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) | ||||||
|  | * [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) | ||||||
|  | * [Expose user avatar URL field in the UI](https://github.com/etkecc/synapse-admin/pull/27) | ||||||
|  | * [Upgrade react-admin to v5](https://github.com/etkecc/synapse-admin/pull/40) | ||||||
|  | * [Restrict actions on specific users](https://github.com/etkecc/synapse-admin/pull/42) | ||||||
|  | * [Add `Contact support` menu item](https://github.com/etkecc/synapse-admin/pull/45) | ||||||
|  |  | ||||||
|  | _the list will be updated as new changes are added_ | ||||||
|  |  | ||||||
|  | ### Development | ||||||
|  |  | ||||||
|  | `just run-dev` to start the development stack (depending on your system speed, you may want to re-run this command if | ||||||
|  |    user creation fails) | ||||||
|  |  | ||||||
|  | After that open `http://localhost:5173` in your browser, login using the following credentials: | ||||||
|  |  | ||||||
|  | * Login: admin | ||||||
|  | * Password: admin | ||||||
|  | * Homeserver URL: http://localhost:8008 | ||||||
|  |  | ||||||
|  | ## Configuration | ||||||
|  |  | ||||||
|  | You can use `config.json` file to configure synapse-admin | ||||||
|  |  | ||||||
|  | The `config.json` can be injected into a Docker container using a bind mount. | ||||||
|  |  | ||||||
|  | ```yml | ||||||
|  | services: | ||||||
|  |   synapse-admin: | ||||||
|  |     ... | ||||||
|  |     volumes: | ||||||
|  |       ./config.json:/app/config.json:ro | ||||||
|  |     ... | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Restricting available homeserver | ||||||
|  |  | ||||||
|  | You can restrict the homeserver(s), so that the user can no longer define it himself. | ||||||
|  |  | ||||||
|  | Edit `config.json` to restrict either to a single homeserver: | ||||||
|  |  | ||||||
|  | ```json | ||||||
|  | { | ||||||
|  |   "restrictBaseUrl": "https://your-matrixs-erver.example.com" | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | or to a list of homeservers: | ||||||
|  |  | ||||||
|  | ```json | ||||||
|  | { | ||||||
|  |   "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, | ||||||
|  | you can specify the list of MXIDs (regexp) that should be prohibited from any changes, except display name and avatar. | ||||||
|  |  | ||||||
|  | Example for [mautrix-telegram](https://github.com/mautrix/telegram) | ||||||
|  |  | ||||||
|  | ```json | ||||||
|  | { | ||||||
|  |   "asManagedUsers": ["^@telegram_[a-zA-Z0-9]+:example\\.com$"] | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### 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`. | ||||||
|  |  | ||||||
|  | ```json | ||||||
|  | { | ||||||
|  |   "supportURL": "https://example.com/support" | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ## Usage | ## Usage | ||||||
|  |  | ||||||
| ### Supported Synapse | ### Supported Synapse | ||||||
| @@ -32,7 +155,7 @@ See also [Synapse administration endpoints](https://element-hq.github.io/synapse | |||||||
| ### Use without install | ### Use without install | ||||||
|  |  | ||||||
| You can use the current version of Synapse Admin without own installation direct | You can use the current version of Synapse Admin without own installation direct | ||||||
| via [GitHub Pages](https://awesome-technologies.github.io/synapse-admin/). | via [admin.etke.cc](https://admin.etke.cc). | ||||||
|  |  | ||||||
| **Note:** | **Note:** | ||||||
| If you want to use the deployment, you have to make sure that the admin endpoints (`/_synapse/admin`) are accessible for your browser. | If you want to use the deployment, you have to make sure that the admin endpoints (`/_synapse/admin`) are accessible for your browser. | ||||||
| @@ -51,22 +174,22 @@ You have three options: | |||||||
|  |  | ||||||
| - make sure you have a webserver installed that can serve static files (any webserver like nginx or apache will do) | - make sure you have a webserver installed that can serve static files (any webserver like nginx or apache will do) | ||||||
| - configure a vhost for synapse admin on your webserver | - configure a vhost for synapse admin on your webserver | ||||||
| - download the .tar.gz from the latest release: https://github.com/Awesome-Technologies/synapse-admin/releases/latest | - download the .tar.gz [from the latest release](https://github.com/etkecc/synapse-admin/releases/latest) | ||||||
| - unpack the .tar.gz | - unpack the .tar.gz | ||||||
| - move or symlink the `synapse-admin-x.x.x` into your vhosts root dir | - move or symlink the `synapse-admin` into your vhosts root dir | ||||||
| - open the url of the vhost in your browser | - open the url of the vhost in your browser | ||||||
|  |  | ||||||
| #### Steps for 2) | #### Steps for 2) | ||||||
|  |  | ||||||
| - make sure you have installed the following: git, yarn, nodejs | - make sure you have installed the following: git, yarn, nodejs | ||||||
| - download the source code: `git clone https://github.com/Awesome-Technologies/synapse-admin.git` | - download the source code: `git clone https://github.com/etkecc/synapse-admin.git` | ||||||
| - change into downloaded directory: `cd synapse-admin` | - change into downloaded directory: `cd synapse-admin` | ||||||
| - download dependencies: `yarn install` | - download dependencies: `yarn install` | ||||||
| - start web server: `yarn start` | - start web server: `yarn start` | ||||||
|  |  | ||||||
| #### Steps for 3) | #### Steps for 3) | ||||||
|  |  | ||||||
| - run the Docker container from the public docker registry: `docker run -p 8080:80 awesometechnologies/synapse-admin` or use the [docker-compose.yml](docker-compose.yml): `docker-compose up -d` | - run the Docker container from the public docker registry: `docker run -p 8080:80 ghcr.io/etkecc/synapse-admin` or use the [docker-compose.yml](docker-compose.yml): `docker-compose up -d` | ||||||
|  |  | ||||||
|   > note: if you're building on an architecture other than amd64 (for example a raspberry pi), make sure to define a maximum ram for node. otherwise the build will fail. |   > note: if you're building on an architecture other than amd64 (for example a raspberry pi), make sure to define a maximum ram for node. otherwise the build will fail. | ||||||
|  |  | ||||||
| @@ -76,7 +199,7 @@ You have three options: | |||||||
|       container_name: synapse-admin |       container_name: synapse-admin | ||||||
|       hostname: synapse-admin |       hostname: synapse-admin | ||||||
|       build: |       build: | ||||||
|         context: https://github.com/Awesome-Technologies/synapse-admin.git |         context: https://github.com/etkecc/synapse-admin.git | ||||||
|         args: |         args: | ||||||
|           - BUILDKIT_CONTEXT_KEEP_GIT_DIR=1 |           - BUILDKIT_CONTEXT_KEEP_GIT_DIR=1 | ||||||
|         #   - NODE_OPTIONS="--max_old_space_size=1024" |         #   - NODE_OPTIONS="--max_old_space_size=1024" | ||||||
| @@ -88,37 +211,6 @@ You have three options: | |||||||
|  |  | ||||||
| - browse to http://localhost:8080 | - browse to http://localhost:8080 | ||||||
|  |  | ||||||
| ### Restricting available homeserver |  | ||||||
|  |  | ||||||
| You can restrict the homeserver(s), so that the user can no longer define it himself. |  | ||||||
|  |  | ||||||
| Edit `config.json` to restrict either to a single homeserver: |  | ||||||
|  |  | ||||||
| ```json |  | ||||||
| { |  | ||||||
|   "restrictBaseUrl": "https://your-matrixs-erver.example.com" |  | ||||||
| } |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| or to a list of homeservers: |  | ||||||
|  |  | ||||||
| ```json |  | ||||||
| { |  | ||||||
|   "restrictBaseUrl": ["https://your-first-matrix-server.example.com", "https://your-second-matrix-server.example.com"] |  | ||||||
| } |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| The `config.json` can be injected into a Docker container using a bind mount. |  | ||||||
|  |  | ||||||
| ```yml |  | ||||||
| services: |  | ||||||
|   synapse-admin: |  | ||||||
|     ... |  | ||||||
|     volumes: |  | ||||||
|       ./config.json:/app/config.json:ro |  | ||||||
|     ... |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### 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. | The path prefix where synapse-admin is served can only be changed during the build step. | ||||||
| @@ -145,7 +237,7 @@ services: | |||||||
|       - /var/run/docker.sock:/var/run/docker.sock:ro |       - /var/run/docker.sock:/var/run/docker.sock:ro | ||||||
|  |  | ||||||
|   synapse-admin: |   synapse-admin: | ||||||
|     image: awesometechnologies/synapse-admin:latest |     image: etkecc/synapse-admin:latest | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     labels: |     labels: | ||||||
|       - "traefik.enable=true" |       - "traefik.enable=true" | ||||||
| @@ -156,10 +248,6 @@ services: | |||||||
|       - "traefik.http.middlewares.admin_path.stripprefix.prefixes=/admin" |       - "traefik.http.middlewares.admin_path.stripprefix.prefixes=/admin" | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ## Screenshots |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Development | ## Development | ||||||
|  |  | ||||||
| - See https://yarnpkg.com/getting-started/editor-sdks how to setup your IDE | - See https://yarnpkg.com/getting-started/editor-sdks how to setup your IDE | ||||||
|   | |||||||
							
								
								
									
										20
									
								
								docker-compose-dev.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								docker-compose-dev.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | services: | ||||||
|  |   synapse: | ||||||
|  |     image: ghcr.io/element-hq/synapse:latest | ||||||
|  |     entrypoint: python | ||||||
|  |     command: "-m synapse.app.homeserver -c /config/homeserver.yaml" | ||||||
|  |     ports: | ||||||
|  |     - "8008:8008" | ||||||
|  |     volumes: | ||||||
|  |     - ./testdata/synapse:/config | ||||||
|  |     - ./testdata/synapse.data:/media-store | ||||||
|  |  | ||||||
|  |   postgres: | ||||||
|  |     image: postgres:alpine | ||||||
|  |     volumes: | ||||||
|  |     - ./testdata/postgres.data:/var/lib/postgresql/data | ||||||
|  |     environment: | ||||||
|  |       POSTGRES_USER: synapse | ||||||
|  |       POSTGRES_PASSWORD: synapse | ||||||
|  |       POSTGRES_DB: synapse | ||||||
|  |       POSTGRES_INITDB_ARGS: "--lc-collate C --lc-ctype C --encoding UTF8" | ||||||
| @@ -2,13 +2,13 @@ services: | |||||||
|   synapse-admin: |   synapse-admin: | ||||||
|     container_name: synapse-admin |     container_name: synapse-admin | ||||||
|     hostname: synapse-admin |     hostname: synapse-admin | ||||||
|     image: awesometechnologies/synapse-admin:latest |     image: ghcr.io/etkecc/synapse-admin:latest | ||||||
|     # build: |     # build: | ||||||
|     #   context: . |     #   context: . | ||||||
|  |  | ||||||
|     # to use the docker-compose as standalone without a local repo clone, |     # to use the docker-compose as standalone without a local repo clone, | ||||||
|     # replace the context definition with this: |     # replace the context definition with this: | ||||||
|     # context: https://github.com/Awesome-Technologies/synapse-admin.git |     # context: https://github.com/etkecc/synapse-admin.git | ||||||
|  |  | ||||||
|     #  args: |     #  args: | ||||||
|     #    - BUILDKIT_CONTEXT_KEEP_GIT_DIR=1 |     #    - BUILDKIT_CONTEXT_KEEP_GIT_DIR=1 | ||||||
|   | |||||||
| @@ -121,8 +121,8 @@ | |||||||
|     </div> |     </div> | ||||||
|     <script type="module" src="/src/index.tsx"></script> |     <script type="module" src="/src/index.tsx"></script> | ||||||
|     <footer |     <footer | ||||||
|       style="position: relative; z-index: 2; height: 2em; margin-top: -2em; line-height: 2em; background-color: #eee; border: 0.5px solid #ddd"> |       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/Awesome-Technologies/synapse-admin" |       <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;"> |         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 |         Synapse-Admin <b><span id="version"></span></b> by Awesome Technologies Innovationslabor GmbH | ||||||
|       </a> |       </a> | ||||||
|   | |||||||
							
								
								
									
										40
									
								
								justfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								justfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | # Shows help | ||||||
|  | default: | ||||||
|  |     @just --list --justfile {{ justfile() }} | ||||||
|  |  | ||||||
|  | # build the app | ||||||
|  | build: __install | ||||||
|  |     @yarn run build --base=./ | ||||||
|  |  | ||||||
|  | # run the app in a development mode | ||||||
|  | run: | ||||||
|  |     @yarn start --host 0.0.0.0 | ||||||
|  |  | ||||||
|  | # run dev stack and start the app in a development mode | ||||||
|  | run-dev: | ||||||
|  |     @echo "Starting the database..." | ||||||
|  |     @docker-compose -f docker-compose-dev.yml up -d postgres | ||||||
|  |     @echo "Starting Synapse..." | ||||||
|  |     @docker-compose -f docker-compose-dev.yml up -d synapse | ||||||
|  |     @echo "Ensure admin user is registered..." | ||||||
|  |     @docker-compose -f docker-compose-dev.yml exec synapse register_new_matrix_user --admin -u admin -p admin -c /config/homeserver.yaml http://localhost:8008 || true | ||||||
|  |     @echo "Starting the app..." | ||||||
|  |     @yarn start --host 0.0.0.0 | ||||||
|  |  | ||||||
|  | # stop the dev stack | ||||||
|  | stop-dev: | ||||||
|  |     @docker-compose -f docker-compose-dev.yml stop | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 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 | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # run the app in a production mode | ||||||
|  | run-prod: build | ||||||
|  |     @python -m http.server -d dist 1313 | ||||||
|  |  | ||||||
|  | # install the project | ||||||
|  | __install: | ||||||
|  |     @yarn install --immutable --network-timeout=300000 | ||||||
							
								
								
									
										36
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								package.json
									
									
									
									
									
								
							| @@ -8,23 +8,22 @@ | |||||||
|   "homepage": ".", |   "homepage": ".", | ||||||
|   "repository": { |   "repository": { | ||||||
|     "type": "git", |     "type": "git", | ||||||
|     "url": "https://github.com/Awesome-Technologies/synapse-admin" |     "url": "https://github.com/etkecc/synapse-admin" | ||||||
|   }, |   }, | ||||||
|   "packageManager": "yarn@4.1.1", |  | ||||||
|   "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.0.0", | ||||||
|     "@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.12", |     "@types/jest": "^29.5.13", | ||||||
|     "@types/lodash": "^4.17.7", |     "@types/lodash": "^4.17.7", | ||||||
|     "@types/node": "^20.14.12", |     "@types/node": "^20.14.12", | ||||||
|     "@types/papaparse": "^5.3.14", |     "@types/papaparse": "^5.3.14", | ||||||
|     "@types/react": "^18.3.3", |     "@types/react": "^18.3.3", | ||||||
|     "@typescript-eslint/eslint-plugin": "^7.16.1", |     "@typescript-eslint/eslint-plugin": "^7.16.1", | ||||||
|     "@typescript-eslint/parser": "^7.16.1", |     "@typescript-eslint/parser": "^7.16.1", | ||||||
|     "@vitejs/plugin-react": "^4.0.0", |     "@vitejs/plugin-react": "^4.3.1", | ||||||
|     "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.29.1", | ||||||
| @@ -37,11 +36,11 @@ | |||||||
|     "jest-fetch-mock": "^3.0.3", |     "jest-fetch-mock": "^3.0.3", | ||||||
|     "prettier": "^3.3.3", |     "prettier": "^3.3.3", | ||||||
|     "react-test-renderer": "^18.3.1", |     "react-test-renderer": "^18.3.1", | ||||||
|     "ts-jest": "^29.2.3", |     "ts-jest": "^29.2.5", | ||||||
|     "ts-node": "^10.9.2", |     "ts-node": "^10.9.2", | ||||||
|     "typescript": "^5.4.5", |     "typescript": "^5.4.5", | ||||||
|     "typescript-eslint": "^7.16.1", |     "typescript-eslint": "^7.16.1", | ||||||
|     "vite": "^5.3.4", |     "vite": "^5.4.6", | ||||||
|     "vite-plugin-version-mark": "^0.1.0" |     "vite-plugin-version-mark": "^0.1.0" | ||||||
|   }, |   }, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
| @@ -49,27 +48,26 @@ | |||||||
|     "@emotion/styled": "^11.13.0", |     "@emotion/styled": "^11.13.0", | ||||||
|     "@haleos/ra-language-german": "^1.0.0", |     "@haleos/ra-language-german": "^1.0.0", | ||||||
|     "@haxqer/ra-language-chinese": "^4.16.2", |     "@haxqer/ra-language-chinese": "^4.16.2", | ||||||
|     "@mui/icons-material": "^5.16.4", |     "@mui/icons-material": "^6.1.1", | ||||||
|     "@mui/material": "^5.16.4", |     "@mui/material": "^6.1.1", | ||||||
|  |     "@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", | ||||||
|     "query-string": "^7.1.3", |     "ra-core": "^5.2.0", | ||||||
|     "ra-core": "^4.16.20", |     "ra-i18n-polyglot": "^5.2.0", | ||||||
|     "ra-i18n-polyglot": "^4.16.20", |     "ra-language-english": "^5.2.0", | ||||||
|     "ra-language-english": "^4.16.20", |     "ra-language-farsi": "^5.0.0", | ||||||
|     "ra-language-farsi": "^4.2.0", |     "ra-language-french": "^5.2.0", | ||||||
|     "ra-language-french": "^4.16.20", |  | ||||||
|     "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": "^4.16.20", |     "react-admin": "^5.2.0", | ||||||
|     "react-dom": "^18.3.1", |     "react-dom": "^18.3.1", | ||||||
|     "react-hook-form": "^7.52.1", |     "react-hook-form": "^7.53.0", | ||||||
|     "react-is": "^18.3.1", |     "react-is": "^18.3.1", | ||||||
|     "react-query": "^3.39.3", |     "react-router": "^6.26.2", | ||||||
|     "react-router": "^6.25.1", |     "react-router-dom": "^6.26.2" | ||||||
|     "react-router-dom": "^6.25.1" |  | ||||||
|   }, |   }, | ||||||
|   "scripts": { |   "scripts": { | ||||||
|     "start": "vite serve", |     "start": "vite serve", | ||||||
|   | |||||||
| @@ -1,4 +1,6 @@ | |||||||
| import { render, screen } from "@testing-library/react"; | import { render, screen, waitFor } from "@testing-library/react"; | ||||||
|  | import fetchMock from "jest-fetch-mock"; | ||||||
|  | fetchMock.enableMocks(); | ||||||
|  |  | ||||||
| import App from "./App"; | import App from "./App"; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ import polyglotI18nProvider from "ra-i18n-polyglot"; | |||||||
| import { Admin, CustomRoutes, Resource, resolveBrowserLocale } from "react-admin"; | import { Admin, CustomRoutes, Resource, resolveBrowserLocale } from "react-admin"; | ||||||
| import { Route } from "react-router-dom"; | import { Route } from "react-router-dom"; | ||||||
|  |  | ||||||
|  | import { AdminLayout } from "./components/AdminLayout"; | ||||||
| import { ImportFeature } from "./components/ImportFeature"; | import { ImportFeature } from "./components/ImportFeature"; | ||||||
| import germanMessages from "./i18n/de"; | import germanMessages from "./i18n/de"; | ||||||
| import englishMessages from "./i18n/en"; | import englishMessages from "./i18n/en"; | ||||||
| @@ -21,6 +22,7 @@ import userMediaStats from "./resources/user_media_statistics"; | |||||||
| import users from "./resources/users"; | 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"; | ||||||
|  |  | ||||||
| // TODO: Can we use lazy loading together with browser locale? | // TODO: Can we use lazy loading together with browser locale? | ||||||
| const messages = { | const messages = { | ||||||
| @@ -45,15 +47,18 @@ const i18nProvider = polyglotI18nProvider( | |||||||
|   ] |   ] | ||||||
| ); | ); | ||||||
|  |  | ||||||
|  | const queryClient = new QueryClient(); | ||||||
|  |  | ||||||
| const App = () => ( | const App = () => ( | ||||||
|  |   <QueryClientProvider client={queryClient}> | ||||||
|     <Admin |     <Admin | ||||||
|       disableTelemetry |       disableTelemetry | ||||||
|       requireAuth |       requireAuth | ||||||
|  |       layout={AdminLayout} | ||||||
|       loginPage={LoginPage} |       loginPage={LoginPage} | ||||||
|       authProvider={authProvider} |       authProvider={authProvider} | ||||||
|       dataProvider={dataProvider} |       dataProvider={dataProvider} | ||||||
|       i18nProvider={i18nProvider} |       i18nProvider={i18nProvider} | ||||||
|     darkTheme={{ palette: { mode: "dark" } }} |  | ||||||
|     > |     > | ||||||
|       <CustomRoutes> |       <CustomRoutes> | ||||||
|         <Route path="/import_users" element={<ImportFeature />} /> |         <Route path="/import_users" element={<ImportFeature />} /> | ||||||
| @@ -76,6 +81,7 @@ const App = () => ( | |||||||
|       <Resource name="room_state" /> |       <Resource name="room_state" /> | ||||||
|       <Resource name="destination_rooms" /> |       <Resource name="destination_rooms" /> | ||||||
|     </Admin> |     </Admin> | ||||||
|  |   </QueryClientProvider> | ||||||
| ); | ); | ||||||
|  |  | ||||||
| export default App; | export default App; | ||||||
|   | |||||||
| @@ -2,6 +2,8 @@ import { createContext, useContext } from "react"; | |||||||
|  |  | ||||||
| interface AppContextType { | interface AppContextType { | ||||||
|   restrictBaseUrl: string | string[]; |   restrictBaseUrl: string | string[]; | ||||||
|  |   asManagedUsers: string[]; | ||||||
|  |   supportURL: string; | ||||||
| } | } | ||||||
|  |  | ||||||
| export const AppContext = createContext({}); | export const AppContext = createContext({}); | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								src/components/AdminLayout.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								src/components/AdminLayout.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | import { Layout, Menu } from 'react-admin'; | ||||||
|  | import LiveHelpIcon from '@mui/icons-material/LiveHelp'; | ||||||
|  |  | ||||||
|  | const DEFAULT_SUPPORT_LINK = "https://github.com/etkecc/synapse-admin/issues"; | ||||||
|  | const supportLink = (): string => { | ||||||
|  |     try { | ||||||
|  |         new URL(localStorage.getItem("support_url") || ''); // Check if the URL is valid | ||||||
|  |         return localStorage.getItem("support_url") || DEFAULT_SUPPORT_LINK; | ||||||
|  |     } catch (e) { | ||||||
|  |         return DEFAULT_SUPPORT_LINK; | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | const AdminMenu = () => ( | ||||||
|  |     <Menu> | ||||||
|  |         <Menu.ResourceItems /> | ||||||
|  |         <Menu.Item to={supportLink()} target="_blank" primaryText="Contact support" leftIcon={<LiveHelpIcon />} /> | ||||||
|  |     </Menu> | ||||||
|  | ); | ||||||
|  |  | ||||||
|  | export const AdminLayout = ({ children }) => ( | ||||||
|  |     <Layout menu={AdminMenu}> | ||||||
|  |         {children} | ||||||
|  |     </Layout> | ||||||
|  | ); | ||||||
							
								
								
									
										104
									
								
								src/components/DeleteRoomButton.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										104
									
								
								src/components/DeleteRoomButton.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,104 @@ | |||||||
|  | import { Button, Dialog, DialogActions, DialogContent, DialogContentText, DialogTitle } from "@mui/material"; | ||||||
|  | import { Fragment, useState } from "react"; | ||||||
|  | import { SimpleForm, BooleanInput, useTranslate, RaRecord, useNotify, useRedirect, useDelete, NotificationType, useDeleteMany, Identifier, useUnselectAll } from "react-admin"; | ||||||
|  | import ActionDelete from "@mui/icons-material/Delete"; | ||||||
|  | import ActionCheck from "@mui/icons-material/CheckCircle"; | ||||||
|  | import AlertError from "@mui/icons-material/ErrorOutline"; | ||||||
|  |  | ||||||
|  | interface DeleteRoomButtonProps { | ||||||
|  |   selectedIds: Identifier[]; | ||||||
|  |   confirmTitle: string; | ||||||
|  |   confirmContent: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const resourceName = "rooms"; | ||||||
|  |  | ||||||
|  | const DeleteRoomButton: React.FC<DeleteRoomButtonProps> = (props) => { | ||||||
|  |   const translate = useTranslate(); | ||||||
|  |   const [open, setOpen] = useState(false); | ||||||
|  |   const [block, setBlock] = useState(true); | ||||||
|  |  | ||||||
|  |   const notify = useNotify(); | ||||||
|  |   const redirect = useRedirect(); | ||||||
|  |  | ||||||
|  |   const [deleteMany, { isLoading }] = useDeleteMany(); | ||||||
|  |   const unselectAll = useUnselectAll(resourceName); | ||||||
|  |   const recordIds = props.selectedIds; | ||||||
|  |  | ||||||
|  |   const handleDialogOpen = () => setOpen(true); | ||||||
|  |   const handleDialogClose = () => setOpen(false); | ||||||
|  |  | ||||||
|  |   const handleDelete = (values: {block: boolean}) => { | ||||||
|  |     deleteMany( | ||||||
|  |       resourceName, | ||||||
|  |       { ids: recordIds, meta: values }, | ||||||
|  |       { | ||||||
|  |         onSuccess: () => { | ||||||
|  |           notify("resources.rooms.action.erase.success"); | ||||||
|  |           handleDialogClose(); | ||||||
|  |           unselectAll(); | ||||||
|  |           redirect("/rooms"); | ||||||
|  |         }, | ||||||
|  |         onError: (error) => | ||||||
|  |           notify("resources.rooms.action.erase.failure", { type: 'error' as NotificationType }), | ||||||
|  |       } | ||||||
|  |     ); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const handleConfirm = () => { | ||||||
|  |     setOpen(false); | ||||||
|  |     handleDelete({ block: block }); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <Fragment> | ||||||
|  |       <Button | ||||||
|  |         onClick={handleDialogOpen} | ||||||
|  |         disabled={isLoading} | ||||||
|  |         className={"ra-delete-button"} | ||||||
|  |         key="button" | ||||||
|  |         size="small" | ||||||
|  |         sx={{ | ||||||
|  |           "&.MuiButton-sizeSmall": { | ||||||
|  |             lineHeight: 1.5, | ||||||
|  |           }, | ||||||
|  |         }} | ||||||
|  |         color={"error"} | ||||||
|  |         startIcon={<ActionDelete />} | ||||||
|  |       > | ||||||
|  |         {translate("ra.action.delete")} | ||||||
|  |       </Button> | ||||||
|  |       <Dialog open={open} onClose={handleDialogClose}> | ||||||
|  |         <DialogTitle>{translate(props.confirmTitle)}</DialogTitle> | ||||||
|  |         <DialogContent> | ||||||
|  |           <DialogContentText>{translate(props.confirmContent)}</DialogContentText> | ||||||
|  |           <SimpleForm toolbar={false}> | ||||||
|  |             <BooleanInput | ||||||
|  |               source="block" | ||||||
|  |               value={block} | ||||||
|  |               onChange={(event: React.ChangeEvent<HTMLInputElement>) => setBlock(event.target.checked)} | ||||||
|  |               label="resources.rooms.action.erase.fields.block" | ||||||
|  |               defaultValue={true} | ||||||
|  |             /> | ||||||
|  |           </SimpleForm> | ||||||
|  |         </DialogContent> | ||||||
|  |         <DialogActions> | ||||||
|  |           <Button disabled={false} onClick={handleDialogClose} startIcon={<AlertError />}> | ||||||
|  |             {translate("ra.action.cancel")} | ||||||
|  |           </Button> | ||||||
|  |           <Button | ||||||
|  |             disabled={false} | ||||||
|  |             onClick={handleConfirm} | ||||||
|  |             className={"ra-confirm RaConfirm-confirmPrimary"} | ||||||
|  |             autoFocus | ||||||
|  |             startIcon={<ActionCheck />} | ||||||
|  |           > | ||||||
|  |             {translate("ra.action.confirm")} | ||||||
|  |           </Button> | ||||||
|  |         </DialogActions> | ||||||
|  |       </Dialog> | ||||||
|  |     </Fragment> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default DeleteRoomButton; | ||||||
| @@ -15,7 +15,7 @@ import { | |||||||
| import { DataProvider, useTranslate } from "ra-core"; | import { DataProvider, useTranslate } from "ra-core"; | ||||||
| import { useDataProvider, useNotify, RaRecord, Title } from "react-admin"; | import { useDataProvider, useNotify, RaRecord, Title } from "react-admin"; | ||||||
|  |  | ||||||
| import { generateRandomMxId, generateRandomPassword } from "../synapse/synapse"; | import { generateRandomMxId, generateRandomPassword, returnMXID } from "../synapse/synapse"; | ||||||
|  |  | ||||||
| const LOGGING = true; | const LOGGING = true; | ||||||
|  |  | ||||||
| @@ -74,7 +74,7 @@ const FilePicker = () => { | |||||||
|  |  | ||||||
|   const [conflictMode, setConflictMode] = useState("stop"); |   const [conflictMode, setConflictMode] = useState("stop"); | ||||||
|   const [passwordMode, setPasswordMode] = useState(true); |   const [passwordMode, setPasswordMode] = useState(true); | ||||||
|   const [useridMode, setUseridMode] = useState("ignore"); |   const [useridMode, setUseridMode] = useState("update"); | ||||||
|  |  | ||||||
|   const translate = useTranslate(); |   const translate = useTranslate(); | ||||||
|   const notify = useNotify(); |   const notify = useNotify(); | ||||||
| @@ -121,7 +121,11 @@ 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 => meta.fields?.find(mF => eF === mF)); |     const missingFields = expectedFields.filter(eF => { | ||||||
|  |       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] })); | ||||||
| @@ -262,12 +266,15 @@ const FilePicker = () => { | |||||||
|         const userRecord = { ...entry }; |         const userRecord = { ...entry }; | ||||||
|         // No need to do a bunch of cryptographic random number getting if |         // No need to do a bunch of cryptographic random number getting if | ||||||
|         // we are using neither a generated password nor a generated user id. |         // we are using neither a generated password nor a generated user id. | ||||||
|         if (useridMode === "ignore" || userRecord.id === undefined) { |         if (useridMode === "ignore" || userRecord.id === undefined || userRecord.id === "") { | ||||||
|           userRecord.id = generateRandomMxId(); |           userRecord.id = generateRandomMxId(); | ||||||
|         } |         } | ||||||
|         if (passwordMode === false || entry.password === undefined) { |         if (passwordMode === false || entry.password === undefined || entry.password === "") { | ||||||
|           userRecord.password = generateRandomPassword(); |           userRecord.password = generateRandomPassword(); | ||||||
|         } |         } | ||||||
|  |         // we want to ensure that the ID is always full MXID, otherwise randomly-generated MXIDs will be in the full | ||||||
|  |         // form, but the ones from the CSV will be localpart-only. | ||||||
|  |         userRecord.id = returnMXID(userRecord.id); | ||||||
|         /* TODO record update stats (especially admin no -> yes, deactivated x -> !x, ... */ |         /* TODO record update stats (especially admin no -> yes, deactivated x -> !x, ... */ | ||||||
|  |  | ||||||
|         /* For these modes we will consider the ID that's in the record. |         /* For these modes we will consider the ID that's in the record. | ||||||
|   | |||||||
| @@ -20,7 +20,7 @@ import { | |||||||
|   useTranslate, |   useTranslate, | ||||||
|   useUnselectAll, |   useUnselectAll, | ||||||
| } from "react-admin"; | } from "react-admin"; | ||||||
| import { useMutation } from "react-query"; | import { useMutation } from "@tanstack/react-query"; | ||||||
|  |  | ||||||
| const ServerNoticeDialog = ({ open, onClose, onSubmit }) => { | const ServerNoticeDialog = ({ open, onClose, onSubmit }) => { | ||||||
|   const translate = useTranslate(); |   const translate = useTranslate(); | ||||||
| @@ -43,7 +43,6 @@ const ServerNoticeDialog = ({ open, onClose, onSubmit }) => { | |||||||
|           <TextInput |           <TextInput | ||||||
|             source="body" |             source="body" | ||||||
|             label="resources.servernotices.fields.body" |             label="resources.servernotices.fields.body" | ||||||
|             fullWidth |  | ||||||
|             multiline |             multiline | ||||||
|             rows="4" |             rows="4" | ||||||
|             resettable |             resettable | ||||||
| @@ -64,6 +63,10 @@ export const ServerNoticeButton = () => { | |||||||
|   const handleDialogOpen = () => setOpen(true); |   const handleDialogOpen = () => setOpen(true); | ||||||
|   const handleDialogClose = () => setOpen(false); |   const handleDialogClose = () => setOpen(false); | ||||||
|  |  | ||||||
|  |   if (!record) { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   const handleSend = (values: Partial<RaRecord>) => { |   const handleSend = (values: Partial<RaRecord>) => { | ||||||
|     create( |     create( | ||||||
|       "servernotices", |       "servernotices", | ||||||
| @@ -100,13 +103,12 @@ export const ServerNoticeBulkButton = () => { | |||||||
|   const unselectAllUsers = useUnselectAll("users"); |   const unselectAllUsers = useUnselectAll("users"); | ||||||
|   const dataProvider = useDataProvider(); |   const dataProvider = useDataProvider(); | ||||||
|  |  | ||||||
|   const { mutate: sendNotices, isLoading } = useMutation( |   const { mutate: sendNotices, isPending } = useMutation({ | ||||||
|     data => |     mutationFn: (data) => | ||||||
|       dataProvider.createMany("servernotices", { |       dataProvider.createMany("servernotices", { | ||||||
|         ids: selectedIds, |         ids: selectedIds, | ||||||
|         data: data, |         data: data, | ||||||
|       }), |       }), | ||||||
|     { |  | ||||||
|     onSuccess: () => { |     onSuccess: () => { | ||||||
|       notify("resources.servernotices.action.send_success"); |       notify("resources.servernotices.action.send_success"); | ||||||
|       unselectAllUsers(); |       unselectAllUsers(); | ||||||
| @@ -116,12 +118,11 @@ export const ServerNoticeBulkButton = () => { | |||||||
|       notify("resources.servernotices.action.send_failure", { |       notify("resources.servernotices.action.send_failure", { | ||||||
|         type: "error", |         type: "error", | ||||||
|       }), |       }), | ||||||
|     } |   }); | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <> |     <> | ||||||
|       <Button label="resources.servernotices.send" onClick={openDialog} disabled={isLoading}> |       <Button label="resources.servernotices.send" onClick={openDialog} disabled={isPending}> | ||||||
|         <MessageIcon /> |         <MessageIcon /> | ||||||
|       </Button> |       </Button> | ||||||
|       <ServerNoticeDialog open={open} onClose={closeDialog} onSubmit={sendNotices} /> |       <ServerNoticeDialog open={open} onClose={closeDialog} onSubmit={sendNotices} /> | ||||||
|   | |||||||
| @@ -1,9 +1,15 @@ | |||||||
| import { DeleteWithConfirmButton, DeleteWithConfirmButtonProps, useRecordContext } from "react-admin"; | import { DeleteWithConfirmButton, DeleteWithConfirmButtonProps, useRecordContext } from "react-admin"; | ||||||
|  | import { isASManaged } from "./mxid"; | ||||||
|  |  | ||||||
| export const DeviceRemoveButton = (props: DeleteWithConfirmButtonProps) => { | export const DeviceRemoveButton = (props: DeleteWithConfirmButtonProps) => { | ||||||
|   const record = useRecordContext(); |   const record = useRecordContext(); | ||||||
|   if (!record) return null; |   if (!record) return null; | ||||||
|  |  | ||||||
|  |   let isASManagedUser = false; | ||||||
|  |   if (record.user_id) { | ||||||
|  |     isASManagedUser = isASManaged(record.user_id); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <DeleteWithConfirmButton |     <DeleteWithConfirmButton | ||||||
|       {...props} |       {...props} | ||||||
| @@ -12,6 +18,7 @@ export const DeviceRemoveButton = (props: DeleteWithConfirmButtonProps) => { | |||||||
|       confirmContent="resources.devices.action.erase.content" |       confirmContent="resources.devices.action.erase.content" | ||||||
|       mutationMode="pessimistic" |       mutationMode="pessimistic" | ||||||
|       redirect={false} |       redirect={false} | ||||||
|  |       disabled={isASManagedUser} | ||||||
|       translateOptions={{ |       translateOptions={{ | ||||||
|         id: record.id, |         id: record.id, | ||||||
|         name: record.display_name ? record.display_name : record.id, |         name: record.display_name ? record.display_name : record.id, | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								src/components/error.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								src/components/error.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | export type MatrixError = { | ||||||
|  | 	errcode: string; | ||||||
|  | 	error: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const displayError = (errcode: string, status: number, message: string) => `${errcode} (${status}): ${message}`; | ||||||
| @@ -28,7 +28,7 @@ import { | |||||||
|   useRefresh, |   useRefresh, | ||||||
|   useTranslate, |   useTranslate, | ||||||
| } from "react-admin"; | } from "react-admin"; | ||||||
| import { useMutation } from "react-query"; | import { useMutation } from "@tanstack/react-query"; | ||||||
| import { Link } from "react-router-dom"; | import { Link } from "react-router-dom"; | ||||||
|  |  | ||||||
| import { dateParser } from "./date"; | import { dateParser } from "./date"; | ||||||
| @@ -55,14 +55,12 @@ const DeleteMediaDialog = ({ open, onClose, onSubmit }) => { | |||||||
|         <DialogContentText>{translate("delete_media.helper.send")}</DialogContentText> |         <DialogContentText>{translate("delete_media.helper.send")}</DialogContentText> | ||||||
|         <SimpleForm toolbar={<DeleteMediaToolbar />} onSubmit={onSubmit}> |         <SimpleForm toolbar={<DeleteMediaToolbar />} onSubmit={onSubmit}> | ||||||
|           <DateTimeInput |           <DateTimeInput | ||||||
|             fullWidth |  | ||||||
|             source="before_ts" |             source="before_ts" | ||||||
|             label="delete_media.fields.before_ts" |             label="delete_media.fields.before_ts" | ||||||
|             defaultValue={0} |             defaultValue={0} | ||||||
|             parse={dateParser} |             parse={dateParser} | ||||||
|           /> |           /> | ||||||
|           <NumberInput |           <NumberInput | ||||||
|             fullWidth |  | ||||||
|             source="size_gt" |             source="size_gt" | ||||||
|             label="delete_media.fields.size_gt" |             label="delete_media.fields.size_gt" | ||||||
|             defaultValue={0} |             defaultValue={0} | ||||||
| @@ -70,7 +68,6 @@ const DeleteMediaDialog = ({ open, onClose, onSubmit }) => { | |||||||
|             step={1024} |             step={1024} | ||||||
|           /> |           /> | ||||||
|           <BooleanInput |           <BooleanInput | ||||||
|             fullWidth |  | ||||||
|             source="keep_profiles" |             source="keep_profiles" | ||||||
|             label="delete_media.fields.keep_profiles" |             label="delete_media.fields.keep_profiles" | ||||||
|             defaultValue={true} |             defaultValue={true} | ||||||
| @@ -86,9 +83,8 @@ export const DeleteMediaButton = (props: ButtonProps) => { | |||||||
|   const [open, setOpen] = useState(false); |   const [open, setOpen] = useState(false); | ||||||
|   const notify = useNotify(); |   const notify = useNotify(); | ||||||
|   const dataProvider = useDataProvider<SynapseDataProvider>(); |   const dataProvider = useDataProvider<SynapseDataProvider>(); | ||||||
|   const { mutate: deleteMedia, isLoading } = useMutation( |   const { mutate: deleteMedia, isPending } = useMutation({ | ||||||
|     (values: DeleteMediaParams) => dataProvider.deleteMedia(values), |     mutationFn: (values: DeleteMediaParams) => dataProvider.deleteMedia(values), | ||||||
|     { |  | ||||||
|     onSuccess: () => { |     onSuccess: () => { | ||||||
|       notify("delete_media.action.send_success"); |       notify("delete_media.action.send_success"); | ||||||
|       closeDialog(); |       closeDialog(); | ||||||
| @@ -98,8 +94,7 @@ export const DeleteMediaButton = (props: ButtonProps) => { | |||||||
|         type: "error", |         type: "error", | ||||||
|       }); |       }); | ||||||
|     }, |     }, | ||||||
|     } |   }); | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   const openDialog = () => setOpen(true); |   const openDialog = () => setOpen(true); | ||||||
|   const closeDialog = () => setOpen(false); |   const closeDialog = () => setOpen(false); | ||||||
| @@ -110,7 +105,7 @@ export const DeleteMediaButton = (props: ButtonProps) => { | |||||||
|         {...props} |         {...props} | ||||||
|         label="delete_media.action.send" |         label="delete_media.action.send" | ||||||
|         onClick={openDialog} |         onClick={openDialog} | ||||||
|         disabled={isLoading} |         disabled={isPending} | ||||||
|         sx={{ |         sx={{ | ||||||
|           color: theme.palette.error.main, |           color: theme.palette.error.main, | ||||||
|           "&:hover": { |           "&:hover": { | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								src/components/mxid.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								src/components/mxid.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | import { Identifier } from "ra-core"; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Check if a user is managed by an application service | ||||||
|  |  * @param id The user ID to check | ||||||
|  |  * @returns Whether the user is managed by an application service | ||||||
|  |  */ | ||||||
|  | export const isASManaged = (id: string | Identifier): boolean => { | ||||||
|  |   const managedUsersString = localStorage.getItem("as_managed_users") || ''; | ||||||
|  |   try { | ||||||
|  |     const asManagedUsers = JSON.parse(managedUsersString).map(regex => new RegExp(regex)); | ||||||
|  |     return asManagedUsers.some(regex => regex.test(id)); | ||||||
|  |   } catch (e) { | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  | }; | ||||||
| @@ -4,6 +4,14 @@ import { SynapseTranslationMessages } from "."; | |||||||
|  |  | ||||||
| const de: SynapseTranslationMessages = { | const de: SynapseTranslationMessages = { | ||||||
|   ...formalGermanMessages, |   ...formalGermanMessages, | ||||||
|  |   ra: { | ||||||
|  |     ...formalGermanMessages.ra, | ||||||
|  |     navigation: { | ||||||
|  |       ...formalGermanMessages.ra.navigation, | ||||||
|  |       no_filtered_results: "Keine Ergebnisse", | ||||||
|  |       clear_filters: "Alle Filter entfernen", | ||||||
|  |     }, | ||||||
|  |   }, | ||||||
|   synapseadmin: { |   synapseadmin: { | ||||||
|     auth: { |     auth: { | ||||||
|       base_url: "Heimserver URL", |       base_url: "Heimserver URL", | ||||||
| @@ -125,6 +133,7 @@ const de: SynapseTranslationMessages = { | |||||||
|         erased: "Gelöscht", |         erased: "Gelöscht", | ||||||
|         guests: "Zeige Gäste", |         guests: "Zeige Gäste", | ||||||
|         show_deactivated: "Zeige deaktivierte Benutzer", |         show_deactivated: "Zeige deaktivierte Benutzer", | ||||||
|  |         show_locked: "Zeige gesperrte Benutzer", | ||||||
|         user_id: "Suche Benutzer", |         user_id: "Suche Benutzer", | ||||||
|         displayname: "Anzeigename", |         displayname: "Anzeigename", | ||||||
|         password: "Passwort", |         password: "Passwort", | ||||||
| @@ -141,10 +150,13 @@ const de: SynapseTranslationMessages = { | |||||||
|       helper: { |       helper: { | ||||||
|         password: "Durch die Änderung des Passworts wird der Benutzer von allen Sitzungen abgemeldet.", |         password: "Durch die Änderung des Passworts wird der Benutzer von allen Sitzungen abgemeldet.", | ||||||
|         deactivate: "Sie müssen ein Passwort angeben, um ein Konto wieder zu aktivieren.", |         deactivate: "Sie müssen ein Passwort angeben, um ein Konto wieder zu aktivieren.", | ||||||
|         erase: "DSGVO konformes Löschen der Benutzerdaten", |         erase: "DSGVO konformes Löschen der Benutzerdaten.", | ||||||
|  |         erase_admin_error: "Das Löschen des eigenen Benutzers ist nicht erlaubt.", | ||||||
|  |         modify_managed_user_error: "Das Ändern eines vom System verwalteten Benutzers ist nicht zulässig.", | ||||||
|       }, |       }, | ||||||
|       action: { |       action: { | ||||||
|         erase: "Lösche Benutzerdaten", |         erase: "Lösche Benutzerdaten", | ||||||
|  |         erase_avatar: "Avatar löschen" | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|     rooms: { |     rooms: { | ||||||
| @@ -197,6 +209,11 @@ const de: SynapseTranslationMessages = { | |||||||
|           title: "Raum löschen", |           title: "Raum löschen", | ||||||
|           content: |           content: | ||||||
|             "Sind Sie sicher dass Sie den Raum löschen möchten? Diese Aktion kann nicht rückgängig gemacht werden. Alle Nachrichten und Medien, die der Raum beinhaltet werden vom Server gelöscht!", |             "Sind Sie sicher dass Sie den Raum löschen möchten? Diese Aktion kann nicht rückgängig gemacht werden. Alle Nachrichten und Medien, die der Raum beinhaltet werden vom Server gelöscht!", | ||||||
|  |           fields: { | ||||||
|  |             block: "Benutzer blockieren und daran hindern, dem Raum beizutreten", | ||||||
|  |           }, | ||||||
|  |           success: "Raum/Räume erfolgreich gelöscht.", | ||||||
|  |           failure: "Der/die Raum/Räume konnten nicht gelöscht werden.", | ||||||
|         }, |         }, | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|   | |||||||
| @@ -124,6 +124,7 @@ const en: SynapseTranslationMessages = { | |||||||
|         erased: "Erased", |         erased: "Erased", | ||||||
|         guests: "Show guests", |         guests: "Show guests", | ||||||
|         show_deactivated: "Show deactivated users", |         show_deactivated: "Show deactivated users", | ||||||
|  |         show_locked: "Show locked users", | ||||||
|         user_id: "Search user", |         user_id: "Search user", | ||||||
|         displayname: "Displayname", |         displayname: "Displayname", | ||||||
|         password: "Password", |         password: "Password", | ||||||
| @@ -141,9 +142,12 @@ const en: SynapseTranslationMessages = { | |||||||
|         password: "Changing password will log user out of all sessions.", |         password: "Changing password will log user out of all sessions.", | ||||||
|         deactivate: "You must provide a password to re-activate an account.", |         deactivate: "You must provide a password to re-activate an account.", | ||||||
|         erase: "Mark the user as GDPR-erased", |         erase: "Mark the user as GDPR-erased", | ||||||
|  |         erase_admin_error: "Deleting own user is not allowed.", | ||||||
|  |         modify_managed_user_error: "Modifying a system-managed user is not allowed.", | ||||||
|       }, |       }, | ||||||
|       action: { |       action: { | ||||||
|         erase: "Erase user data", |         erase: "Erase user data", | ||||||
|  |         erase_avatar: "Erase avatar" | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|     rooms: { |     rooms: { | ||||||
| @@ -196,6 +200,11 @@ const en: SynapseTranslationMessages = { | |||||||
|           title: "Delete room", |           title: "Delete room", | ||||||
|           content: |           content: | ||||||
|             "Are you sure you want to delete the room? This cannot be undone. All messages and shared media in the room will be deleted from the server!", |             "Are you sure you want to delete the room? This cannot be undone. All messages and shared media in the room will be deleted from the server!", | ||||||
|  |           fields: { | ||||||
|  |             block: "Block and prevent users from joining the room", | ||||||
|  |           }, | ||||||
|  |           success: "Room/s successfully deleted.", | ||||||
|  |           failure: "The room/s could not be deleted.", | ||||||
|         }, |         }, | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|   | |||||||
| @@ -120,6 +120,7 @@ const fa: SynapseTranslationMessages = { | |||||||
|         deactivated: "غیرفعال", |         deactivated: "غیرفعال", | ||||||
|         guests: "نمایش مهمانان", |         guests: "نمایش مهمانان", | ||||||
|         show_deactivated: "نمایش کاربران غیرفعال شده", |         show_deactivated: "نمایش کاربران غیرفعال شده", | ||||||
|  |         show_locked: "نمایش کاربران قفل شده", | ||||||
|         user_id: "جستجوی کاربر", |         user_id: "جستجوی کاربر", | ||||||
|         displayname: "نام نمایشی", |         displayname: "نام نمایشی", | ||||||
|         password: "رمز عبور", |         password: "رمز عبور", | ||||||
| @@ -137,6 +138,8 @@ const fa: SynapseTranslationMessages = { | |||||||
|         password: "با تغییر رمز عبور کاربر از تمام دستگاه ها خارج می شود.", |         password: "با تغییر رمز عبور کاربر از تمام دستگاه ها خارج می شود.", | ||||||
|         deactivate: "برای فعالسازی مجدد حساب باید رمز عبور وارد کنید.", |         deactivate: "برای فعالسازی مجدد حساب باید رمز عبور وارد کنید.", | ||||||
|         erase: "کاربر را به عنوان GDPR پاک شده علامت گذاری کنید", |         erase: "کاربر را به عنوان GDPR پاک شده علامت گذاری کنید", | ||||||
|  |         erase_admin_error: "حذف المستخدم الخاص غير مسموح به.", | ||||||
|  |         modify_managed_user_error: "لا يُسمح بتغيير المستخدم الذي يديره النظام.", | ||||||
|       }, |       }, | ||||||
|       action: { |       action: { | ||||||
|         erase: "پاک کردن اطلاعات کاربر", |         erase: "پاک کردن اطلاعات کاربر", | ||||||
|   | |||||||
| @@ -124,6 +124,7 @@ const fr: SynapseTranslationMessages = { | |||||||
|         deactivated: "Désactivé", |         deactivated: "Désactivé", | ||||||
|         guests: "Afficher les visiteurs", |         guests: "Afficher les visiteurs", | ||||||
|         show_deactivated: "Afficher les utilisateurs désactivés", |         show_deactivated: "Afficher les utilisateurs désactivés", | ||||||
|  |         show_locked: "Afficher les utilisateurs verrouillés", | ||||||
|         user_id: "Rechercher un utilisateur", |         user_id: "Rechercher un utilisateur", | ||||||
|         displayname: "Nom d'affichage", |         displayname: "Nom d'affichage", | ||||||
|         password: "Mot de passe", |         password: "Mot de passe", | ||||||
| @@ -139,9 +140,12 @@ const fr: SynapseTranslationMessages = { | |||||||
|       helper: { |       helper: { | ||||||
|         deactivate: "Vous devrez fournir un mot de passe pour réactiver le compte.", |         deactivate: "Vous devrez fournir un mot de passe pour réactiver le compte.", | ||||||
|         erase: "Marquer l'utilisateur comme effacé conformément au RGPD", |         erase: "Marquer l'utilisateur comme effacé conformément au RGPD", | ||||||
|  |         erase_admin_error: "La suppression de son propre utilisateur n'est pas autorisée.", | ||||||
|  |         modify_managed_user_error: "La modification d'un utilisateur géré par le système n'est pas autorisée.", | ||||||
|       }, |       }, | ||||||
|       action: { |       action: { | ||||||
|         erase: "Effacer les données de l'utilisateur", |         erase: "Effacer les données de l'utilisateur", | ||||||
|  |         erase_avatar: "Effacer l'avatar", | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|     rooms: { |     rooms: { | ||||||
| @@ -194,6 +198,11 @@ const fr: SynapseTranslationMessages = { | |||||||
|           title: "Supprimer le salon", |           title: "Supprimer le salon", | ||||||
|           content: |           content: | ||||||
|             "Voulez-vous vraiment supprimer le salon ? Cette opération ne peut être annulée. Tous les messages et médias partagés du salon seront supprimés du serveur !", |             "Voulez-vous vraiment supprimer le salon ? Cette opération ne peut être annulée. Tous les messages et médias partagés du salon seront supprimés du serveur !", | ||||||
|  |             fields: { | ||||||
|  |               block: "Bloquer et empêcher les utilisateurs de rejoindre la salle", | ||||||
|  |             }, | ||||||
|  |           success: "Salle/s supprimées avec succès.", | ||||||
|  |           failure: "La/les salle/s n'ont pas pu être supprimées.", | ||||||
|         }, |         }, | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								src/i18n/index.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								src/i18n/index.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -120,6 +120,7 @@ interface SynapseTranslationMessages extends TranslationMessages { | |||||||
|         erased?: string; // TODO: fa, fr, it, zh |         erased?: string; // TODO: fa, fr, it, zh | ||||||
|         guests: string; |         guests: string; | ||||||
|         show_deactivated: string; |         show_deactivated: string; | ||||||
|  |         show_locked?: string; // TODO: de, fa, fr, it, zh | ||||||
|         user_id: string; |         user_id: string; | ||||||
|         displayname: string; |         displayname: string; | ||||||
|         password: string; |         password: string; | ||||||
| @@ -137,9 +138,12 @@ interface SynapseTranslationMessages extends TranslationMessages { | |||||||
|         password?: string; |         password?: string; | ||||||
|         deactivate: string; |         deactivate: string; | ||||||
|         erase: string; |         erase: string; | ||||||
|  |         erase_admin_error: string; | ||||||
|  |         modify_managed_user_error: string; | ||||||
|       }; |       }; | ||||||
|       action: { |       action: { | ||||||
|         erase: string; |         erase: string; | ||||||
|  |         erase_avatar: string; | ||||||
|       }; |       }; | ||||||
|     }; |     }; | ||||||
|     rooms: { |     rooms: { | ||||||
| @@ -190,6 +194,11 @@ interface SynapseTranslationMessages extends TranslationMessages { | |||||||
|         erase: { |         erase: { | ||||||
|           title: string; |           title: string; | ||||||
|           content: string; |           content: string; | ||||||
|  |           fields: { | ||||||
|  |             block: string; | ||||||
|  |           }, | ||||||
|  |           success: string; | ||||||
|  |           failure: string; | ||||||
|         }; |         }; | ||||||
|       }; |       }; | ||||||
|     }; |     }; | ||||||
|   | |||||||
| @@ -121,6 +121,7 @@ const it: SynapseTranslationMessages = { | |||||||
|         deactivated: "Disattivato", |         deactivated: "Disattivato", | ||||||
|         guests: "Mostra gli ospiti", |         guests: "Mostra gli ospiti", | ||||||
|         show_deactivated: "Mostra gli utenti disattivati", |         show_deactivated: "Mostra gli utenti disattivati", | ||||||
|  |         show_locked: "Mostra gli utenti bloccati", | ||||||
|         user_id: "Cerca utente", |         user_id: "Cerca utente", | ||||||
|         displayname: "Nickname", |         displayname: "Nickname", | ||||||
|         password: "Password", |         password: "Password", | ||||||
| @@ -141,6 +142,8 @@ const it: SynapseTranslationMessages = { | |||||||
|       }, |       }, | ||||||
|       action: { |       action: { | ||||||
|         erase: "Cancella i dati dell'utente", |         erase: "Cancella i dati dell'utente", | ||||||
|  |         erase_admin_error: "Non è consentito eliminare il proprio utente.", | ||||||
|  |         modify_managed_user_error: "La modifica di un utente gestito dal sistema non è consentita.", | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|     rooms: { |     rooms: { | ||||||
|   | |||||||
| @@ -4,6 +4,14 @@ import { SynapseTranslationMessages } from "."; | |||||||
|  |  | ||||||
| const ru: SynapseTranslationMessages = { | const ru: SynapseTranslationMessages = { | ||||||
|   ...russianMessages, |   ...russianMessages, | ||||||
|  |   ra: { | ||||||
|  |     ...russianMessages.ra, | ||||||
|  |     navigation: { | ||||||
|  |       ...russianMessages.ra.navigation, | ||||||
|  |       no_filtered_results: "Нет результатов", | ||||||
|  |       clear_filters: "Все фильтры сбросить", | ||||||
|  |     }, | ||||||
|  |   }, | ||||||
|   synapseadmin: { |   synapseadmin: { | ||||||
|     auth: { |     auth: { | ||||||
|       base_url: "Адрес домашнего сервера", |       base_url: "Адрес домашнего сервера", | ||||||
| @@ -133,6 +141,7 @@ const ru: SynapseTranslationMessages = { | |||||||
|         erased: "Удалён", |         erased: "Удалён", | ||||||
|         guests: "Показывать гостей", |         guests: "Показывать гостей", | ||||||
|         show_deactivated: "Показывать деактивированных", |         show_deactivated: "Показывать деактивированных", | ||||||
|  |         show_locked: "Показывать заблокированных", | ||||||
|         user_id: "Поиск пользователя", |         user_id: "Поиск пользователя", | ||||||
|         displayname: "Отображаемое имя", |         displayname: "Отображаемое имя", | ||||||
|         password: "Пароль", |         password: "Пароль", | ||||||
| @@ -150,9 +159,12 @@ const ru: SynapseTranslationMessages = { | |||||||
|         password: "Смена пароля завершит все сессии пользователя.", |         password: "Смена пароля завершит все сессии пользователя.", | ||||||
|         deactivate: "Вы должны предоставить пароль для реактивации учётной записи.", |         deactivate: "Вы должны предоставить пароль для реактивации учётной записи.", | ||||||
|         erase: "Пометить пользователя как удалённого в соответствии с GDPR", |         erase: "Пометить пользователя как удалённого в соответствии с GDPR", | ||||||
|  |         erase_admin_error: "Удаление собственного пользователя запрещено.", | ||||||
|  |         modify_managed_user_error: "Изменение пользователя, управляемого системой, не допускается.", | ||||||
|       }, |       }, | ||||||
|       action: { |       action: { | ||||||
|         erase: "Удалить данные пользователя", |         erase: "Удалить данные пользователя", | ||||||
|  |         erase_avatar: "Удалить аватар", | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|     rooms: { |     rooms: { | ||||||
| @@ -208,6 +220,11 @@ const ru: SynapseTranslationMessages = { | |||||||
|           title: "Удалить комнату", |           title: "Удалить комнату", | ||||||
|           content: |           content: | ||||||
|             "Действительно удалить эту комнату? Это действие будет невозможно отменить. Все сообщения и файлы в комнате будут удалены с сервера!", |             "Действительно удалить эту комнату? Это действие будет невозможно отменить. Все сообщения и файлы в комнате будут удалены с сервера!", | ||||||
|  |           fields: { | ||||||
|  |             block: "Заблокировать и запретить пользователям присоединяться к комнате", | ||||||
|  |           }, | ||||||
|  |           success: "Комната/ы успешно удалены", | ||||||
|  |           failure: "Комната/ы не могут быть удалены.", | ||||||
|         }, |         }, | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|   | |||||||
| @@ -4,6 +4,14 @@ import { SynapseTranslationMessages } from "."; | |||||||
|  |  | ||||||
| const zh: SynapseTranslationMessages = { | const zh: SynapseTranslationMessages = { | ||||||
|   ...chineseMessages, |   ...chineseMessages, | ||||||
|  |   ra: { | ||||||
|  |     ...chineseMessages.ra, | ||||||
|  |     navigation: { | ||||||
|  |       ...chineseMessages.ra.navigation, | ||||||
|  |       no_filtered_results: "没有结果", | ||||||
|  |       clear_filters: "清除所有过滤器", | ||||||
|  |     }, | ||||||
|  |   }, | ||||||
|   synapseadmin: { |   synapseadmin: { | ||||||
|     auth: { |     auth: { | ||||||
|       base_url: "服务器 URL", |       base_url: "服务器 URL", | ||||||
| @@ -120,6 +128,7 @@ const zh: SynapseTranslationMessages = { | |||||||
|         deactivated: "被禁用", |         deactivated: "被禁用", | ||||||
|         guests: "显示访客", |         guests: "显示访客", | ||||||
|         show_deactivated: "显示被禁用的账户", |         show_deactivated: "显示被禁用的账户", | ||||||
|  |         show_locked: "显示被锁定的账户", | ||||||
|         user_id: "搜索用户", |         user_id: "搜索用户", | ||||||
|         displayname: "显示名字", |         displayname: "显示名字", | ||||||
|         password: "密码", |         password: "密码", | ||||||
| @@ -134,9 +143,12 @@ const zh: SynapseTranslationMessages = { | |||||||
|       helper: { |       helper: { | ||||||
|         deactivate: "您必须提供一串密码来激活账户。", |         deactivate: "您必须提供一串密码来激活账户。", | ||||||
|         erase: "将用户标记为根据 GDPR 的要求抹除了", |         erase: "将用户标记为根据 GDPR 的要求抹除了", | ||||||
|  |         erase_admin_error: "不允许删除自己的用户", | ||||||
|  |         modify_managed_user_error: "不允许修改系统管理的用户。", | ||||||
|       }, |       }, | ||||||
|       action: { |       action: { | ||||||
|         erase: "抹除用户信息", |         erase: "抹除用户信息", | ||||||
|  |         erase_avatar: "抹掉头像", | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|     rooms: { |     rooms: { | ||||||
|   | |||||||
| @@ -4,15 +4,18 @@ import { createRoot } from "react-dom/client"; | |||||||
|  |  | ||||||
| import App from "./App"; | import App from "./App"; | ||||||
| import { AppContext } from "./AppContext"; | import { AppContext } from "./AppContext"; | ||||||
|  | import storage from "./storage"; | ||||||
|  |  | ||||||
| fetch("config.json") | fetch("config.json") | ||||||
|   .then(res => res.json()) |   .then(res => res.json()) | ||||||
|   .then(props => |   .then(props => { | ||||||
|     createRoot(document.getElementById("root")).render( |     storage.setItem("as_managed_users", JSON.stringify(props.asManagedUsers)); | ||||||
|  |     storage.setItem("support_url", props.supportURL); | ||||||
|  |     return createRoot(document.getElementById("root")).render( | ||||||
|       <React.StrictMode> |       <React.StrictMode> | ||||||
|         <AppContext.Provider value={props}> |         <AppContext.Provider value={props}> | ||||||
|           <App /> |           <App /> | ||||||
|         </AppContext.Provider> |         </AppContext.Provider> | ||||||
|       </React.StrictMode> |       </React.StrictMode> | ||||||
|     ) |     ) | ||||||
|   ); |   }); | ||||||
|   | |||||||
| @@ -8,14 +8,17 @@ import { AppContext } from "../AppContext"; | |||||||
| import englishMessages from "../i18n/en"; | import englishMessages from "../i18n/en"; | ||||||
|  |  | ||||||
| const i18nProvider = polyglotI18nProvider(() => englishMessages, "en", [{ locale: "en", name: "English" }]); | const i18nProvider = polyglotI18nProvider(() => englishMessages, "en", [{ locale: "en", name: "English" }]); | ||||||
|  | import { act } from "@testing-library/react"; | ||||||
|  |  | ||||||
| describe("LoginForm", () => { | describe("LoginForm", () => { | ||||||
|   it("renders with no restriction to homeserver", () => { |   it("renders with no restriction to homeserver", async () => { | ||||||
|  |     await act(async () => { | ||||||
|       render( |       render( | ||||||
|         <AdminContext i18nProvider={i18nProvider}> |         <AdminContext i18nProvider={i18nProvider}> | ||||||
|           <LoginPage /> |           <LoginPage /> | ||||||
|         </AdminContext> |         </AdminContext> | ||||||
|       ); |       ); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|     screen.getByText(englishMessages.synapseadmin.auth.welcome); |     screen.getByText(englishMessages.synapseadmin.auth.welcome); | ||||||
|     screen.getByRole("combobox", { name: "" }); |     screen.getByRole("combobox", { name: "" }); | ||||||
|   | |||||||
| @@ -168,7 +168,9 @@ const LoginPage = () => { | |||||||
|     const [matrixVersions, setMatrixVersions] = useState(""); |     const [matrixVersions, setMatrixVersions] = useState(""); | ||||||
|  |  | ||||||
|     const handleUsernameChange = () => { |     const handleUsernameChange = () => { | ||||||
|       if (formData.base_url || allowSingleBaseUrl) return; |       if (formData.base_url || allowSingleBaseUrl) { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|       // check if username is a full qualified userId then set base_url accordingly |       // check if username is a full qualified userId then set base_url accordingly | ||||||
|       const domain = splitMxid(formData.username)?.domain; |       const domain = splitMxid(formData.username)?.domain; | ||||||
|       if (domain) { |       if (domain) { | ||||||
| @@ -180,6 +182,9 @@ const LoginPage = () => { | |||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     useEffect(() => { |     useEffect(() => { | ||||||
|  |       if (!formData.base_url) { | ||||||
|  |         form.setValue("base_url", ""); | ||||||
|  |       } | ||||||
|       if (formData.base_url === "" && allowMultipleBaseUrls) { |       if (formData.base_url === "" && allowMultipleBaseUrls) { | ||||||
|         form.setValue("base_url", restrictBaseUrl[0]); |         form.setValue("base_url", restrictBaseUrl[0]); | ||||||
|       } |       } | ||||||
| @@ -217,7 +222,6 @@ const LoginPage = () => { | |||||||
|             disabled={loading || !supportPassAuth} |             disabled={loading || !supportPassAuth} | ||||||
|             onBlur={handleUsernameChange} |             onBlur={handleUsernameChange} | ||||||
|             resettable |             resettable | ||||||
|             fullWidth |  | ||||||
|             validate={required()} |             validate={required()} | ||||||
|           /> |           /> | ||||||
|         </Box> |         </Box> | ||||||
| @@ -229,7 +233,6 @@ const LoginPage = () => { | |||||||
|             autoComplete="current-password" |             autoComplete="current-password" | ||||||
|             disabled={loading || !supportPassAuth} |             disabled={loading || !supportPassAuth} | ||||||
|             resettable |             resettable | ||||||
|             fullWidth |  | ||||||
|             validate={required()} |             validate={required()} | ||||||
|           /> |           /> | ||||||
|         </Box> |         </Box> | ||||||
| @@ -242,7 +245,6 @@ const LoginPage = () => { | |||||||
|             disabled={loading} |             disabled={loading} | ||||||
|             readOnly={allowSingleBaseUrl} |             readOnly={allowSingleBaseUrl} | ||||||
|             resettable={allowAnyBaseUrl} |             resettable={allowAnyBaseUrl} | ||||||
|             fullWidth |  | ||||||
|             validate={[required(), validateBaseUrl]} |             validate={[required(), validateBaseUrl]} | ||||||
|           > |           > | ||||||
|             {allowMultipleBaseUrls && |             {allowMultipleBaseUrls && | ||||||
| @@ -275,9 +277,9 @@ const LoginPage = () => { | |||||||
|           <Box className="hint">{translate("synapseadmin.auth.welcome")}</Box> |           <Box className="hint">{translate("synapseadmin.auth.welcome")}</Box> | ||||||
|           <Box className="form"> |           <Box className="form"> | ||||||
|             <Select |             <Select | ||||||
|  |               fullWidth | ||||||
|               value={locale} |               value={locale} | ||||||
|               onChange={e => setLocale(e.target.value)} |               onChange={e => setLocale(e.target.value)} | ||||||
|               fullWidth |  | ||||||
|               disabled={loading} |               disabled={loading} | ||||||
|               className="select" |               className="select" | ||||||
|             > |             > | ||||||
|   | |||||||
| @@ -27,14 +27,19 @@ import { | |||||||
|   useNotify, |   useNotify, | ||||||
|   useRefresh, |   useRefresh, | ||||||
|   useTranslate, |   useTranslate, | ||||||
|  |   DateFieldProps, | ||||||
| } from "react-admin"; | } from "react-admin"; | ||||||
|  |  | ||||||
| import { DATE_FORMAT } from "../components/date"; | import { DATE_FORMAT } from "../components/date"; | ||||||
|  | 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) => ({ | const destinationRowSx = (record: RaRecord) => ({ | ||||||
|   backgroundColor: record.retry_last_ts > 0 ? "#ffcccc" : "white", |   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 />]; | ||||||
| @@ -92,6 +97,14 @@ const DestinationTitle = () => { | |||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | const RetryDateField = (props: DateFieldProps) => { | ||||||
|  |   const record = useRecordContext(props); | ||||||
|  |   if (props.source && get(record, props.source) === 0) { | ||||||
|  |     return <DateField {...props} record={{ ...record, [props.source]: null }} />; | ||||||
|  |   } | ||||||
|  |   return <DateField {...props} />; | ||||||
|  | }; | ||||||
|  |  | ||||||
| export const DestinationList = (props: ListProps) => { | export const DestinationList = (props: ListProps) => { | ||||||
|   return ( |   return ( | ||||||
|     <List |     <List | ||||||
| @@ -103,7 +116,7 @@ export const DestinationList = (props: ListProps) => { | |||||||
|       <Datagrid rowSx={destinationRowSx} rowClick={id => `${id}/show/rooms`} bulkActionButtons={false}> |       <Datagrid rowSx={destinationRowSx} rowClick={id => `${id}/show/rooms`} bulkActionButtons={false}> | ||||||
|         <TextField source="destination" /> |         <TextField source="destination" /> | ||||||
|         <DateField source="failure_ts" showTime options={DATE_FORMAT} /> |         <DateField source="failure_ts" showTime options={DATE_FORMAT} /> | ||||||
|         <DateField 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 /> | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ import { | |||||||
|   useRefresh, |   useRefresh, | ||||||
|   useUnselectAll, |   useUnselectAll, | ||||||
| } from "react-admin"; | } from "react-admin"; | ||||||
| import { useMutation } from "react-query"; | import { useMutation } from "@tanstack/react-query"; | ||||||
|  |  | ||||||
| import AvatarField from "../components/AvatarField"; | import AvatarField from "../components/AvatarField"; | ||||||
|  |  | ||||||
| @@ -70,13 +70,12 @@ export const RoomDirectoryBulkPublishButton = (props: ButtonProps) => { | |||||||
|   const refresh = useRefresh(); |   const refresh = useRefresh(); | ||||||
|   const unselectAllRooms = useUnselectAll("rooms"); |   const unselectAllRooms = useUnselectAll("rooms"); | ||||||
|   const dataProvider = useDataProvider(); |   const dataProvider = useDataProvider(); | ||||||
|   const { mutate, isLoading } = useMutation( |   const { mutate, isPending } = useMutation({ | ||||||
|     () => |     mutationFn: () => | ||||||
|       dataProvider.createMany("room_directory", { |       dataProvider.createMany("room_directory", { | ||||||
|         ids: selectedIds, |         ids: selectedIds, | ||||||
|         data: {}, |         data: {}, | ||||||
|       }), |       }), | ||||||
|     { |  | ||||||
|     onSuccess: () => { |     onSuccess: () => { | ||||||
|       notify("resources.room_directory.action.send_success"); |       notify("resources.room_directory.action.send_success"); | ||||||
|       unselectAllRooms(); |       unselectAllRooms(); | ||||||
| @@ -86,11 +85,10 @@ export const RoomDirectoryBulkPublishButton = (props: ButtonProps) => { | |||||||
|       notify("resources.room_directory.action.send_failure", { |       notify("resources.room_directory.action.send_failure", { | ||||||
|         type: "error", |         type: "error", | ||||||
|       }), |       }), | ||||||
|     } |   }); | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <Button {...props} label="resources.room_directory.action.create" onClick={mutate} disabled={isLoading}> |     <Button {...props} label="resources.room_directory.action.create" onClick={mutate} disabled={isPending}> | ||||||
|       <RoomDirectoryIcon /> |       <RoomDirectoryIcon /> | ||||||
|     </Button> |     </Button> | ||||||
|   ); |   ); | ||||||
| @@ -102,6 +100,10 @@ export const RoomDirectoryPublishButton = (props: ButtonProps) => { | |||||||
|   const refresh = useRefresh(); |   const refresh = useRefresh(); | ||||||
|   const [create, { isLoading }] = useCreate(); |   const [create, { isLoading }] = useCreate(); | ||||||
|  |  | ||||||
|  |   if (!record) { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   const handleSend = () => { |   const handleSend = () => { | ||||||
|     create( |     create( | ||||||
|       "room_directory", |       "room_directory", | ||||||
|   | |||||||
| @@ -36,6 +36,7 @@ import { | |||||||
|   TopToolbar, |   TopToolbar, | ||||||
|   useRecordContext, |   useRecordContext, | ||||||
|   useTranslate, |   useTranslate, | ||||||
|  |   useListContext, | ||||||
| } from "react-admin"; | } from "react-admin"; | ||||||
|  |  | ||||||
| import { | import { | ||||||
| @@ -45,6 +46,7 @@ import { | |||||||
|   RoomDirectoryPublishButton, |   RoomDirectoryPublishButton, | ||||||
| } from "./room_directory"; | } from "./room_directory"; | ||||||
| import { DATE_FORMAT } from "../components/date"; | import { DATE_FORMAT } from "../components/date"; | ||||||
|  | import DeleteRoomButton from "../components/DeleteRoomButton"; | ||||||
|  |  | ||||||
| const RoomPagination = () => <Pagination rowsPerPageOptions={[10, 25, 50, 100, 500, 1000]} />; | const RoomPagination = () => <Pagination rowsPerPageOptions={[10, 25, 50, 100, 500, 1000]} />; | ||||||
|  |  | ||||||
| @@ -65,13 +67,16 @@ const RoomTitle = () => { | |||||||
|  |  | ||||||
| const RoomShowActions = () => { | const RoomShowActions = () => { | ||||||
|   const record = useRecordContext(); |   const record = useRecordContext(); | ||||||
|  |   if (!record) { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|   const publishButton = record?.public ? <RoomDirectoryUnpublishButton /> : <RoomDirectoryPublishButton />; |   const publishButton = record?.public ? <RoomDirectoryUnpublishButton /> : <RoomDirectoryPublishButton />; | ||||||
|   // FIXME: refresh after (un)publish |   // FIXME: refresh after (un)publish | ||||||
|   return ( |   return ( | ||||||
|     <TopToolbar> |     <TopToolbar> | ||||||
|       {publishButton} |       {publishButton} | ||||||
|       <DeleteButton |       <DeleteRoomButton | ||||||
|         mutationMode="pessimistic" |         selectedIds={[record.id]} | ||||||
|         confirmTitle="resources.rooms.action.erase.title" |         confirmTitle="resources.rooms.action.erase.title" | ||||||
|         confirmContent="resources.rooms.action.erase.content" |         confirmContent="resources.rooms.action.erase.content" | ||||||
|       /> |       /> | ||||||
| @@ -207,17 +212,20 @@ export const RoomShow = (props: ShowProps) => { | |||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const RoomBulkActionButtons = () => ( | const RoomBulkActionButtons = () => { | ||||||
|  |   const record = useListContext(); | ||||||
|  |   return ( | ||||||
|     <> |     <> | ||||||
|       <RoomDirectoryBulkPublishButton /> |       <RoomDirectoryBulkPublishButton /> | ||||||
|       <RoomDirectoryBulkUnpublishButton /> |       <RoomDirectoryBulkUnpublishButton /> | ||||||
|     <BulkDeleteButton |       <DeleteRoomButton | ||||||
|  |         selectedIds={record.selectedIds} | ||||||
|         confirmTitle="resources.rooms.action.erase.title" |         confirmTitle="resources.rooms.action.erase.title" | ||||||
|         confirmContent="resources.rooms.action.erase.content" |         confirmContent="resources.rooms.action.erase.content" | ||||||
|       mutationMode="pessimistic" |  | ||||||
|       /> |       /> | ||||||
|     </> |     </> | ||||||
| ); |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
| const roomFilters = [<SearchInput source="search_term" alwaysOn />]; | const roomFilters = [<SearchInput source="search_term" alwaysOn />]; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,6 +8,8 @@ import PermMediaIcon from "@mui/icons-material/PermMedia"; | |||||||
| import PersonPinIcon from "@mui/icons-material/PersonPin"; | import PersonPinIcon from "@mui/icons-material/PersonPin"; | ||||||
| import SettingsInputComponentIcon from "@mui/icons-material/SettingsInputComponent"; | import SettingsInputComponentIcon from "@mui/icons-material/SettingsInputComponent"; | ||||||
| import ViewListIcon from "@mui/icons-material/ViewList"; | import ViewListIcon from "@mui/icons-material/ViewList"; | ||||||
|  | import { useEffect, useState } from "react"; | ||||||
|  | import { Alert, ownerDocument } from "@mui/material"; | ||||||
| import { | import { | ||||||
|   ArrayInput, |   ArrayInput, | ||||||
|   ArrayField, |   ArrayField, | ||||||
| @@ -42,15 +44,24 @@ import { | |||||||
|   useRecordContext, |   useRecordContext, | ||||||
|   useTranslate, |   useTranslate, | ||||||
|   Pagination, |   Pagination, | ||||||
|  |   SaveButton, | ||||||
|   CreateButton, |   CreateButton, | ||||||
|   ExportButton, |   ExportButton, | ||||||
|   TopToolbar, |   TopToolbar, | ||||||
|  |   Toolbar, | ||||||
|   NumberField, |   NumberField, | ||||||
|   useListContext, |   useListContext, | ||||||
|  |   useNotify, | ||||||
|  |   ToolbarClasses, | ||||||
|  |   Identifier, | ||||||
|  |   RaRecord, | ||||||
|  |   ImageInput, | ||||||
|  |   ImageField, | ||||||
| } from "react-admin"; | } from "react-admin"; | ||||||
| import { Link } from "react-router-dom"; | import { Link } from "react-router-dom"; | ||||||
|  |  | ||||||
| import AvatarField from "../components/AvatarField"; | import AvatarField from "../components/AvatarField"; | ||||||
|  | import { isASManaged } from "../components/mxid"; | ||||||
| import { ServerNoticeButton, ServerNoticeBulkButton } from "../components/ServerNotices"; | import { ServerNoticeButton, ServerNoticeBulkButton } from "../components/ServerNotices"; | ||||||
| import { DATE_FORMAT } from "../components/date"; | import { DATE_FORMAT } from "../components/date"; | ||||||
| import { DeviceRemoveButton } from "../components/devices"; | import { DeviceRemoveButton } from "../components/devices"; | ||||||
| @@ -90,29 +101,70 @@ const userFilters = [ | |||||||
|   <SearchInput source="name" alwaysOn />, |   <SearchInput source="name" alwaysOn />, | ||||||
|   <BooleanInput source="guests" alwaysOn />, |   <BooleanInput source="guests" alwaysOn />, | ||||||
|   <BooleanInput label="resources.users.fields.show_deactivated" source="deactivated" alwaysOn />, |   <BooleanInput label="resources.users.fields.show_deactivated" source="deactivated" alwaysOn />, | ||||||
|  |   <BooleanInput label="resources.users.fields.show_locked" source="locked" alwaysOn />, | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
| const UserBulkActionButtons = () => ( | const UserPreventSelfDelete: React.FC<{ children: React.ReactNode; ownUserIsSelected: boolean; asManagedUserIsSelected: boolean }> = props => { | ||||||
|  |   const ownUserIsSelected = props.ownUserIsSelected; | ||||||
|  |   const asManagedUserIsSelected = props.asManagedUserIsSelected; | ||||||
|  |   const notify = useNotify(); | ||||||
|  |   const translate = useTranslate(); | ||||||
|  |  | ||||||
|  |   const handleDeleteClick = (ev: React.MouseEvent<HTMLDivElement>) => { | ||||||
|  |     if (ownUserIsSelected) { | ||||||
|  |       notify(<Alert severity="error">{translate("resources.users.helper.erase_admin_error")}</Alert>); | ||||||
|  |       ev.stopPropagation(); | ||||||
|  |     } else if (asManagedUserIsSelected) { | ||||||
|  |       notify(<Alert severity="error">{translate("resources.users.helper.modify_managed_user_error")}</Alert>); | ||||||
|  |       ev.stopPropagation(); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   return <div onClickCapture={handleDeleteClick}>{props.children}</div>; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const UserBulkActionButtons = () => { | ||||||
|  |   const record = useListContext(); | ||||||
|  |   const [ownUserIsSelected, setOwnUserIsSelected] = useState(false); | ||||||
|  |   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)); | ||||||
|  |     setAsManagedUserIsSelected(selectedIds.some(id => isASManaged(id))); | ||||||
|  |   }, [selectedIds]); | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|     <> |     <> | ||||||
|       <ServerNoticeBulkButton /> |       <ServerNoticeBulkButton /> | ||||||
|  |       <UserPreventSelfDelete ownUserIsSelected={ownUserIsSelected} asManagedUserIsSelected={asManagedUserIsSelected}> | ||||||
|         <BulkDeleteButton |         <BulkDeleteButton | ||||||
|           label="resources.users.action.erase" |           label="resources.users.action.erase" | ||||||
|           confirmTitle="resources.users.helper.erase" |           confirmTitle="resources.users.helper.erase" | ||||||
|           mutationMode="pessimistic" |           mutationMode="pessimistic" | ||||||
|         /> |         /> | ||||||
|  |       </UserPreventSelfDelete> | ||||||
|     </> |     </> | ||||||
| ); |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | 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} | ||||||
|     filters={userFilters} |     filters={userFilters} | ||||||
|     filterDefaultValues={{ guests: true, deactivated: false }} |     filterDefaultValues={{ guests: true, deactivated: false, locked: false }} | ||||||
|     sort={{ field: "name", order: "ASC" }} |     sort={{ field: "name", order: "ASC" }} | ||||||
|     actions={<UserListActions />} |     actions={<UserListActions />} | ||||||
|     pagination={<UserPagination />} |     pagination={<UserPagination />} | ||||||
|   > |   > | ||||||
|     <Datagrid rowClick="edit" bulkActionButtons={<UserBulkActionButtons />}> |     <Datagrid rowClick={usersRowClick} bulkActionButtons={<UserBulkActionButtons />}> | ||||||
|       <AvatarField source="avatar_src" sx={{ height: "40px", width: "40px" }} sortBy="avatar_url" /> |       <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" /> | ||||||
| @@ -137,10 +189,18 @@ const validateAddress = [required(), maxLength(255)]; | |||||||
| const UserEditActions = () => { | const UserEditActions = () => { | ||||||
|   const record = useRecordContext(); |   const record = useRecordContext(); | ||||||
|   const translate = useTranslate(); |   const translate = useTranslate(); | ||||||
|  |   const ownUserId = localStorage.getItem("user_id"); | ||||||
|  |   let ownUserIsSelected = false; | ||||||
|  |   let asManagedUserIsSelected = false; | ||||||
|  |   if (record && record.id) { | ||||||
|  |     ownUserIsSelected = record.id === ownUserId; | ||||||
|  |     asManagedUserIsSelected = isASManaged(record.id); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <TopToolbar> |     <TopToolbar> | ||||||
|       {!record?.deactivated && <ServerNoticeButton />} |       {!record?.deactivated && <ServerNoticeButton />} | ||||||
|  |       <UserPreventSelfDelete ownUserIsSelected={ownUserIsSelected} asManagedUserIsSelected={asManagedUserIsSelected}> | ||||||
|         <DeleteButton |         <DeleteButton | ||||||
|           label="resources.users.action.erase" |           label="resources.users.action.erase" | ||||||
|           confirmTitle={translate("resources.users.helper.erase", { |           confirmTitle={translate("resources.users.helper.erase", { | ||||||
| @@ -148,12 +208,18 @@ const UserEditActions = () => { | |||||||
|           })} |           })} | ||||||
|           mutationMode="pessimistic" |           mutationMode="pessimistic" | ||||||
|         /> |         /> | ||||||
|  |       </UserPreventSelfDelete> | ||||||
|     </TopToolbar> |     </TopToolbar> | ||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export const UserCreate = (props: CreateProps) => ( | export const UserCreate = (props: CreateProps) => ( | ||||||
|   <Create {...props}> |   <Create | ||||||
|  |     {...props} | ||||||
|  |     redirect={(resource, id, data) => { | ||||||
|  |       return `users/${id}`; | ||||||
|  |     }} | ||||||
|  |   > | ||||||
|     <SimpleForm> |     <SimpleForm> | ||||||
|       <TextInput source="id" autoComplete="off" validate={validateUser} /> |       <TextInput source="id" autoComplete="off" validate={validateUser} /> | ||||||
|       <TextInput source="displayname" validate={maxLength(256)} /> |       <TextInput source="displayname" validate={maxLength(256)} /> | ||||||
| @@ -178,31 +244,101 @@ export const UserCreate = (props: CreateProps) => ( | |||||||
|  |  | ||||||
| const UserTitle = () => { | const UserTitle = () => { | ||||||
|   const record = useRecordContext(); |   const record = useRecordContext(); | ||||||
|  |   if (!record) { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   const translate = useTranslate(); |   const translate = useTranslate(); | ||||||
|  |   let username = record ? (record.displayname ? `"${record.displayname}"` : `"${record.name}"`) : "" | ||||||
|  |   if (isASManaged(record?.id)) { | ||||||
|  |     username += " 🤖"; | ||||||
|  |   } | ||||||
|   return ( |   return ( | ||||||
|     <span> |     <span> | ||||||
|       {translate("resources.users.name", { |       {translate("resources.users.name", { | ||||||
|         smart_count: 1, |         smart_count: 1, | ||||||
|       })}{" "} |       })}{" "} | ||||||
|       {record ? `"${record.displayname}"` : ""} |       {username} | ||||||
|     </span> |     </span> | ||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | const UserEditToolbar = () => { | ||||||
|  |   const record = useRecordContext(); | ||||||
|  |   const ownUserId = localStorage.getItem("user_id"); | ||||||
|  |   let ownUserIsSelected = false; | ||||||
|  |   let asManagedUserIsSelected = false; | ||||||
|  |   if (record && record.id) { | ||||||
|  |     ownUserIsSelected = record.id === ownUserId; | ||||||
|  |     asManagedUserIsSelected = isASManaged(record.id); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <> | ||||||
|  |       <div className={ToolbarClasses.defaultToolbar}> | ||||||
|  |         <Toolbar sx={{ justifyContent: "space-between" }}> | ||||||
|  |           <SaveButton /> | ||||||
|  |           <UserPreventSelfDelete ownUserIsSelected={ownUserIsSelected} asManagedUserIsSelected={asManagedUserIsSelected}> | ||||||
|  |             <DeleteButton /> | ||||||
|  |           </UserPreventSelfDelete> | ||||||
|  |         </Toolbar> | ||||||
|  |       </div> | ||||||
|  |     </> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const UserBooleanInput = props => { | ||||||
|  |   const record = useRecordContext(); | ||||||
|  |   const ownUserId = localStorage.getItem("user_id"); | ||||||
|  |   let ownUserIsSelected = false; | ||||||
|  |   let asManagedUserIsSelected = false; | ||||||
|  |   if (record) { | ||||||
|  |     ownUserIsSelected = record.id === ownUserId; | ||||||
|  |     asManagedUserIsSelected = isASManaged(record.id); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <UserPreventSelfDelete ownUserIsSelected={ownUserIsSelected} asManagedUserIsSelected={asManagedUserIsSelected}> | ||||||
|  |       <BooleanInput {...props} disabled={ownUserIsSelected || asManagedUserIsSelected} /> | ||||||
|  |     </UserPreventSelfDelete> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const UserPasswordInput = props => { | ||||||
|  |   const record = useRecordContext(); | ||||||
|  |   let asManagedUserIsSelected = false; | ||||||
|  |   if (record) { | ||||||
|  |     asManagedUserIsSelected = isASManaged(record.id); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |       <PasswordInput {...props} helperText="resources.users.helper.modify_managed_user_error" disabled={asManagedUserIsSelected} /> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
| export const UserEdit = (props: EditProps) => { | export const UserEdit = (props: EditProps) => { | ||||||
|   const translate = useTranslate(); |   const translate = useTranslate(); | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <Edit {...props} title={<UserTitle />} actions={<UserEditActions />}> |     <Edit {...props} title={<UserTitle />} actions={<UserEditActions />} mutationMode="pessimistic"> | ||||||
|       <TabbedForm> |       <TabbedForm toolbar={<UserEditToolbar />}> | ||||||
|         <FormTab label={translate("resources.users.name", { smart_count: 1 })} icon={<PersonPinIcon />}> |         <FormTab label={translate("resources.users.name", { smart_count: 1 })} icon={<PersonPinIcon />}> | ||||||
|           <AvatarField source="avatar_src" sortable={false} sx={{ height: "120px", width: "120px", float: "right" }} /> |           <AvatarField source="avatar_src" sortable={false} sx={{ height: "120px", width: "120px" }} /> | ||||||
|           <TextInput source="id" disabled /> |           <BooleanInput source="avatar_erase" label="resources.users.action.erase_avatar" /> | ||||||
|  |           <ImageInput | ||||||
|  |             source="avatar_file" | ||||||
|  |             label="resources.users.fields.avatar" | ||||||
|  |             accept={{ "image/*": [".png", ".jpg"] }} | ||||||
|  |           > | ||||||
|  |             <ImageField source="src" title="Avatar" /> | ||||||
|  |           </ImageInput> | ||||||
|  |           <TextInput source="id" readOnly /> | ||||||
|           <TextInput source="displayname" /> |           <TextInput source="displayname" /> | ||||||
|           <PasswordInput source="password" autoComplete="new-password" helperText="resources.users.helper.password" /> |           <UserPasswordInput source="password" autoComplete="new-password" helperText="resources.users.helper.password" /> | ||||||
|           <SelectInput source="user_type" choices={choices_type} translateChoice={false} resettable /> |           <SelectInput source="user_type" choices={choices_type} translateChoice={false} resettable /> | ||||||
|           <BooleanInput source="admin" /> |           <BooleanInput source="admin" /> | ||||||
|           <BooleanInput source="locked" /> |           <UserBooleanInput source="locked" /> | ||||||
|           <BooleanInput source="deactivated" helperText="resources.users.helper.deactivate" /> |           <UserBooleanInput source="deactivated" helperText="resources.users.helper.deactivate" /> | ||||||
|           <BooleanInput source="erased" disabled /> |           <BooleanInput source="erased" disabled /> | ||||||
|           <DateField source="creation_ts_ms" showTime options={DATE_FORMAT} /> |           <DateField source="creation_ts_ms" showTime options={DATE_FORMAT} /> | ||||||
|           <TextField source="consent_version" /> |           <TextField source="consent_version" /> | ||||||
| @@ -228,7 +364,7 @@ export const UserEdit = (props: EditProps) => { | |||||||
|  |  | ||||||
|         <FormTab label={translate("resources.devices.name", { smart_count: 2 })} icon={<DevicesIcon />} path="devices"> |         <FormTab label={translate("resources.devices.name", { smart_count: 2 })} icon={<DevicesIcon />} path="devices"> | ||||||
|           <ReferenceManyField reference="devices" target="user_id" label={false}> |           <ReferenceManyField reference="devices" target="user_id" label={false}> | ||||||
|             <Datagrid style={{ width: "100%" }}> |             <Datagrid style={{ width: "100%" }} bulkActionButtons={false}> | ||||||
|               <TextField source="device_id" sortable={false} /> |               <TextField source="device_id" sortable={false} /> | ||||||
|               <TextField source="display_name" sortable={false} /> |               <TextField source="display_name" sortable={false} /> | ||||||
|               <TextField source="last_seen_ip" sortable={false} /> |               <TextField source="last_seen_ip" sortable={false} /> | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ import fetchMock from "jest-fetch-mock"; | |||||||
|  |  | ||||||
| import authProvider from "./authProvider"; | import authProvider from "./authProvider"; | ||||||
| import storage from "../storage"; | import storage from "../storage"; | ||||||
|  | import { HttpError } from "ra-core"; | ||||||
|  |  | ||||||
| fetchMock.enableMocks(); | fetchMock.enableMocks(); | ||||||
|  |  | ||||||
| @@ -30,7 +31,7 @@ describe("authProvider", () => { | |||||||
|  |  | ||||||
|       expect(ret).toBe(undefined); |       expect(ret).toBe(undefined); | ||||||
|       expect(fetch).toBeCalledWith("http://example.com/_matrix/client/r0/login", { |       expect(fetch).toBeCalledWith("http://example.com/_matrix/client/r0/login", { | ||||||
|         body: '{"device_id":null,"initial_device_display_name":"Synapse Admin","type":"m.login.password","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", | ||||||
|           "Content-Type": "application/json", |           "Content-Type": "application/json", | ||||||
| @@ -100,11 +101,11 @@ describe("authProvider", () => { | |||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     it("should reject if error.status is 401", async () => { |     it("should reject if error.status is 401", async () => { | ||||||
|       await expect(authProvider.checkError({ status: 401 })).rejects.toBeUndefined(); |       await expect(authProvider.checkError(new HttpError("test-error", 401, {errcode: "test-errcode", error: "test-error"}))).rejects.toBeDefined(); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     it("should reject if error.status is 403", async () => { |     it("should reject if error.status is 403", async () => { | ||||||
|       await expect(authProvider.checkError({ status: 403 })).rejects.toBeUndefined(); |       await expect(authProvider.checkError(new HttpError("test-error", 403, {errcode: "test-errcode", error: "test-error"}))).rejects.toBeDefined(); | ||||||
|     }); |     }); | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| import { AuthProvider, Options, fetchUtils } from "react-admin"; | import { AuthProvider, HttpError, Options, fetchUtils, useTranslate } from "react-admin"; | ||||||
|  |  | ||||||
| import storage from "../storage"; | import storage from "../storage"; | ||||||
|  | import { MatrixError, displayError } from "../components/error"; | ||||||
|  |  | ||||||
| const authProvider: AuthProvider = { | const authProvider: AuthProvider = { | ||||||
|   // called when the user attempts to log in |   // called when the user attempts to log in | ||||||
| @@ -31,7 +32,10 @@ const authProvider: AuthProvider = { | |||||||
|               } |               } | ||||||
|             : { |             : { | ||||||
|                 type: "m.login.password", |                 type: "m.login.password", | ||||||
|  |                 identifier: { | ||||||
|  |                   type: "m.id.user", | ||||||
|                   user: username, |                   user: username, | ||||||
|  |                 }, | ||||||
|                 password: password, |                 password: password, | ||||||
|               } |               } | ||||||
|         ) |         ) | ||||||
| @@ -41,13 +45,36 @@ const authProvider: AuthProvider = { | |||||||
|     // use the base_url from login instead of the well_known entry from the |     // use the base_url from login instead of the well_known entry from the | ||||||
|     // server, since the admin might want to access the admin API via some |     // server, since the admin might want to access the admin API via some | ||||||
|     // private address |     // private address | ||||||
|  |     if (!base_url) { | ||||||
|  |       // there is some kind of bug with base_url being present in the form, but not submitted | ||||||
|  |       // ref: https://github.com/etkecc/synapse-admin/issues/14 | ||||||
|  |       storage.removeItem("base_url") | ||||||
|  |       throw new Error("Homeserver URL is required."); | ||||||
|  |     } | ||||||
|     base_url = base_url.replace(/\/+$/g, ""); |     base_url = base_url.replace(/\/+$/g, ""); | ||||||
|     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"; |     const login_api_url = decoded_base_url + "/_matrix/client/r0/login"; | ||||||
|  |  | ||||||
|     const { json } = await fetchUtils.fetchJson(login_api_url, options); |     let response; | ||||||
|  |     try { | ||||||
|  |       response = await fetchUtils.fetchJson(login_api_url, options); | ||||||
|  |     } catch(err) { | ||||||
|  |       const error = err as HttpError; | ||||||
|  |       const errorStatus = error.status; | ||||||
|  |       const errorBody = error.body as MatrixError; | ||||||
|  |       const errMsg = !!errorBody?.errcode ? displayError(errorBody.errcode, errorStatus, errorBody.error) : displayError("M_INVALID", errorStatus, error.message); | ||||||
|  |  | ||||||
|  |       return Promise.reject( | ||||||
|  |         new HttpError( | ||||||
|  |           errMsg, | ||||||
|  |           errorStatus, | ||||||
|  |         ) | ||||||
|  |     ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const json = response.json; | ||||||
|     storage.setItem("home_server", json.home_server); |     storage.setItem("home_server", json.home_server); | ||||||
|     storage.setItem("user_id", json.user_id); |     storage.setItem("user_id", json.user_id); | ||||||
|     storage.setItem("access_token", json.access_token); |     storage.setItem("access_token", json.access_token); | ||||||
| @@ -74,10 +101,12 @@ const authProvider: AuthProvider = { | |||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   // called when the API returns an error |   // called when the API returns an error | ||||||
|   checkError: ({ status }: { status: number }) => { |   checkError: (err: HttpError) => { | ||||||
|     console.log("checkError " + status); |     const errorBody = err.body as MatrixError; | ||||||
|  |     const status = err.status; | ||||||
|  |  | ||||||
|     if (status === 401 || status === 403) { |     if (status === 401 || status === 403) { | ||||||
|       return Promise.reject(); |       return Promise.reject({message: displayError(errorBody.errcode, status, errorBody.error)}); | ||||||
|     } |     } | ||||||
|     return Promise.resolve(); |     return Promise.resolve(); | ||||||
|   }, |   }, | ||||||
|   | |||||||
| @@ -18,7 +18,7 @@ describe("dataProvider", () => { | |||||||
|       JSON.stringify({ |       JSON.stringify({ | ||||||
|         users: [ |         users: [ | ||||||
|           { |           { | ||||||
|             name: "user_id1", |             name: "@user_id1:provider", | ||||||
|             password_hash: "password_hash1", |             password_hash: "password_hash1", | ||||||
|             is_guest: 0, |             is_guest: 0, | ||||||
|             admin: 0, |             admin: 0, | ||||||
| @@ -27,7 +27,7 @@ describe("dataProvider", () => { | |||||||
|             displayname: "User One", |             displayname: "User One", | ||||||
|           }, |           }, | ||||||
|           { |           { | ||||||
|             name: "user_id2", |             name: "@user_id2:provider", | ||||||
|             password_hash: "password_hash2", |             password_hash: "password_hash2", | ||||||
|             is_guest: 0, |             is_guest: 0, | ||||||
|             admin: 1, |             admin: 1, | ||||||
| @@ -47,7 +47,7 @@ describe("dataProvider", () => { | |||||||
|       filter: { author_id: 12 }, |       filter: { author_id: 12 }, | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     expect(users.data[0].id).toEqual("user_id1"); |     expect(users.data[0].id).toEqual("@user_id1:provider"); | ||||||
|     expect(users.total).toEqual(200); |     expect(users.total).toEqual(200); | ||||||
|     expect(fetch).toHaveBeenCalledTimes(1); |     expect(fetch).toHaveBeenCalledTimes(1); | ||||||
|   }); |   }); | ||||||
| @@ -55,7 +55,7 @@ describe("dataProvider", () => { | |||||||
|   it("fetches one user", async () => { |   it("fetches one user", async () => { | ||||||
|     fetchMock.mockResponseOnce( |     fetchMock.mockResponseOnce( | ||||||
|       JSON.stringify({ |       JSON.stringify({ | ||||||
|         name: "user_id1", |         name: "@user_id1:provider", | ||||||
|         password: "user_password", |         password: "user_password", | ||||||
|         displayname: "User", |         displayname: "User", | ||||||
|         threepids: [ |         threepids: [ | ||||||
| @@ -74,9 +74,9 @@ describe("dataProvider", () => { | |||||||
|       }) |       }) | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|     const user = await dataProvider.getOne("users", { id: "user_id1" }); |     const user = await dataProvider.getOne("users", { id: "@user_id1:provider" }); | ||||||
|  |  | ||||||
|     expect(user.data.id).toEqual("user_id1"); |     expect(user.data.id).toEqual("@user_id1:provider"); | ||||||
|     expect(user.data.displayname).toEqual("User"); |     expect(user.data.displayname).toEqual("User"); | ||||||
|     expect(fetch).toHaveBeenCalledTimes(1); |     expect(fetch).toHaveBeenCalledTimes(1); | ||||||
|   }); |   }); | ||||||
|   | |||||||
| @@ -1,20 +1,44 @@ | |||||||
| import { stringify } from "query-string"; | import { | ||||||
|  |   DataProvider, | ||||||
| import { DataProvider, DeleteParams, Identifier, Options, RaRecord, fetchUtils } from "react-admin"; |   DeleteParams, | ||||||
|  |   HttpError, | ||||||
|  |   Identifier, | ||||||
|  |   Options, | ||||||
|  |   PaginationPayload, | ||||||
|  |   RaRecord, | ||||||
|  |   SortPayload, | ||||||
|  |   UpdateParams, | ||||||
|  |   fetchUtils, | ||||||
|  |   withLifecycleCallbacks, | ||||||
|  | } from "react-admin"; | ||||||
|  |  | ||||||
| import storage from "../storage"; | import storage from "../storage"; | ||||||
|  | import { returnMXID } from "./synapse"; | ||||||
|  | import { MatrixError, displayError } from "../components/error"; | ||||||
|  |  | ||||||
| // Adds the access token to all requests | // Adds the access token to all requests | ||||||
| const jsonClient = (url: string, options: Options = {}) => { | const jsonClient = async (url: string, options: Options = {}) => { | ||||||
|   const token = storage.getItem("access_token"); |   const token = storage.getItem("access_token"); | ||||||
|   console.log("httpClient " + url); |   console.log("httpClient " + url); | ||||||
|   if (token != null) { |   if (token !== null) { | ||||||
|     options.user = { |     options.user = { | ||||||
|       authenticated: true, |       authenticated: true, | ||||||
|       token: `Bearer ${token}`, |       token: `Bearer ${token}`, | ||||||
|     }; |     }; | ||||||
|   } |   } | ||||||
|   return fetchUtils.fetchJson(url, options); |   try { | ||||||
|  |     const response = await fetchUtils.fetchJson(url, options); | ||||||
|  |     return response; | ||||||
|  |   } catch (err: any) { | ||||||
|  |     const error = err as HttpError; | ||||||
|  |     const errorStatus = error.status; | ||||||
|  |     const errorBody = error.body as MatrixError; | ||||||
|  |     const errMsg = !!errorBody?.errcode | ||||||
|  |       ? displayError(errorBody.errcode, errorStatus, errorBody.error) | ||||||
|  |       : displayError("M_INVALID", errorStatus, error.message); | ||||||
|  |  | ||||||
|  |     return Promise.reject(new HttpError(errMsg, errorStatus, errorBody)); | ||||||
|  |   } | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const mxcUrlToHttp = (mxcUrl: string) => { | const mxcUrlToHttp = (mxcUrl: string) => { | ||||||
| @@ -28,6 +52,10 @@ const mxcUrlToHttp = (mxcUrl: string) => { | |||||||
|   return `${homeserver}/_matrix/media/r0/thumbnail/${serverName}/${mediaId}?width=24&height=24&method=scale`; |   return `${homeserver}/_matrix/media/r0/thumbnail/${serverName}/${mediaId}?width=24&height=24&method=scale`; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | const filterUndefined = (obj: Record<string, any>) => { | ||||||
|  |   return Object.fromEntries(Object.entries(obj).filter(([key, value]) => value !== undefined)); | ||||||
|  | }; | ||||||
|  |  | ||||||
| interface Room { | interface Room { | ||||||
|   room_id: string; |   room_id: string; | ||||||
|   name?: string; |   name?: string; | ||||||
| @@ -214,8 +242,19 @@ export interface DeleteMediaResult { | |||||||
|   total: number; |   total: number; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | export interface UploadMediaParams { | ||||||
|  |   file: File; | ||||||
|  |   filename: string; | ||||||
|  |   content_type: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export interface UploadMediaResult { | ||||||
|  |   content_uri: string; | ||||||
|  | } | ||||||
|  |  | ||||||
| export interface SynapseDataProvider extends DataProvider { | export interface SynapseDataProvider extends DataProvider { | ||||||
|   deleteMedia: (params: DeleteMediaParams) => Promise<DeleteMediaResult>; |   deleteMedia: (params: DeleteMediaParams) => Promise<DeleteMediaResult>; | ||||||
|  |   uploadMedia: (params: UploadMediaParams) => Promise<UploadMediaResult>; | ||||||
| } | } | ||||||
|  |  | ||||||
| const resourceMap = { | const resourceMap = { | ||||||
| @@ -223,7 +262,7 @@ const resourceMap = { | |||||||
|     path: "/_synapse/admin/v2/users", |     path: "/_synapse/admin/v2/users", | ||||||
|     map: (u: User) => ({ |     map: (u: User) => ({ | ||||||
|       ...u, |       ...u, | ||||||
|       id: u.name, |       id: returnMXID(u.name), | ||||||
|       avatar_src: u.avatar_url ? mxcUrlToHttp(u.avatar_url) : undefined, |       avatar_src: u.avatar_url ? mxcUrlToHttp(u.avatar_url) : undefined, | ||||||
|       is_guest: !!u.is_guest, |       is_guest: !!u.is_guest, | ||||||
|       admin: !!u.admin, |       admin: !!u.admin, | ||||||
| @@ -234,12 +273,12 @@ const resourceMap = { | |||||||
|     data: "users", |     data: "users", | ||||||
|     total: json => json.total, |     total: json => json.total, | ||||||
|     create: (data: RaRecord) => ({ |     create: (data: RaRecord) => ({ | ||||||
|       endpoint: `/_synapse/admin/v2/users/@${encodeURIComponent(data.id)}:${storage.getItem("home_server")}`, |       endpoint: `/_synapse/admin/v2/users/${encodeURIComponent(returnMXID(data.id))}`, | ||||||
|       body: data, |       body: data, | ||||||
|       method: "PUT", |       method: "PUT", | ||||||
|     }), |     }), | ||||||
|     delete: (params: DeleteParams) => ({ |     delete: (params: DeleteParams) => ({ | ||||||
|       endpoint: `/_synapse/admin/v1/deactivate/${encodeURIComponent(params.id)}`, |       endpoint: `/_synapse/admin/v1/deactivate/${encodeURIComponent(returnMXID(params.id))}`, | ||||||
|       body: { erase: true }, |       body: { erase: true }, | ||||||
|       method: "POST", |       method: "POST", | ||||||
|     }), |     }), | ||||||
| @@ -259,7 +298,7 @@ const resourceMap = { | |||||||
|     total: json => json.total_rooms, |     total: json => json.total_rooms, | ||||||
|     delete: (params: DeleteParams) => ({ |     delete: (params: DeleteParams) => ({ | ||||||
|       endpoint: `/_synapse/admin/v2/rooms/${params.id}`, |       endpoint: `/_synapse/admin/v2/rooms/${params.id}`, | ||||||
|       body: { block: false }, |       body: { block: params.meta?.block ?? false }, | ||||||
|     }), |     }), | ||||||
|   }, |   }, | ||||||
|   reports: { |   reports: { | ||||||
| @@ -338,7 +377,7 @@ const resourceMap = { | |||||||
|       id: um.media_id, |       id: um.media_id, | ||||||
|     }), |     }), | ||||||
|     reference: (id: Identifier) => ({ |     reference: (id: Identifier) => ({ | ||||||
|       endpoint: `/_synapse/admin/v1/users/${encodeURIComponent(id)}/media`, |       endpoint: `/_synapse/admin/v1/users/${encodeURIComponent(returnMXID(id))}/media`, | ||||||
|     }), |     }), | ||||||
|     data: "media", |     data: "media", | ||||||
|     total: json => json.total, |     total: json => json.total, | ||||||
| @@ -373,7 +412,7 @@ const resourceMap = { | |||||||
|     create: (data: RaServerNotice) => ({ |     create: (data: RaServerNotice) => ({ | ||||||
|       endpoint: "/_synapse/admin/v1/send_server_notice", |       endpoint: "/_synapse/admin/v1/send_server_notice", | ||||||
|       body: { |       body: { | ||||||
|         user_id: data.id, |         user_id: returnMXID(data.id), | ||||||
|         content: { |         content: { | ||||||
|           msgtype: "m.text", |           msgtype: "m.text", | ||||||
|           body: data.body, |           body: data.body, | ||||||
| @@ -386,7 +425,7 @@ const resourceMap = { | |||||||
|     path: "/_synapse/admin/v1/statistics/users/media", |     path: "/_synapse/admin/v1/statistics/users/media", | ||||||
|     map: (usms: UserMediaStatistic) => ({ |     map: (usms: UserMediaStatistic) => ({ | ||||||
|       ...usms, |       ...usms, | ||||||
|       id: usms.user_id, |       id: returnMXID(usms.user_id), | ||||||
|     }), |     }), | ||||||
|     data: "users", |     data: "users", | ||||||
|     total: json => json.total, |     total: json => json.total, | ||||||
| @@ -488,12 +527,12 @@ function getSearchOrder(order: "ASC" | "DESC") { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| const dataProvider: SynapseDataProvider = { | const baseDataProvider: SynapseDataProvider = { | ||||||
|   getList: async (resource, params) => { |   getList: async (resource, params) => { | ||||||
|     console.log("getList " + resource); |     console.log("getList " + resource); | ||||||
|     const { user_id, name, guests, deactivated, search_term, destination, valid } = params.filter; |     const { user_id, name, guests, deactivated, locked, search_term, destination, valid } = params.filter; | ||||||
|     const { page, perPage } = params.pagination; |     const { page, perPage } = params.pagination as PaginationPayload; | ||||||
|     const { field, order } = params.sort; |     const { field, order } = params.sort as SortPayload; | ||||||
|     const from = (page - 1) * perPage; |     const from = (page - 1) * perPage; | ||||||
|     const query = { |     const query = { | ||||||
|       from: from, |       from: from, | ||||||
| @@ -504,6 +543,7 @@ const dataProvider: SynapseDataProvider = { | |||||||
|       destination: destination, |       destination: destination, | ||||||
|       guests: guests, |       guests: guests, | ||||||
|       deactivated: deactivated, |       deactivated: deactivated, | ||||||
|  |       locked: locked, | ||||||
|       valid: valid, |       valid: valid, | ||||||
|       order_by: field, |       order_by: field, | ||||||
|       dir: getSearchOrder(order), |       dir: getSearchOrder(order), | ||||||
| @@ -514,7 +554,7 @@ const dataProvider: SynapseDataProvider = { | |||||||
|     const res = resourceMap[resource]; |     const res = resourceMap[resource]; | ||||||
|  |  | ||||||
|     const endpoint_url = homeserver + res.path; |     const endpoint_url = homeserver + res.path; | ||||||
|     const url = `${endpoint_url}?${stringify(query)}`; |     const url = `${endpoint_url}?${new URLSearchParams(filterUndefined(query)).toString()}`; | ||||||
|  |  | ||||||
|     const { json } = await jsonClient(url); |     const { json } = await jsonClient(url); | ||||||
|     return { |     return { | ||||||
| @@ -568,7 +608,7 @@ const dataProvider: SynapseDataProvider = { | |||||||
|     const res = resourceMap[resource]; |     const res = resourceMap[resource]; | ||||||
|  |  | ||||||
|     const ref = res.reference(params.id); |     const ref = res.reference(params.id); | ||||||
|     const endpoint_url = `${homeserver}${ref.endpoint}?${stringify(query)}`; |     const endpoint_url = `${homeserver}${ref.endpoint}?${new URLSearchParams(filterUndefined(query)).toString()}`; | ||||||
|  |  | ||||||
|     const { json } = await jsonClient(endpoint_url); |     const { json } = await jsonClient(endpoint_url); | ||||||
|     return { |     return { | ||||||
| @@ -729,6 +769,46 @@ const dataProvider: SynapseDataProvider = { | |||||||
|     const { json } = await jsonClient(endpoint_url, { method: "POST" }); |     const { json } = await jsonClient(endpoint_url, { method: "POST" }); | ||||||
|     return json as DeleteMediaResult; |     return json as DeleteMediaResult; | ||||||
|   }, |   }, | ||||||
|  |  | ||||||
|  |   uploadMedia: async ({ file, filename, content_type }: UploadMediaParams) => { | ||||||
|  |     const base_url = storage.getItem("base_url"); | ||||||
|  |     const uploadMediaURL = `${base_url}/_matrix/media/v3/upload`; | ||||||
|  |  | ||||||
|  |     const { json } = await jsonClient(`${uploadMediaURL}?filename=${filename}`, { | ||||||
|  |       method: "POST", | ||||||
|  |       body: file, | ||||||
|  |       headers: new Headers({ | ||||||
|  |         Accept: "application/json", | ||||||
|  |         "Content-Type": content_type, | ||||||
|  |       }) as Headers, | ||||||
|  |     }); | ||||||
|  |     return json as UploadMediaResult; | ||||||
|  |   }, | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | const dataProvider = withLifecycleCallbacks(baseDataProvider, [ | ||||||
|  |   { | ||||||
|  |     resource: "users", | ||||||
|  |     beforeUpdate: async (params: UpdateParams<any>, dataProvider: DataProvider) => { | ||||||
|  |       const avatarFile = params.data.avatar_file?.rawFile; | ||||||
|  |       const avatarErase = params.data.avatar_erase; | ||||||
|  |  | ||||||
|  |       if (avatarErase) { | ||||||
|  |         params.data.avatar_url = ""; | ||||||
|  |         return params; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (avatarFile instanceof File) { | ||||||
|  |         const reponse = await dataProvider.uploadMedia({ | ||||||
|  |           file: avatarFile, | ||||||
|  |           filename: params.data.avatar_file.title, | ||||||
|  |           content_type: params.data.avatar_file.rawFile.type, | ||||||
|  |         }); | ||||||
|  |         params.data.avatar_url = reponse.content_uri; | ||||||
|  |       } | ||||||
|  |       return params; | ||||||
|  |     }, | ||||||
|  |   }, | ||||||
|  | ]); | ||||||
|  |  | ||||||
| export default dataProvider; | export default dataProvider; | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { fetchUtils } from "react-admin"; | import { Identifier, fetchUtils } from "react-admin"; | ||||||
|  |  | ||||||
| import storage from "../storage"; | import storage from "../storage"; | ||||||
|  |  | ||||||
| @@ -72,6 +72,26 @@ export function generateRandomMxId(): string { | |||||||
|   return `@${localpart}:${homeserver}`; |   return `@${localpart}:${homeserver}`; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Return the full MXID from an arbitrary input | ||||||
|  |  * @param input  the input string | ||||||
|  |  * @returns full MXID as string | ||||||
|  |  */ | ||||||
|  | export function returnMXID(input: string | Identifier): string { | ||||||
|  |   const homeserver = storage.getItem("home_server"); | ||||||
|  |  | ||||||
|  |   // Check if the input already looks like a valid MXID (i.e., starts with "@" and contains ":") | ||||||
|  |   const mxidPattern = /^@[^@:]+:[^@:]+$/; | ||||||
|  |   if (typeof input === 'string' && mxidPattern.test(input)) { | ||||||
|  |     return input; // Already a valid MXID | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // If input is not a valid MXID, assume it's a localpart and construct the MXID | ||||||
|  |   const localpart = typeof input === 'string' && input.startsWith('@') ? input.slice(1) : input; | ||||||
|  |   return `@${localpart}:${homeserver}`; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Generate a random user password |  * Generate a random user password | ||||||
|  * @returns a new random password as string |  * @returns a new random password as string | ||||||
|   | |||||||
							
								
								
									
										191
									
								
								testdata/synapse/homeserver.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										191
									
								
								testdata/synapse/homeserver.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,191 @@ | |||||||
|  | account_threepid_delegates: | ||||||
|  |   msisdn: '' | ||||||
|  | alias_creation_rules: | ||||||
|  | - action: allow | ||||||
|  |   alias: '*' | ||||||
|  |   room_id: '*' | ||||||
|  |   user_id: '*' | ||||||
|  | allow_guest_access: false | ||||||
|  | allow_public_rooms_over_federation: true | ||||||
|  | allow_public_rooms_without_auth: true | ||||||
|  | app_service_config_files: [] | ||||||
|  | autocreate_auto_join_rooms: true | ||||||
|  | background_updates: null | ||||||
|  | caches: | ||||||
|  |   global_factor: 0.5 | ||||||
|  |   per_cache_factors: null | ||||||
|  | cas_config: null | ||||||
|  | database: | ||||||
|  |   args: | ||||||
|  |     cp_max: 10 | ||||||
|  |     cp_min: 5 | ||||||
|  |     database: synapse | ||||||
|  |     host: postgres | ||||||
|  |     password: synapse | ||||||
|  |     port: 5432 | ||||||
|  |     user: synapse | ||||||
|  |   name: psycopg2 | ||||||
|  |   txn_limit: 0 | ||||||
|  | default_room_version: '10' | ||||||
|  | disable_msisdn_registration: true | ||||||
|  | email: | ||||||
|  | enable_media_repo: true | ||||||
|  | enable_metrics: false | ||||||
|  | enable_registration: false | ||||||
|  | enable_registration_captcha: false | ||||||
|  | enable_registration_without_verification: false | ||||||
|  | enable_room_list_search: true | ||||||
|  | encryption_enabled_by_default_for_room_type: 'off' | ||||||
|  | event_cache_size: 100K | ||||||
|  | federation_rr_transactions_per_room_per_second: 50 | ||||||
|  | form_secret: sLKKoFMsQUZgLAW0vU1PQQ8ca1POGMDheurGtKW0uJ20iGqtxR9O7JQ6Knvs44Wi | ||||||
|  | include_profile_data_on_invite: true | ||||||
|  | instance_map: {} | ||||||
|  | limit_profile_requests_to_users_who_share_rooms: false | ||||||
|  | limit_remote_rooms: null | ||||||
|  | listeners: | ||||||
|  | - bind_addresses: | ||||||
|  |   - '::' | ||||||
|  |   port: 8008 | ||||||
|  |   resources: | ||||||
|  |   - compress: false | ||||||
|  |     names: | ||||||
|  |     - client | ||||||
|  |   tls: false | ||||||
|  |   type: http | ||||||
|  |   x_forwarded: true | ||||||
|  | log_config: /config/synapse.log.config | ||||||
|  | macaroon_secret_key: Lg8DxGGfy95J367eVJZHLxmqP9XtN4FKdKxWpPvBS3mhviq9at8sw7KHRPkGmyqE | ||||||
|  | manhole_settings: null | ||||||
|  | max_spider_size: 10M | ||||||
|  | max_upload_size: 1024M | ||||||
|  | media_retention: | ||||||
|  |   local_media_lifetime: 30d | ||||||
|  |   remote_media_lifetime: 7d | ||||||
|  | media_storage_providers: [] | ||||||
|  | media_store_path: /media-store | ||||||
|  | metrics_flags: null | ||||||
|  | modules: [] | ||||||
|  | oembed: null | ||||||
|  | oidc_providers: null | ||||||
|  | old_signing_keys: null | ||||||
|  | opentracing: null | ||||||
|  | password_config: | ||||||
|  |   enabled: true | ||||||
|  |   localdb_enabled: true | ||||||
|  |   pepper: zfvnYqxe3GTkdJ9BlfZiAqy2zMsjOg02uBTEiWLp2hjQGqlDw33pTSTplE6HoWlF | ||||||
|  |   policy: null | ||||||
|  | pid_file: /homeserver.pid | ||||||
|  | presence: | ||||||
|  |   enabled: true | ||||||
|  | public_baseurl: http://synapse:8008/ | ||||||
|  | push: | ||||||
|  |   include_content: true | ||||||
|  | rc_admin_redaction: | ||||||
|  |   burst_count: 50 | ||||||
|  |   per_second: 1 | ||||||
|  | rc_federation: | ||||||
|  |   concurrent: 3 | ||||||
|  |   reject_limit: 50 | ||||||
|  |   sleep_delay: 500 | ||||||
|  |   sleep_limit: 10 | ||||||
|  |   window_size: 1000 | ||||||
|  | rc_invites: | ||||||
|  |   per_issuer: | ||||||
|  |     burst_count: 10 | ||||||
|  |     per_second: 0.3 | ||||||
|  |   per_room: | ||||||
|  |     burst_count: 10 | ||||||
|  |     per_second: 0.3 | ||||||
|  |   per_user: | ||||||
|  |     burst_count: 5 | ||||||
|  |     per_second: 0.003 | ||||||
|  | rc_joins: | ||||||
|  |   local: | ||||||
|  |     burst_count: 10 | ||||||
|  |     per_second: 0.1 | ||||||
|  |   remote: | ||||||
|  |     burst_count: 10 | ||||||
|  |     per_second: 0.01 | ||||||
|  | rc_login: | ||||||
|  |   account: | ||||||
|  |     burst_count: 3 | ||||||
|  |     per_second: 0.17 | ||||||
|  |   address: | ||||||
|  |     burst_count: 3 | ||||||
|  |     per_second: 0.17 | ||||||
|  |   failed_attempts: | ||||||
|  |     burst_count: 3 | ||||||
|  |     per_second: 0.17 | ||||||
|  | rc_message: | ||||||
|  |   burst_count: 10 | ||||||
|  |   per_second: 0.2 | ||||||
|  | rc_registration: | ||||||
|  |   burst_count: 3 | ||||||
|  |   per_second: 0.17 | ||||||
|  | recaptcha_private_key: '' | ||||||
|  | recaptcha_public_key: '' | ||||||
|  | redaction_retention_period: 5m | ||||||
|  | redis: | ||||||
|  |   enabled: false | ||||||
|  |   host: null | ||||||
|  |   password: null | ||||||
|  |   port: 6379 | ||||||
|  | registration_requires_token: false | ||||||
|  | registration_shared_secret: jBUKJozByo8s3bvKtYFpB350ZAnxGlzXsDpAZkgOFJuQfKAFHhqbc2dw8D54u4T9 | ||||||
|  | report_stats: false | ||||||
|  | require_auth_for_profile_requests: false | ||||||
|  | retention: | ||||||
|  |   enabled: true | ||||||
|  |   purge_jobs: | ||||||
|  |   - interval: 12h | ||||||
|  | room_list_publication_rules: | ||||||
|  | - action: allow | ||||||
|  |   alias: '*' | ||||||
|  |   room_id: '*' | ||||||
|  |   user_id: '*' | ||||||
|  | room_prejoin_state: null | ||||||
|  | saml2_config: | ||||||
|  |   sp_config: null | ||||||
|  |   user_mapping_provider: | ||||||
|  |     config: null | ||||||
|  | server_name: synapse | ||||||
|  | signing_key_path: /config/synapse.signing.key | ||||||
|  | spam_checker: [] | ||||||
|  | sso: null | ||||||
|  | stats: null | ||||||
|  | stream_writers: {} | ||||||
|  | templates: null | ||||||
|  | tls_certificate_path: null | ||||||
|  | tls_private_key_path: null | ||||||
|  | trusted_key_servers: | ||||||
|  | - server_name: matrix.org | ||||||
|  | turn_allow_guests: false | ||||||
|  | ui_auth: null | ||||||
|  | url_preview_accept_language: | ||||||
|  | - en-US | ||||||
|  | - en | ||||||
|  | url_preview_enabled: true | ||||||
|  | url_preview_ip_range_blacklist: | ||||||
|  | - 127.0.0.0/8 | ||||||
|  | - 10.0.0.0/8 | ||||||
|  | - 172.16.0.0/12 | ||||||
|  | - 192.168.0.0/16 | ||||||
|  | - 100.64.0.0/10 | ||||||
|  | - 192.0.0.0/24 | ||||||
|  | - 169.254.0.0/16 | ||||||
|  | - 192.88.99.0/24 | ||||||
|  | - 198.18.0.0/15 | ||||||
|  | - 192.0.2.0/24 | ||||||
|  | - 198.51.100.0/24 | ||||||
|  | - 203.0.113.0/24 | ||||||
|  | - 224.0.0.0/4 | ||||||
|  | - ::1/128 | ||||||
|  | - fe80::/10 | ||||||
|  | - fc00::/7 | ||||||
|  | - 2001:db8::/32 | ||||||
|  | - ff00::/8 | ||||||
|  | - fec0::/10 | ||||||
|  | user_directory: null | ||||||
|  | user_ips_max_age: 5m | ||||||
|  |  | ||||||
							
								
								
									
										28
									
								
								testdata/synapse/synapse.log.config
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								testdata/synapse/synapse.log.config
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | |||||||
|  | version: 1 | ||||||
|  | formatters: | ||||||
|  |     precise: | ||||||
|  |         format: '%(asctime)s - %(name)s - %(lineno)d - %(levelname)s - %(request)s - %(message)s' | ||||||
|  | filters: | ||||||
|  |     context: | ||||||
|  |         (): synapse.util.logcontext.LoggingContextFilter | ||||||
|  |         request: "" | ||||||
|  | handlers: | ||||||
|  |     console: | ||||||
|  |         class: logging.StreamHandler | ||||||
|  |         formatter: precise | ||||||
|  |         filters: [context] | ||||||
|  | loggers: | ||||||
|  |     synapse: | ||||||
|  |         level: INFO | ||||||
|  |     shared_secret_authenticator: | ||||||
|  |         level: INFO | ||||||
|  |     rest_auth_provider: | ||||||
|  |         level: INFO | ||||||
|  |     synapse.storage.SQL: | ||||||
|  |         # beware: increasing this to DEBUG will make synapse log sensitive | ||||||
|  |         # information such as access tokens. | ||||||
|  |         level: INFO | ||||||
|  | root: | ||||||
|  |     level: INFO | ||||||
|  |     handlers: [console] | ||||||
|  |  | ||||||
							
								
								
									
										1
									
								
								testdata/synapse/synapse.signing.key
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								testdata/synapse/synapse.signing.key
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | ed25519 a_FswB rsh+VxdR4YUv6rFM6393VmSEJJxzaDrdwlVwLe2rcRo | ||||||
| @@ -24,7 +24,7 @@ | |||||||
|     /* Strict Type-Checking Options */ |     /* Strict Type-Checking Options */ | ||||||
|     "strict": true                            /* Enable all strict type-checking options. */, |     "strict": true                            /* Enable all strict type-checking options. */, | ||||||
|     "noImplicitAny": false                    /* Raise error on expressions and declarations with an implied 'any' type. */, |     "noImplicitAny": false                    /* Raise error on expressions and declarations with an implied 'any' type. */, | ||||||
|     // "strictNullChecks": true,              /* Enable strict null checks. */ |     "strictNullChecks": true,                 /* Enable strict null checks. */ | ||||||
|     // "strictFunctionTypes": true,           /* Enable strict checking of function types. */ |     // "strictFunctionTypes": true,           /* Enable strict checking of function types. */ | ||||||
|     // "strictPropertyInitialization": true,  /* Enable strict checking of property initialization in classes. */ |     // "strictPropertyInitialization": true,  /* Enable strict checking of property initialization in classes. */ | ||||||
|     // "noImplicitThis": true,                /* Raise error on 'this' expressions with an implied 'any' type. */ |     // "noImplicitThis": true,                /* Raise error on 'this' expressions with an implied 'any' type. */ | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ export default defineConfig({ | |||||||
|   plugins: [ |   plugins: [ | ||||||
|     react(), |     react(), | ||||||
|     vitePluginVersionMark({ |     vitePluginVersionMark({ | ||||||
|       command: "git describe --tags", |       command: "git describe --tags || git rev-parse --short HEAD", | ||||||
|       ifMeta: true, |       ifMeta: true, | ||||||
|       ifLog: true, |       ifLog: true, | ||||||
|       ifGlobal: true, |       ifGlobal: true, | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user