mirror of
				https://github.com/superseriousbusiness/gotosocial.git
				synced 2025-11-02 16:52:25 -06:00 
			
		
		
		
	* [feature] Add domain permission drafts and excludes * fix typescript complaining * lint * make filenames more consistent * test own domain excluded
		
			
				
	
	
		
			212 lines
		
	
	
	
		
			6 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			212 lines
		
	
	
	
		
			6 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
/*
 | 
						|
	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 React, { useMemo } from "react";
 | 
						|
import { useLocation, useParams } from "wouter";
 | 
						|
import { PermType } from "../../../lib/types/perm";
 | 
						|
import { useDeleteHeaderAllowMutation, useDeleteHeaderBlockMutation, useGetHeaderAllowQuery, useGetHeaderBlockQuery } from "../../../lib/query/admin/http-header-permissions";
 | 
						|
import { HeaderPermission } from "../../../lib/types/http-header-permissions";
 | 
						|
import { FetchBaseQueryError } from "@reduxjs/toolkit/query";
 | 
						|
import { SerializedError } from "@reduxjs/toolkit";
 | 
						|
import Loading from "../../../components/loading";
 | 
						|
import { Error } from "../../../components/error";
 | 
						|
import UsernameLozenge from "../../../components/username-lozenge";
 | 
						|
import { useBaseUrl } from "../../../lib/navigation/util";
 | 
						|
import BackButton from "../../../components/back-button";
 | 
						|
import MutationButton from "../../../components/form/mutation-button";
 | 
						|
 | 
						|
const testString = `/* To test this properly, set "flavor" to "Golang", as that's the language GoToSocial uses for regular expressions */
 | 
						|
 | 
						|
/* Amazon crawler User-Agent example */
 | 
						|
Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10_1) AppleWebKit/600.2.5 (KHTML\\, like Gecko) Version/8.0.2 Safari/600.2.5 (Amazonbot/0.1; +https://developer.amazon.com/support/amazonbot)
 | 
						|
 | 
						|
/* Some other test strings */
 | 
						|
Some Test Value
 | 
						|
Another Test Value`;
 | 
						|
 | 
						|
export default function HeaderPermDetail() {
 | 
						|
	let params = useParams();
 | 
						|
	if (params.permType !== "blocks" && params.permType !== "allows") {
 | 
						|
		throw "unrecognized perm type " + params.permType;
 | 
						|
	}
 | 
						|
	const permType = useMemo(() => {
 | 
						|
		return params.permType?.slice(0, -1) as PermType;
 | 
						|
	}, [params]);
 | 
						|
 | 
						|
	let permID = params.permId as string | undefined;
 | 
						|
	if (!permID) {
 | 
						|
		throw "no perm ID";
 | 
						|
	}
 | 
						|
 | 
						|
	if (permType === "block") {
 | 
						|
		return <BlockDetail id={permID} />;
 | 
						|
	} else {
 | 
						|
		return <AllowDetail id={permID} />;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
function BlockDetail({ id }: { id: string }) {
 | 
						|
	return (
 | 
						|
		<PermDeets
 | 
						|
			permType={"Block"}
 | 
						|
			{...useGetHeaderBlockQuery(id)}
 | 
						|
		/>
 | 
						|
	);
 | 
						|
}
 | 
						|
 | 
						|
function AllowDetail({ id }: { id: string }) {
 | 
						|
	return (
 | 
						|
		<PermDeets
 | 
						|
			permType={"Allow"}
 | 
						|
			{...useGetHeaderAllowQuery(id)}
 | 
						|
		/>
 | 
						|
	);
 | 
						|
}
 | 
						|
 | 
						|
interface PermDeetsProps {
 | 
						|
	permType: string;
 | 
						|
	data?: HeaderPermission;
 | 
						|
	isLoading: boolean;
 | 
						|
	isFetching: boolean;
 | 
						|
	isError: boolean;
 | 
						|
	error?: FetchBaseQueryError | SerializedError;
 | 
						|
}
 | 
						|
 | 
						|
function PermDeets({
 | 
						|
	permType,
 | 
						|
	data: perm,
 | 
						|
	isLoading,
 | 
						|
	isFetching,
 | 
						|
	isError,
 | 
						|
	error,
 | 
						|
}: PermDeetsProps) {
 | 
						|
	const [ location ] = useLocation();
 | 
						|
	const baseUrl = useBaseUrl();
 | 
						|
 | 
						|
	// Wait til the perm itself is loaded.
 | 
						|
	if (isLoading || isFetching) {
 | 
						|
		return <Loading />;
 | 
						|
	} else if (isError) {
 | 
						|
		return <Error error={error} />;
 | 
						|
	} else if (perm === undefined) {
 | 
						|
		throw "perm undefined";
 | 
						|
	}
 | 
						|
 | 
						|
	const created = new Date(perm.created_at).toDateString();	
 | 
						|
	
 | 
						|
	// Create parameters to link to regex101
 | 
						|
	// with this regular expression prepopulated.
 | 
						|
	const testParams = new URLSearchParams();
 | 
						|
	testParams.set("regex", perm.regex);
 | 
						|
	testParams.set("flags", "gm");
 | 
						|
	testParams.set("testString", testString);
 | 
						|
	const regexLink = `https://regex101.com/?${testParams.toString()}`;	
 | 
						|
 | 
						|
	return (
 | 
						|
		<div className="http-header-permission-details">
 | 
						|
			<h1><BackButton to={`~${baseUrl}/${permType.toLowerCase()}s`} /> HTTP Header {permType} Detail</h1>
 | 
						|
			<dl className="info-list">
 | 
						|
				<div className="info-list-entry">
 | 
						|
					<dt>ID</dt>
 | 
						|
					<dd className="monospace">{perm.id}</dd>
 | 
						|
				</div>
 | 
						|
				<div className="info-list-entry">
 | 
						|
					<dt>Created</dt>
 | 
						|
					<dd><time dateTime={perm.created_at}>{created}</time></dd>
 | 
						|
				</div>
 | 
						|
				<div className="info-list-entry">
 | 
						|
					<dt>Created By</dt>
 | 
						|
					<dd>
 | 
						|
						<UsernameLozenge
 | 
						|
							account={perm.created_by}
 | 
						|
							linkTo={`~/settings/moderation/accounts/${perm.created_by}`}
 | 
						|
							backLocation={`~${baseUrl}${location}`}
 | 
						|
						/>
 | 
						|
					</dd>
 | 
						|
				</div>
 | 
						|
				<div className="info-list-entry">
 | 
						|
					<dt>Header Name</dt>
 | 
						|
					<dd className="monospace">{perm.header}</dd>
 | 
						|
				</div>
 | 
						|
				<div className="info-list-entry">
 | 
						|
					<dt>Value Regex</dt>
 | 
						|
					<dd className="monospace">{perm.regex}</dd>
 | 
						|
				</div>
 | 
						|
				<div className="info-list-entry">
 | 
						|
					<dt>Test This Regex</dt>
 | 
						|
					<dd>
 | 
						|
						<a
 | 
						|
							href={regexLink}
 | 
						|
							target="_blank"
 | 
						|
							rel="noreferrer"
 | 
						|
						>
 | 
						|
							<i className="fa fa-fw fa-external-link" aria-hidden="true"></i> Link to Regex101 (opens in a new tab)
 | 
						|
						</a>
 | 
						|
					</dd>
 | 
						|
				</div>
 | 
						|
			</dl>
 | 
						|
			{ permType === "Block"
 | 
						|
				? <DeleteBlock id={perm.id} />
 | 
						|
				: <DeleteAllow id={perm.id} />
 | 
						|
			}
 | 
						|
		</div>
 | 
						|
	);
 | 
						|
}
 | 
						|
 | 
						|
