2023-10-17 10:46:06 +00:00
|
|
|
/*
|
|
|
|
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 { replaceCacheOnMutation, removeFromCacheOnMutation } from "../query-modifiers";
|
|
|
|
import { gtsApi } from "../gts-api";
|
|
|
|
import { listToKeyedObject } from "../transforms";
|
2024-05-02 15:57:53 +00:00
|
|
|
import { ActionAccountParams, AdminAccount, HandleSignupParams, SearchAccountParams, SearchAccountResp } from "../../types/account";
|
2024-04-24 10:12:47 +00:00
|
|
|
import { InstanceRule, MappedRules } from "../../types/rules";
|
2024-05-01 13:11:22 +00:00
|
|
|
import parse from "parse-link-header";
|
2023-10-17 10:46:06 +00:00
|
|
|
|
|
|
|
const extended = gtsApi.injectEndpoints({
|
|
|
|
endpoints: (build) => ({
|
|
|
|
updateInstance: build.mutation({
|
|
|
|
query: (formData) => ({
|
|
|
|
method: "PATCH",
|
|
|
|
url: `/api/v1/instance`,
|
|
|
|
asForm: true,
|
|
|
|
body: formData,
|
|
|
|
discardEmpty: true
|
|
|
|
}),
|
|
|
|
...replaceCacheOnMutation("instanceV1"),
|
|
|
|
}),
|
|
|
|
|
2024-04-13 11:25:10 +00:00
|
|
|
getAccount: build.query<AdminAccount, string>({
|
2023-10-17 10:46:06 +00:00
|
|
|
query: (id) => ({
|
2024-04-13 11:25:10 +00:00
|
|
|
url: `/api/v1/admin/accounts/${id}`
|
2023-10-17 10:46:06 +00:00
|
|
|
}),
|
2024-04-13 11:25:10 +00:00
|
|
|
providesTags: (_result, _error, id) => [
|
|
|
|
{ type: 'Account', id }
|
|
|
|
],
|
2023-10-17 10:46:06 +00:00
|
|
|
}),
|
|
|
|
|
2024-05-01 13:11:22 +00:00
|
|
|
searchAccounts: build.query<SearchAccountResp, SearchAccountParams>({
|
2024-04-13 11:25:10 +00:00
|
|
|
query: (form) => {
|
|
|
|
const params = new(URLSearchParams);
|
|
|
|
Object.entries(form).forEach(([k, v]) => {
|
|
|
|
if (v !== undefined) {
|
|
|
|
params.append(k, v);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
let query = "";
|
|
|
|
if (params.size !== 0) {
|
|
|
|
query = `?${params.toString()}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
url: `/api/v2/admin/accounts${query}`
|
|
|
|
};
|
|
|
|
},
|
2024-05-02 15:57:53 +00:00
|
|
|
// Headers required for paging.
|
2024-05-01 13:11:22 +00:00
|
|
|
transformResponse: (apiResp: AdminAccount[], meta) => {
|
|
|
|
const accounts = apiResp;
|
|
|
|
const linksStr = meta?.response?.headers.get("Link");
|
|
|
|
const links = parse(linksStr);
|
|
|
|
return { accounts, links };
|
|
|
|
},
|
2024-05-02 15:57:53 +00:00
|
|
|
// Only provide LIST tag id since this model is not the
|
|
|
|
// same as getAccount model (due to transformResponse).
|
|
|
|
providesTags: [{ type: "Account", id: "TRANSFORMED" }]
|
2024-04-13 11:25:10 +00:00
|
|
|
}),
|
|
|
|
|
2024-05-02 15:57:53 +00:00
|
|
|
actionAccount: build.mutation<string, ActionAccountParams>({
|
2023-10-17 10:46:06 +00:00
|
|
|
query: ({ id, action, reason }) => ({
|
|
|
|
method: "POST",
|
|
|
|
url: `/api/v1/admin/accounts/${id}/action`,
|
|
|
|
asForm: true,
|
|
|
|
body: {
|
|
|
|
type: action,
|
|
|
|
text: reason
|
|
|
|
}
|
|
|
|
}),
|
2024-05-02 15:57:53 +00:00
|
|
|
// Do an optimistic update on this account to mark
|
|
|
|
// it according to whatever action was submitted.
|
|
|
|
async onQueryStarted({ id, action }, { dispatch, queryFulfilled }) {
|
|
|
|
const patchResult = dispatch(
|
|
|
|
extended.util.updateQueryData("getAccount", id, (draft) => {
|
|
|
|
if (action === "suspend") {
|
|
|
|
draft.suspended = true;
|
|
|
|
draft.account.suspended = true;
|
|
|
|
}
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
|
|
|
// Revert optimistic
|
|
|
|
// update if query fails.
|
|
|
|
try {
|
|
|
|
await queryFulfilled;
|
|
|
|
} catch {
|
|
|
|
patchResult.undo();
|
|
|
|
}
|
|
|
|
}
|
2023-10-17 10:46:06 +00:00
|
|
|
}),
|
|
|
|
|
2024-04-13 11:25:10 +00:00
|
|
|
handleSignup: build.mutation<AdminAccount, HandleSignupParams>({
|
|
|
|
query: ({id, approve_or_reject, ...formData}) => {
|
|
|
|
return {
|
|
|
|
method: "POST",
|
|
|
|
url: `/api/v1/admin/accounts/${id}/${approve_or_reject}`,
|
|
|
|
asForm: true,
|
|
|
|
body: approve_or_reject === "reject" ?? formData,
|
|
|
|
};
|
|
|
|
},
|
2024-05-02 15:57:53 +00:00
|
|
|
// Do an optimistic update on this account to mark it approved
|
|
|
|
// if approved was true, else just invalidate getAccount.
|
|
|
|
async onQueryStarted({ id, approve_or_reject }, { dispatch, queryFulfilled }) {
|
|
|
|
if (approve_or_reject === "reject") {
|
|
|
|
// Just invalidate this ID and getAccounts.
|
|
|
|
dispatch(extended.util.invalidateTags([
|
|
|
|
{ type: "Account", id: id },
|
|
|
|
{ type: "Account", id: "TRANSFORMED" }
|
|
|
|
]));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const patchResult = dispatch(
|
|
|
|
extended.util.updateQueryData("getAccount", id, (draft) => {
|
|
|
|
draft.approved = true;
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
|
|
|
// Revert optimistic
|
|
|
|
// update if query fails.
|
|
|
|
try {
|
|
|
|
await queryFulfilled;
|
|
|
|
} catch {
|
|
|
|
patchResult.undo();
|
|
|
|
}
|
|
|
|
}
|
2023-10-17 10:46:06 +00:00
|
|
|
}),
|
|
|
|
|
2024-04-24 10:12:47 +00:00
|
|
|
instanceRules: build.query<MappedRules, void>({
|
2023-10-17 10:46:06 +00:00
|
|
|
query: () => ({
|
|
|
|
url: `/api/v1/admin/instance/rules`
|
|
|
|
}),
|
2024-04-24 10:12:47 +00:00
|
|
|
transformResponse: listToKeyedObject<InstanceRule>("id")
|
2023-10-17 10:46:06 +00:00
|
|
|
}),
|
|
|
|
|
2024-04-24 10:12:47 +00:00
|
|
|
addInstanceRule: build.mutation<MappedRules, any>({
|
2023-10-17 10:46:06 +00:00
|
|
|
query: (formData) => ({
|
|
|
|
method: "POST",
|
|
|
|
url: `/api/v1/admin/instance/rules`,
|
|
|
|
asForm: true,
|
|
|
|
body: formData,
|
|
|
|
discardEmpty: true
|
|
|
|
}),
|
2024-04-24 10:12:47 +00:00
|
|
|
transformResponse: listToKeyedObject<InstanceRule>("id"),
|
2023-10-17 10:46:06 +00:00
|
|
|
...replaceCacheOnMutation("instanceRules"),
|
|
|
|
}),
|
|
|
|
|
|
|
|
updateInstanceRule: build.mutation({
|
|
|
|
query: ({ id, ...edit }) => ({
|
|
|
|
method: "PATCH",
|
|
|
|
url: `/api/v1/admin/instance/rules/${id}`,
|
|
|
|
asForm: true,
|
|
|
|
body: edit,
|
|
|
|
discardEmpty: true
|
|
|
|
}),
|
|
|
|
transformResponse: (data) => {
|
|
|
|
return {
|
|
|
|
[data.id]: data
|
|
|
|
};
|
|
|
|
},
|
|
|
|
...replaceCacheOnMutation("instanceRules"),
|
|
|
|
}),
|
|
|
|
|
|
|
|
deleteInstanceRule: build.mutation({
|
|
|
|
query: (id) => ({
|
|
|
|
method: "DELETE",
|
|
|
|
url: `/api/v1/admin/instance/rules/${id}`
|
|
|
|
}),
|
|
|
|
...removeFromCacheOnMutation("instanceRules", {
|
|
|
|
key: (_draft, rule) => rule.id,
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
});
|
|
|
|
|
|
|
|
export const {
|
|
|
|
useUpdateInstanceMutation,
|
|
|
|
useGetAccountQuery,
|
2024-05-05 11:47:22 +00:00
|
|
|
useLazyGetAccountQuery,
|
2023-10-17 10:46:06 +00:00
|
|
|
useActionAccountMutation,
|
2024-04-13 11:25:10 +00:00
|
|
|
useSearchAccountsQuery,
|
|
|
|
useLazySearchAccountsQuery,
|
|
|
|
useHandleSignupMutation,
|
2023-10-17 10:46:06 +00:00
|
|
|
useInstanceRulesQuery,
|
|
|
|
useAddInstanceRuleMutation,
|
|
|
|
useUpdateInstanceRuleMutation,
|
|
|
|
useDeleteInstanceRuleMutation,
|
|
|
|
} = extended;
|