diff options
Diffstat (limited to 'web/source/settings/lib')
| -rw-r--r-- | web/source/settings/lib/query/gts-api.ts | 28 | ||||
| -rw-r--r-- | web/source/settings/lib/query/user/export-import.ts | 138 | ||||
| -rw-r--r-- | web/source/settings/lib/types/account.ts | 10 | 
3 files changed, 175 insertions, 1 deletions
diff --git a/web/source/settings/lib/query/gts-api.ts b/web/source/settings/lib/query/gts-api.ts index d6741df3a..1c715e284 100644 --- a/web/source/settings/lib/query/gts-api.ts +++ b/web/source/settings/lib/query/gts-api.ts @@ -48,6 +48,11 @@ export interface GTSFetchArgs extends FetchArgs {  	 * as FormData before submission.   	 */  	asForm?: boolean; +	/** +	 * If set, then Accept header will +	 * be set to the provided contentType. +	 */ +	acceptContentType?: string;  }  /** @@ -77,6 +82,10 @@ const gtsBaseQuery: BaseQueryFn<  	// Derive baseUrl dynamically.  	let baseUrl: string | undefined; +	// Assume Accept value of +	// "application/json" by default. +	let accept = "application/json"; +  	// Check if simple string baseUrl provided  	// as args, or if more complex args provided.  	if (typeof args === "string") { @@ -101,11 +110,16 @@ const gtsBaseQuery: BaseQueryFn<  			});  		} +		if (args.acceptContentType !== undefined) { +			accept = args.acceptContentType; +		} +  		// Delete any of our extended arguments  		// to avoid confusing fetchBaseQuery.  		delete args.baseUrl;  		delete args.discardEmpty;  		delete args.asForm; +		delete args.acceptContentType;  	}  	if (!baseUrl) { @@ -124,9 +138,21 @@ const gtsBaseQuery: BaseQueryFn<  			if (token != undefined) {  				headers.set('Authorization', token);  			} -			headers.set("Accept", "application/json"); +			 +			headers.set("Accept", accept);  			return headers;  		}, +		responseHandler: (response) => { +			// Return just text if caller has +			// set a custom accept content-type. +			if (accept !== "application/json") { +				return response.text(); +			} + +			// Else return good old +			// fashioned JSON baby! +			return response.json(); +		},  	})(args, api, extraOptions);  }; diff --git a/web/source/settings/lib/query/user/export-import.ts b/web/source/settings/lib/query/user/export-import.ts new file mode 100644 index 000000000..56c48e364 --- /dev/null +++ b/web/source/settings/lib/query/user/export-import.ts @@ -0,0 +1,138 @@ +/* +	GoToSocial +	Copyright (C) GoToSocial Authors admin@gotosocial.org +	SPDX-License-Identifier: AGPL-3.0-or-later + +	This program is free software: you can redistribute it and/or modify +	it under the terms of the GNU Affero General Public License as published by +	the Free Software Foundation, either version 3 of the License, or +	(at your option) any later version. + +	This program is distributed in the hope that it will be useful, +	but WITHOUT ANY WARRANTY; without even the implied warranty of +	MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the +	GNU Affero General Public License for more details. + +	You should have received a copy of the GNU Affero General Public License +	along with this program.  If not, see <http://www.gnu.org/licenses/>. +*/ + +import fileDownload from "js-file-download"; + +import { gtsApi } from "../gts-api"; +import { FetchBaseQueryError } from "@reduxjs/toolkit/query"; +import { AccountExportStats } from "../../types/account"; + +const extended = gtsApi.injectEndpoints({ +	endpoints: (build) => ({ +		exportStats: build.query<AccountExportStats, void>({ +			query: () => ({ +				url: `/api/v1/exports/stats` +			}) +		}), +		 +		exportFollowing: build.mutation<string | null, void>({ +			async queryFn(_arg, _api, _extraOpts, fetchWithBQ) { +				const csvRes = await fetchWithBQ({ +					url: `/api/v1/exports/following.csv`, +					acceptContentType: "text/csv", +				}); +				if (csvRes.error) { +					return { error: csvRes.error as FetchBaseQueryError }; +				} + +				if (csvRes.meta?.response?.status !== 200) { +					return { error: csvRes.data }; +				} + +				fileDownload(csvRes.data, "following.csv", "text/csv"); +				return { data: null }; +			} +		}), + +		exportFollowers: build.mutation<string | null, void>({ +			async queryFn(_arg, _api, _extraOpts, fetchWithBQ) { +				const csvRes = await fetchWithBQ({ +					url: `/api/v1/exports/followers.csv`, +					acceptContentType: "text/csv", +				}); +				if (csvRes.error) { +					return { error: csvRes.error as FetchBaseQueryError }; +				} + +				if (csvRes.meta?.response?.status !== 200) { +					return { error: csvRes.data }; +				} + +				fileDownload(csvRes.data, "followers.csv", "text/csv"); +				return { data: null }; +			} +		}), + +		exportLists: build.mutation<string | null, void>({ +			async queryFn(_arg, _api, _extraOpts, fetchWithBQ) { +				const csvRes = await fetchWithBQ({ +					url: `/api/v1/exports/lists.csv`, +					acceptContentType: "text/csv", +				}); +				if (csvRes.error) { +					return { error: csvRes.error as FetchBaseQueryError }; +				} + +				if (csvRes.meta?.response?.status !== 200) { +					return { error: csvRes.data }; +				} + +				fileDownload(csvRes.data, "lists.csv", "text/csv"); +				return { data: null }; +			} +		}), + +		exportBlocks: build.mutation<string | null, void>({ +			async queryFn(_arg, _api, _extraOpts, fetchWithBQ) { +				const csvRes = await fetchWithBQ({ +					url: `/api/v1/exports/blocks.csv`, +					acceptContentType: "text/csv", +				}); +				if (csvRes.error) { +					return { error: csvRes.error as FetchBaseQueryError }; +				} + +				if (csvRes.meta?.response?.status !== 200) { +					return { error: csvRes.data }; +				} + +				fileDownload(csvRes.data, "blocks.csv", "text/csv"); +				return { data: null }; +			} +		}), + +		exportMutes: build.mutation<string | null, void>({ +			async queryFn(_arg, _api, _extraOpts, fetchWithBQ) { +				const csvRes = await fetchWithBQ({ +					url: `/api/v1/exports/mutes.csv`, +					acceptContentType: "text/csv", +				}); +				if (csvRes.error) { +					return { error: csvRes.error as FetchBaseQueryError }; +				} + +				if (csvRes.meta?.response?.status !== 200) { +					return { error: csvRes.data }; +				} + +				fileDownload(csvRes.data, "mutes.csv", "text/csv"); +				return { data: null }; +			} +		}), +	}) +}); + +export const { +	useExportStatsQuery, +	useExportFollowingMutation, +	useExportFollowersMutation, +	useExportListsMutation, +	useExportBlocksMutation, +	useExportMutesMutation, +} = extended; diff --git a/web/source/settings/lib/types/account.ts b/web/source/settings/lib/types/account.ts index 590b2c98e..6b8d2bc4d 100644 --- a/web/source/settings/lib/types/account.ts +++ b/web/source/settings/lib/types/account.ts @@ -110,3 +110,13 @@ export interface ActionAccountParams {  	action: "suspend";  	reason: string;  } + +export interface AccountExportStats { +	media_storage: string; +	followers_count: number; +	following_count: number; +	statuses_count: number; +	lists_count: number; +	blocks_count: number; +	mutes_count: number; +}  | 