function DeleteBlock({ id }: { id: string }) {
 | 
						|
	const [ _location, setLocation ] = useLocation();
 | 
						|
	const baseUrl = useBaseUrl();
 | 
						|
	const [ removeTrigger, removeResult ] = useDeleteHeaderBlockMutation();
 | 
						|
	
 | 
						|
	return (
 | 
						|
		<MutationButton
 | 
						|
			type="button"
 | 
						|
			onClick={() => {
 | 
						|
				removeTrigger(id);
 | 
						|
				setLocation(`~${baseUrl}/blocks`);
 | 
						|
			}}
 | 
						|
			label="Remove this block"
 | 
						|
			result={removeResult}
 | 
						|
			className="button danger"
 | 
						|
			showError={false}
 | 
						|
			disabled={false}
 | 
						|
		/>
 | 
						|
	);
 | 
						|
}
 | 
						|
 | 
						|
function DeleteAllow({ id }: { id: string }) {
 | 
						|
	const [ _location, setLocation ] = useLocation();
 | 
						|
	const baseUrl = useBaseUrl();
 | 
						|
	const [ removeTrigger, removeResult ] = useDeleteHeaderAllowMutation();
 | 
						|
	
 | 
						|
	return (
 | 
						|
		<MutationButton
 | 
						|
			type="button"
 | 
						|
			onClick={() => {
 | 
						|
				removeTrigger(id);
 | 
						|
				setLocation(`~${baseUrl}/allows`);
 | 
						|
			}}
 | 
						|
			label="Remove this allow"
 | 
						|
			result={removeResult}
 | 
						|
			className="button danger"
 | 
						|
			showError={false}
 | 
						|
			disabled={false}
 | 
						|
		/>
 | 
						|
	);
 | 
						|
}
 |