summaryrefslogtreecommitdiff
path: root/web/source/settings/admin/federation/import-export.js
diff options
context:
space:
mode:
authorLibravatar f0x52 <f0x@cthu.lu>2023-01-18 14:45:14 +0100
committerLibravatar GitHub <noreply@github.com>2023-01-18 14:45:14 +0100
commit9b139b632098e6741b10fa87ff6224dcb5045947 (patch)
treec72b5c666ed01db7d1a18e531e5e01e07f504a46 /web/source/settings/admin/federation/import-export.js
parent[chore] Change default sqlite busy timeout to 5m (#1352) (diff)
downloadgotosocial-9b139b632098e6741b10fa87ff6224dcb5045947.tar.xz
[frogend] Settings refactor (#1318)
* yakshave new form field structure * fully refactor user profile settings form * use rtk query api for profile settings * refactor user post settings * refactor password change form * refactor admin settings * FormWithData structure for user forms * admin actions refactor * whitespace * fix user settings data prop * remove superfluous logging * cleanup old code * refactor federation/suspend (overview, detail) * mostly abstracted (emoji) checkbox list * refactor parse-from-toot * refactor custom-emoji, progress on federation bulk * loading icon styling to prevent big spinny * refactor federation import-export interface * cleanup old files * [chore] Update/add license headers for 2023 * redux fixes * text-field exports * appease the linter * refactor authentication with RTK Query * fix login/logout state transition weirdness * fixes/cleanup * small linter-related fixes * add eslint license header check, fix existing files * remove old code, clarify comment * clarify suspend on subdomains * collapse if/else * fa-fw width info comment
Diffstat (limited to 'web/source/settings/admin/federation/import-export.js')
-rw-r--r--web/source/settings/admin/federation/import-export.js307
1 files changed, 307 insertions, 0 deletions
diff --git a/web/source/settings/admin/federation/import-export.js b/web/source/settings/admin/federation/import-export.js
new file mode 100644
index 000000000..a11174b1f
--- /dev/null
+++ b/web/source/settings/admin/federation/import-export.js
@@ -0,0 +1,307 @@
+/*
+ GoToSocial
+ Copyright (C) 2021-2023 GoToSocial Authors admin@gotosocial.org
+
+ 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/>.
+*/
+
+"use strict";
+
+const React = require("react");
+const { Switch, Route, Redirect, useLocation } = require("wouter");
+
+const query = require("../../lib/query");
+
+const {
+ useTextInput,
+ useBoolInput,
+ useRadioInput,
+ useCheckListInput
+} = require("../../lib/form");
+
+const useFormSubmit = require("../../lib/form/submit");
+
+const {
+ TextInput,
+ TextArea,
+ Checkbox,
+ Select,
+ RadioGroup
+} = require("../../components/form/inputs");
+
+const CheckList = require("../../components/check-list");
+const MutationButton = require("../../components/form/mutation-button");
+const isValidDomain = require("is-valid-domain");
+const FormWithData = require("../../lib/form/form-with-data");
+const { Error } = require("../../components/error");
+
+const baseUrl = "/settings/admin/federation/import-export";
+
+module.exports = function ImportExport() {
+ const [updateFromFile, setUpdateFromFile] = React.useState(false);
+ const form = {
+ domains: useTextInput("domains"),
+ exportType: useTextInput("exportType", { defaultValue: "plain", dontReset: true })
+ };
+
+ const [submitParse, parseResult] = useFormSubmit(form, query.useProcessDomainListMutation());
+ const [submitExport, exportResult] = useFormSubmit(form, query.useExportDomainListMutation());
+
+ function fileChanged(e) {
+ const reader = new FileReader();
+ reader.onload = function (read) {
+ form.domains.setter(read.target.result);
+ setUpdateFromFile(true);
+ };
+ reader.readAsText(e.target.files[0]);
+ }
+
+ React.useEffect(() => {
+ if (exportResult.isSuccess) {
+ form.domains.setter(exportResult.data);
+ }
+ /* eslint-disable-next-line react-hooks/exhaustive-deps */
+ }, [exportResult]);
+
+ const [_location, setLocation] = useLocation();
+
+ if (updateFromFile) {
+ setUpdateFromFile(false);
+ submitParse();
+ }
+
+ return (
+ <Switch>
+ <Route path={`${baseUrl}/list`}>
+ {!parseResult.isSuccess && <Redirect to={baseUrl} />}
+
+ <h1>
+ <span className="button" onClick={() => {
+ parseResult.reset();
+ setLocation(baseUrl);
+ }}>
+ &lt; back
+ </span> Confirm import:
+ </h1>
+ <FormWithData
+ dataQuery={query.useInstanceBlocksQuery}
+ DataForm={ImportList}
+ list={parseResult.data}
+ />
+ </Route>
+
+ <Route>
+ {parseResult.isSuccess && <Redirect to={`${baseUrl}/list`} />}
+ <h2>Import / Export suspended domains</h2>
+
+ <div>
+ <form onSubmit={submitParse}>
+ <TextArea
+ field={form.domains}
+ label="Domains, one per line (plaintext) or JSON"
+ placeholder={`google.com\nfacebook.com`}
+ rows={8}
+ />
+
+ <div className="row">
+ <MutationButton label="Import" result={parseResult} showError={false} />
+ <button type="button" className="with-padding">
+ <label>
+ Import file
+ <input className="hidden" type="file" onChange={fileChanged} accept="application/json,text/plain" />
+ </label>
+ </button>
+ </div>
+ </form>
+ <form onSubmit={submitExport}>
+ <div className="row">
+ <MutationButton name="export" label="Export" result={exportResult} showError={false} />
+ <MutationButton name="export-file" label="Export file" result={exportResult} showError={false} />
+ <Select
+ field={form.exportType}
+ options={<>
+ <option value="plain">Text</option>
+ <option value="json">JSON</option>
+ </>}
+ />
+ </div>
+ </form>
+ {parseResult.error && <Error error={parseResult.error} />}
+ {exportResult.error && <Error error={exportResult.error} />}
+ </div>
+ </Route>
+ </Switch>
+ );
+};
+
+function ImportList({ list, data: blockedInstances }) {
+ const hasComment = React.useMemo(() => {
+ let hasPublic = false;
+ let hasPrivate = false;
+
+ list.some((entry) => {
+ if (entry.public_comment?.length > 0) {
+ hasPublic = true;
+ }
+
+ if (entry.private_comment?.length > 0) {
+ hasPrivate = true;
+ }
+
+ return hasPublic && hasPrivate;
+ });
+
+ if (hasPublic && hasPrivate) {
+ return { both: true };
+ } else if (hasPublic) {
+ return { type: "public_comment" };
+ } else if (hasPrivate) {
+ return { type: "private_comment" };
+ } else {
+ return {};
+ }
+ }, [list]);
+
+ const showComment = useTextInput("showComment", { defaultValue: hasComment.type ?? "public_comment" });
+ let commentName = "";
+ if (showComment.value == "public_comment") { commentName = "Public comment"; }
+ if (showComment.value == "private_comment") { commentName = "Private comment"; }
+
+ const form = {
+ domains: useCheckListInput("domains", {
+ entries: list,
+ uniqueKey: "domain"
+ }),
+ obfuscate: useBoolInput("obfuscate"),
+ privateComment: useTextInput("private_comment", {
+ defaultValue: `Imported on ${new Date().toLocaleString()}`
+ }),
+ privateCommentBehavior: useRadioInput("private_comment_behavior", {
+ defaultValue: "append",
+ options: {
+ append: "Append to",
+ replace: "Replace"
+ }
+ }),
+ publicComment: useTextInput("public_comment"),
+ publicCommentBehavior: useRadioInput("public_comment_behavior", {
+ defaultValue: "append",
+ options: {
+ append: "Append to",
+ replace: "Replace"
+ }
+ }),
+ };
+
+ const [importDomains, importResult] = useFormSubmit(form, query.useImportDomainListMutation(), { changedOnly: false });
+
+ return (
+ <>
+ <form onSubmit={importDomains} className="suspend-import-list">
+ <span>{list.length} domain{list.length != 1 ? "s" : ""} in this list</span>
+
+ {hasComment.both &&
+ <Select field={showComment} options={
+ <>
+ <option value="public_comment">Show public comments</option>
+ <option value="private_comment">Show private comments</option>
+ </>
+ } />
+ }
+
+ <CheckList
+ field={form.domains}
+ Component={DomainEntry}
+ header={
+ <>
+ <b>Domain</b>
+ <b></b>
+ <b>{commentName}</b>
+ </>
+ }
+ blockedInstances={blockedInstances}
+ commentType={showComment.value}
+ />
+
+ <TextArea
+ field={form.privateComment}
+ label="Private comment"
+ rows={3}
+ />
+ <RadioGroup
+ field={form.privateCommentBehavior}
+ label="imported private comment"
+ />
+
+ <TextArea
+ field={form.publicComment}
+ label="Public comment"
+ rows={3}
+ />
+ <RadioGroup
+ field={form.publicCommentBehavior}
+ label="imported public comment"
+ />
+
+ <Checkbox
+ field={form.obfuscate}
+ label="Obfuscate domains in public lists"
+ />
+
+ <MutationButton label="Import" result={importResult} />
+ </form>
+ </>
+ );
+}
+
+function DomainEntry({ entry, onChange, blockedInstances, commentType }) {
+ const domainField = useTextInput("domain", {
+ defaultValue: entry.domain,
+ validator: (value) => {
+ return (entry.checked && !isValidDomain(value, { wildcard: true, allowUnicode: true }))
+ ? "Invalid domain"
+ : "";
+ }
+ });
+
+ React.useEffect(() => {
+ onChange({ valid: domainField.valid });
+ /* eslint-disable-next-line react-hooks/exhaustive-deps */
+ }, [domainField.valid]);
+
+ let icon = null;
+
+ if (blockedInstances[domainField.value] != undefined) {
+ icon = (
+ <>
+ <i className="fa fa-history already-blocked" aria-hidden="true" title="Domain block already exists"></i>
+ <span className="sr-only">Domain block already exists.</span>
+ </>
+ );
+ }
+
+ return (
+ <>
+ <TextInput
+ field={domainField}
+ onChange={(e) => {
+ domainField.onChange(e);
+ onChange({ domain: e.target.value, checked: true });
+ }}
+ />
+ <span id="icon">{icon}</span>
+ <p>{entry[commentType]}</p>
+ </>
+ );
+} \ No newline at end of file