summaryrefslogtreecommitdiff
path: root/web/source/settings/lib/query/oauth.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/lib/query/oauth.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/lib/query/oauth.js')
-rw-r--r--web/source/settings/lib/query/oauth.js158
1 files changed, 158 insertions, 0 deletions
diff --git a/web/source/settings/lib/query/oauth.js b/web/source/settings/lib/query/oauth.js
new file mode 100644
index 000000000..4fac50429
--- /dev/null
+++ b/web/source/settings/lib/query/oauth.js
@@ -0,0 +1,158 @@
+/*
+ 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 Promise = require("bluebird");
+
+const base = require("./base");
+const { unwrapRes } = require("./lib");
+const oauth = require("../../redux/oauth").actions;
+
+function getSettingsURL() {
+ /* needed in case the settings interface isn't hosted at /settings but
+ some subpath like /gotosocial/settings. Other parts of the code don't
+ take this into account yet so mostly future-proofing.
+
+ Also drops anything past /settings/, because authorization urls that are too long
+ get rejected by GTS.
+ */
+ let [pre, _past] = window.location.pathname.split("/settings");
+ return `${window.location.origin}${pre}/settings`;
+}
+
+const SETTINGS_URL = getSettingsURL();
+
+const endpoints = (build) => ({
+ verifyCredentials: build.query({
+ providesTags: (_res, error) =>
+ error == undefined
+ ? ["Auth"]
+ : [],
+ queryFn: (_arg, api, _extraOpts, baseQuery) => {
+ const state = api.getState();
+
+ return Promise.try(() => {
+ // Process callback code first, if available
+ if (state.oauth.loginState == "callback") {
+ let urlParams = new URLSearchParams(window.location.search);
+ let code = urlParams.get("code");
+
+ if (code == undefined) {
+ throw {
+ message: "Waiting for callback, but no ?code= provided in url."
+ };
+ } else {
+ let app = state.oauth.registration;
+
+ if (app == undefined || app.client_id == undefined) {
+ throw {
+ message: "No stored registration data, can't finish login flow."
+ };
+ }
+
+ return baseQuery({
+ method: "POST",
+ url: "/oauth/token",
+ body: {
+ client_id: app.client_id,
+ client_secret: app.client_secret,
+ redirect_uri: SETTINGS_URL,
+ grant_type: "authorization_code",
+ code: code
+ }
+ }).then(unwrapRes).then((token) => {
+ // remove ?code= from url
+ window.history.replaceState({}, document.title, window.location.pathname);
+ api.dispatch(oauth.setToken(token));
+ });
+ }
+ }
+ }).then(() => {
+ return baseQuery({
+ url: `/api/v1/accounts/verify_credentials`
+ });
+ }).catch((e) => {
+ return { error: e };
+ });
+ }
+ }),
+ authorizeFlow: build.mutation({
+ queryFn: (formData, api, _extraOpts, baseQuery) => {
+ let instance;
+ const state = api.getState();
+
+ return Promise.try(() => {
+ if (!formData.instance.startsWith("http")) {
+ formData.instance = `https://${formData.instance}`;
+ }
+ instance = new URL(formData.instance).origin;
+
+ const stored = state.oauth.instance;
+ if (stored?.instance == instance && stored.registration) {
+ return stored.registration;
+ }
+
+ return baseQuery({
+ method: "POST",
+ baseUrl: instance,
+ url: "/api/v1/apps",
+ body: {
+ client_name: "GoToSocial Settings",
+ scopes: formData.scopes,
+ redirect_uris: SETTINGS_URL,
+ website: SETTINGS_URL
+ }
+ }).then(unwrapRes).then((app) => {
+ app.scopes = formData.scopes;
+
+ api.dispatch(oauth.setInstance({
+ instance: instance,
+ registration: app,
+ loginState: "callback"
+ }));
+
+ return app;
+ });
+ }).then((app) => {
+ let url = new URL(instance);
+ url.pathname = "/oauth/authorize";
+ url.searchParams.set("client_id", app.client_id);
+ url.searchParams.set("redirect_uri", SETTINGS_URL);
+ url.searchParams.set("response_type", "code");
+ url.searchParams.set("scope", app.scopes);
+
+ let redirectURL = url.toString();
+ window.location.assign(redirectURL);
+
+ return { data: null };
+ }).catch((e) => {
+ return { error: e };
+ });
+ },
+ }),
+ logout: build.mutation({
+ queryFn: (_arg, api) => {
+ api.dispatch(oauth.remove());
+ return { data: null };
+ },
+ invalidatesTags: ["Auth"]
+ })
+});
+
+module.exports = base.injectEndpoints({ endpoints }); \ No newline at end of file