Compare commits
	
		
			20 Commits
		
	
	
		
			v0.10.3-et
			...
			master
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fa3f2437a3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8dc5238fcb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 238350b940 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 99bf7b1889 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d72c91644d | ||
|   | e8e28b5df1 | ||
|   | d5c10b6e02 | ||
|   | 3085b9ffa0 | ||
|   | b2a3fb0f87 | ||
|   | 1e8b4cc885 | ||
|   | 4d1a9cc147 | ||
|   | 1b8b702270 | ||
|   | 61c32fb473 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ad876bb790 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2524848dae | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 669c1f3079 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 590f673167 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 307793f000 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 96f549fe42 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3de4332477 | 
							
								
								
									
										3
									
								
								.github/workflows/build-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.github/workflows/build-test.yml
									
									
									
									
										vendored
									
									
								
							| @@ -5,6 +5,9 @@ on: | |||||||
|     branches: ["master"] |     branches: ["master"] | ||||||
|   pull_request: |   pull_request: | ||||||
|  |  | ||||||
|  | permissions: | ||||||
|  |   contents: read | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   check: |   check: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								.github/workflows/edge_ghpage.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/edge_ghpage.yml
									
									
									
									
										vendored
									
									
								
							| @@ -5,6 +5,8 @@ on: | |||||||
