2021-08-22 12:34:58 -07:00
|
|
|
/**
|
|
|
|
* External Auth: workflow for logging in to remote servers.
|
|
|
|
* @module soapbox/actions/external_auth
|
|
|
|
* @see module:soapbox/actions/auth
|
|
|
|
* @see module:soapbox/actions/apps
|
|
|
|
* @see module:soapbox/actions/oauth
|
|
|
|
*/
|
|
|
|
|
2021-08-22 17:13:09 -07:00
|
|
|
import { baseClient } from '../api';
|
2021-08-22 12:34:58 -07:00
|
|
|
import { createApp } from 'soapbox/actions/apps';
|
|
|
|
import { obtainOAuthToken } from 'soapbox/actions/oauth';
|
2021-08-30 16:41:05 -07:00
|
|
|
import { authLoggedIn, verifyCredentials, switchAccount } from 'soapbox/actions/auth';
|
2021-08-22 12:34:58 -07:00
|
|
|
import { parseBaseURL } from 'soapbox/utils/auth';
|
2021-08-22 17:13:09 -07:00
|
|
|
import { getFeatures } from 'soapbox/utils/features';
|
2021-08-22 12:34:58 -07:00
|
|
|
import sourceCode from 'soapbox/utils/code';
|
2021-08-31 09:02:43 -07:00
|
|
|
import { Map as ImmutableMap, fromJS } from 'immutable';
|
2021-08-22 12:34:58 -07:00
|
|
|
|
2021-08-22 17:13:09 -07:00
|
|
|
const fetchExternalInstance = baseURL => {
|
|
|
|
return baseClient(null, baseURL)
|
|
|
|
.get('/api/v1/instance')
|
2021-08-31 09:02:43 -07:00
|
|
|
.then(({ data: instance }) => fromJS(instance))
|
|
|
|
.catch(error => {
|
|
|
|
if (error.response.status === 401) {
|
|
|
|
// Authenticated fetch is enabled.
|
|
|
|
// Continue with a limited featureset.
|
|
|
|
return ImmutableMap({ version: '0.0.0' });
|
|
|
|
} else {
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
});
|
2021-08-22 17:13:09 -07:00
|
|
|
};
|
2021-08-22 12:34:58 -07:00
|
|
|
|
|
|
|
export function createAppAndRedirect(host) {
|
|
|
|
return (dispatch, getState) => {
|
|
|
|
const baseURL = parseBaseURL(host) || parseBaseURL(`https://${host}`);
|
|
|
|
|
2021-08-22 17:13:09 -07:00
|
|
|
return fetchExternalInstance(baseURL).then(instance => {
|
|
|
|
const { scopes } = getFeatures(instance);
|
2021-08-22 12:34:58 -07:00
|
|
|
|
2021-08-22 17:13:09 -07:00
|
|
|
const params = {
|
|
|
|
client_name: sourceCode.displayName,
|
|
|
|
redirect_uris: `${window.location.origin}/auth/external`,
|
|
|
|
website: sourceCode.homepage,
|
|
|
|
scopes,
|
|
|
|
};
|
2021-08-22 12:34:58 -07:00
|
|
|
|
2021-08-22 17:13:09 -07:00
|
|
|
return dispatch(createApp(params, baseURL)).then(app => {
|
|
|
|
const { client_id, redirect_uri } = app;
|
2021-08-22 12:34:58 -07:00
|
|
|
|
2021-08-22 17:13:09 -07:00
|
|
|
const query = new URLSearchParams({
|
|
|
|
client_id,
|
|
|
|
redirect_uri,
|
|
|
|
response_type: 'code',
|
|
|
|
scope: scopes,
|
|
|
|
});
|
2021-08-22 12:34:58 -07:00
|
|
|
|
2021-08-22 17:13:09 -07:00
|
|
|
localStorage.setItem('soapbox:external:app', JSON.stringify(app));
|
|
|
|
localStorage.setItem('soapbox:external:baseurl', baseURL);
|
|
|
|
localStorage.setItem('soapbox:external:scopes', scopes);
|
|
|
|
|
|
|
|
window.location.href = `${baseURL}/oauth/authorize?${query.toString()}`;
|
|
|
|
});
|
2021-08-22 12:34:58 -07:00
|
|
|
});
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export function loginWithCode(code) {
|
|
|
|
return (dispatch, getState) => {
|
2021-08-22 12:46:40 -07:00
|
|
|
const { client_id, client_secret, redirect_uri } = JSON.parse(localStorage.getItem('soapbox:external:app'));
|
2021-08-22 12:34:58 -07:00
|
|
|
const baseURL = localStorage.getItem('soapbox:external:baseurl');
|
2021-08-22 17:13:09 -07:00
|
|
|
const scope = localStorage.getItem('soapbox:external:scopes');
|
2021-08-22 12:34:58 -07:00
|
|
|
|
|
|
|
const params = {
|
|
|
|
client_id,
|
|
|
|
client_secret,
|
2021-08-22 12:46:40 -07:00
|
|
|
redirect_uri,
|
2021-08-22 12:34:58 -07:00
|
|
|
grant_type: 'authorization_code',
|
2021-08-22 15:13:01 -07:00
|
|
|
scope,
|
2021-08-22 12:34:58 -07:00
|
|
|
code,
|
|
|
|
};
|
|
|
|
|
|
|
|
return dispatch(obtainOAuthToken(params, baseURL))
|
|
|
|
.then(token => dispatch(authLoggedIn(token)))
|
|
|
|
.then(({ access_token }) => dispatch(verifyCredentials(access_token, baseURL)))
|
2021-08-30 16:41:05 -07:00
|
|
|
.then(account => dispatch(switchAccount(account.id)))
|
2021-08-22 12:34:58 -07:00
|
|
|
.then(() => window.location.href = '/');
|
|
|
|
};
|
|
|
|
}
|