|     branches: |     branches: | ||||||
|       - main |       - main | ||||||
|       - master |       - master | ||||||
|  | permissions: | ||||||
|  |   contents: write | ||||||
| jobs: | jobs: | ||||||
|   build-and-deploy: |   build-and-deploy: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
| @@ -23,7 +25,7 @@ jobs: | |||||||
|           yarn build --base=/synapse-admin |           yarn build --base=/synapse-admin | ||||||
|  |  | ||||||
|       - name: Deploy 🚀 |       - name: Deploy 🚀 | ||||||
|         uses: JamesIves/github-pages-deploy-action@v4.6.3 |         uses: JamesIves/github-pages-deploy-action@v4.7.3 | ||||||
|         with: |         with: | ||||||
|           branch: gh-pages |           branch: gh-pages | ||||||
|           folder: dist |           folder: dist | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/github-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/github-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -23,7 +23,7 @@ jobs: | |||||||
|           version=`git describe --dirty --tags || echo unknown` |           version=`git describe --dirty --tags || echo unknown` | ||||||
|           cp -r dist synapse-admin-$version |           cp -r dist synapse-admin-$version | ||||||
|           tar chvzf dist/synapse-admin-$version.tar.gz synapse-admin-$version |           tar chvzf dist/synapse-admin-$version.tar.gz synapse-admin-$version | ||||||
|       - uses: softprops/action-gh-release@c062e08bd532815e2082a85e87e3ef29c3e6d191 |       - uses: softprops/action-gh-release@da05d552573ad5aba039eaac05058a918a7bf631 | ||||||
|         with: |         with: | ||||||
|           files: dist/*.tar.gz |           files: dist/*.tar.gz | ||||||
|         env: |         env: | ||||||
|   | |||||||
							
								
								
									
										893
									
								
								.yarn/releases/yarn-4.1.1.cjs
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										893
									
								
								.yarn/releases/yarn-4.1.1.cjs
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										925
									
								
								.yarn/releases/yarn-4.4.1.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										925
									
								
								.yarn/releases/yarn-4.4.1.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1 +1 @@ | |||||||
| yarnPath: .yarn/releases/yarn-4.1.1.cjs | yarnPath: .yarn/releases/yarn-4.4.1.cjs | ||||||
|   | |||||||
							
								
								
									
										29
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										29
									
								
								package.json
									
									
									
									
									
								
							| @@ -10,10 +10,11 @@ | |||||||
|     "type": "git", |     "type": "git", | ||||||
|     "url": "https://github.com/Awesome-Technologies/synapse-admin" |     "url": "https://github.com/Awesome-Technologies/synapse-admin" | ||||||
|   }, |   }, | ||||||
|   "packageManager": "yarn@4.1.1", |   "packageManager": "yarn@4.4.1", | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@eslint/js": "^9.7.0", |     "@eslint/js": "^9.7.0", | ||||||
|     "@testing-library/dom": "^10.0.0", |     "@mui/utils": "^6.1.3", | ||||||
|  |     "@testing-library/dom": "^10.4.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", | ||||||
| @@ -30,7 +31,7 @@ | |||||||
|     "eslint-plugin-import": "^2.29.1", |     "eslint-plugin-import": "^2.29.1", | ||||||
|     "eslint-plugin-jsx-a11y": "^6.9.0", |     "eslint-plugin-jsx-a11y": "^6.9.0", | ||||||
|     "eslint-plugin-prettier": "^5.2.1", |     "eslint-plugin-prettier": "^5.2.1", | ||||||
|     "eslint-plugin-unused-imports": "^3.2.0", |     "eslint-plugin-unused-imports": "^4.1.4", | ||||||
|     "eslint-plugin-yaml": "^1.0.3", |     "eslint-plugin-yaml": "^1.0.3", | ||||||
|     "jest": "^29.7.0", |     "jest": "^29.7.0", | ||||||
|     "jest-environment-jsdom": "^29.7.0", |     "jest-environment-jsdom": "^29.7.0", | ||||||
| @@ -40,8 +41,8 @@ | |||||||
|     "ts-jest": "^29.2.3", |     "ts-jest": "^29.2.3", | ||||||
|     "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": "^8.32.1", | ||||||
|     "vite": "^5.3.4", |     "vite": "^6.3.5", | ||||||
|     "vite-plugin-version-mark": "^0.1.0" |     "vite-plugin-version-mark": "^0.1.0" | ||||||
|   }, |   }, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
| @@ -51,25 +52,25 @@ | |||||||
|     "@haxqer/ra-language-chinese": "^4.16.2", |     "@haxqer/ra-language-chinese": "^4.16.2", | ||||||
|     "@mui/icons-material": "^5.16.4", |     "@mui/icons-material": "^5.16.4", | ||||||
|     "@mui/material": "^5.16.4", |     "@mui/material": "^5.16.4", | ||||||
|  |     "@tanstack/react-query": "^5.59.12", | ||||||
|     "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", |     "query-string": "^7.1.3", | ||||||
|     "ra-core": "^4.16.20", |     "ra-core": "^5.2.3", | ||||||
|     "ra-i18n-polyglot": "^4.16.20", |     "ra-i18n-polyglot": "^5.2.3", | ||||||
|     "ra-language-english": "^4.16.20", |     "ra-language-english": "^5.8.2", | ||||||
|     "ra-language-farsi": "^4.2.0", |     "ra-language-farsi": "^5.0.0", | ||||||
|     "ra-language-french": "^4.16.20", |     "ra-language-french": "^5.2.3", | ||||||
|     "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.3", | ||||||
|     "react-dom": "^18.3.1", |     "react-dom": "^18.3.1", | ||||||
|     "react-hook-form": "^7.52.1", |     "react-hook-form": "^7.52.1", | ||||||
|     "react-is": "^18.3.1", |     "react-is": "^18.3.1", | ||||||
|     "react-query": "^3.39.3", |     "react-router": "^6.28.1", | ||||||
|     "react-router": "^6.25.1", |     "react-router-dom": "^6.28.1" | ||||||
|     "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 } from "@testing-library/react"; | ||||||
|  | import fetchMock from "jest-fetch-mock"; | ||||||
|  | fetchMock.enableMocks(); | ||||||
|  |  | ||||||
| import App from "./App"; | import App from "./App"; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -53,7 +53,6 @@ const App = () => ( | |||||||
|     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 />} /> | ||||||
|   | |||||||
| @@ -121,7 +121,7 @@ const FilePicker = () => { | |||||||
|  |  | ||||||
|   const verifyCsv = ({ data, meta, errors }: ParseResult<ImportLine>, { setValues, setStats, setError }) => { |   const verifyCsv = ({ data, meta, errors }: ParseResult<ImportLine>, { setValues, setStats, setError }) => { | ||||||
|     /* First, verify the presence of required fields */ |     /* First, verify the presence of required fields */ | ||||||
|     const missingFields = expectedFields.filter(eF => meta.fields?.find(mF => eF === mF)); |     const missingFields = expectedFields.filter(eF => !meta.fields?.includes(eF)); | ||||||
|  |  | ||||||
|     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] })); | ||||||
|   | |||||||
| @@ -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; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   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} /> | ||||||
|   | |||||||
| @@ -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": { | ||||||
|   | |||||||
| @@ -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", | ||||||
|   | |||||||
| @@ -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", | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								src/i18n/index.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								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; | ||||||
|   | |||||||
| @@ -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: "Адрес домашнего сервера", | ||||||
|   | |||||||
| @@ -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", | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ import { createRoot } from "react-dom/client"; | |||||||
| import App from "./App"; | import App from "./App"; | ||||||
| import { AppContext } from "./AppContext"; | import { AppContext } from "./AppContext"; | ||||||
|  |  | ||||||
| fetch("config.json") | fetch(`${import.meta.env.BASE_URL}/config.json`) | ||||||
|   .then(res => res.json()) |   .then(res => res.json()) | ||||||
|   .then(props => |   .then(props => | ||||||
|     createRoot(document.getElementById("root")).render( |     createRoot(document.getElementById("root")).render( | ||||||
|   | |||||||
| @@ -217,7 +217,6 @@ const LoginPage = () => { | |||||||
|             disabled={loading || !supportPassAuth} |             disabled={loading || !supportPassAuth} | ||||||
|             onBlur={handleUsernameChange} |             onBlur={handleUsernameChange} | ||||||
|             resettable |             resettable | ||||||
|             fullWidth |  | ||||||
|             validate={required()} |             validate={required()} | ||||||
|           /> |           /> | ||||||
|         </Box> |         </Box> | ||||||
| @@ -229,7 +228,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 +240,6 @@ const LoginPage = () => { | |||||||
|             disabled={loading} |             disabled={loading} | ||||||
|             readOnly={allowSingleBaseUrl} |             readOnly={allowSingleBaseUrl} | ||||||
|             resettable={allowAnyBaseUrl} |             resettable={allowAnyBaseUrl} | ||||||
|             fullWidth |  | ||||||
|             validate={[required(), validateBaseUrl]} |             validate={[required(), validateBaseUrl]} | ||||||
|           > |           > | ||||||
|             {allowMultipleBaseUrls && |             {allowMultipleBaseUrls && | ||||||
|   | |||||||
| @@ -1,9 +1,11 @@ | |||||||
|  | import { get } from "lodash"; | ||||||
| import { MouseEvent } from "react"; | import { MouseEvent } from "react"; | ||||||
|  |  | ||||||
| import AutorenewIcon from "@mui/icons-material/Autorenew"; | import AutorenewIcon from "@mui/icons-material/Autorenew"; | ||||||
| import DestinationsIcon from "@mui/icons-material/CloudQueue"; | import DestinationsIcon from "@mui/icons-material/CloudQueue"; | ||||||
| import FolderSharedIcon from "@mui/icons-material/FolderShared"; | import FolderSharedIcon from "@mui/icons-material/FolderShared"; | ||||||
| import ViewListIcon from "@mui/icons-material/ViewList"; | import ViewListIcon from "@mui/icons-material/ViewList"; | ||||||
|  | import { blue } from "@mui/material/colors"; | ||||||
| import { | import { | ||||||
|   Button, |   Button, | ||||||
|   Datagrid, |   Datagrid, | ||||||
| @@ -27,16 +29,14 @@ 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 { lighten, useTheme } from '@mui/material'; | ||||||
|  |  | ||||||
| const DestinationPagination = () => <Pagination rowsPerPageOptions={[10, 25, 50, 100, 500, 1000]} />; | const DestinationPagination = () => <Pagination rowsPerPageOptions={[10, 25, 50, 100, 500, 1000]} />; | ||||||
|  |  | ||||||
| const destinationRowSx = (record: RaRecord) => ({ |  | ||||||
|   backgroundColor: record.retry_last_ts > 0 ? "#ffcccc" : "white", |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| const destinationFilters = [<SearchInput source="destination" alwaysOn />]; | const destinationFilters = [<SearchInput source="destination" alwaysOn />]; | ||||||
|  |  | ||||||
| export const DestinationReconnectButton = () => { | export const DestinationReconnectButton = () => { | ||||||
| @@ -92,7 +92,25 @@ 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) => { | ||||||
|  |   const { palette: { error, mode }, } = useTheme(); | ||||||
|  |   const destinationRowSx = (record: RaRecord) => ({ | ||||||
|  |     backgroundColor: record.retry_last_ts > 0 ? lighten(error[mode], 0.5) : undefined, | ||||||
|  |     "& > td": mode === 'dark' ? { | ||||||
|  |       color: record.retry_last_ts > 0 ? "black" : "white", | ||||||
|  |       "& > button": { | ||||||
|  |         color: blue[700], | ||||||
|  |       }, | ||||||
|  |    } : undefined, | ||||||
|  |   }); | ||||||
|   return ( |   return ( | ||||||
|     <List |     <List | ||||||
|       {...props} |       {...props} | ||||||
| @@ -103,7 +121,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; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   const handleSend = () => { |   const handleSend = () => { | ||||||
|     create( |     create( | ||||||
|       "room_directory", |       "room_directory", | ||||||
|   | |||||||
| @@ -91,6 +91,7 @@ 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 UserBulkActionButtons = () => ( | ||||||
| @@ -108,7 +109,7 @@ 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 />} | ||||||
|   | |||||||
| @@ -30,7 +30,17 @@ 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: JSON.stringify({ | ||||||
|  |           device_id: null, | ||||||
|  |           initial_device_display_name: "Synapse Admin", | ||||||
|  |           type: "m.login.password", | ||||||
|  |           user: "@user:example.com", | ||||||
|  |           password: "secret", | ||||||
|  |           identifier: { | ||||||
|  |             type: "m.id.user", | ||||||
|  |             user: "@user:example.com", | ||||||
|  |           } | ||||||
|  |         }), | ||||||
|         headers: new Headers({ |         headers: new Headers({ | ||||||
|           Accept: "application/json", |           Accept: "application/json", | ||||||
|           "Content-Type": "application/json", |           "Content-Type": "application/json", | ||||||
|   | |||||||
| @@ -33,6 +33,10 @@ const authProvider: AuthProvider = { | |||||||
|                 type: "m.login.password", |                 type: "m.login.password", | ||||||
|                 user: username, |                 user: username, | ||||||
|                 password: password, |                 password: password, | ||||||
|  |                 identifier: { | ||||||
|  |                   type: "m.id.user", | ||||||
|  |                   user: username, | ||||||
|  |                 }, | ||||||
|               } |               } | ||||||
|         ) |         ) | ||||||
|       ), |       ), | ||||||
|   | |||||||
| @@ -1,7 +1,14 @@ | |||||||
| import { stringify } from "query-string"; | import { stringify } from "query-string"; | ||||||
|  | import { | ||||||
| import { DataProvider, DeleteParams, Identifier, Options, RaRecord, fetchUtils } from "react-admin"; |   DataProvider, | ||||||
|  |   DeleteParams, | ||||||
|  |   Identifier, | ||||||
|  |   Options, | ||||||
|  |   PaginationPayload, | ||||||
|  |   RaRecord, | ||||||
|  |   SortPayload, | ||||||
|  |   fetchUtils | ||||||
|  | } from "react-admin"; | ||||||
| import storage from "../storage"; | import storage from "../storage"; | ||||||
|  |  | ||||||
| // Adds the access token to all requests | // Adds the access token to all requests | ||||||
| @@ -491,9 +498,9 @@ function getSearchOrder(order: "ASC" | "DESC") { | |||||||
| const dataProvider: SynapseDataProvider = { | const dataProvider: 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 +511,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), | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user