bigbuffet-rw/app/soapbox/actions/verification.js

410 lines
10 KiB
JavaScript
Raw Normal View History

2022-03-21 11:09:01 -07:00
import api from '../api';
/**
* LocalStorage 'soapbox:verification'
*
* {
* token: String,
* challenges: {
* email: Number (0 = incomplete, 1 = complete),
* sms: Number,
* age: Number
* }
* }
*/
const LOCAL_STORAGE_VERIFICATION_KEY = 'soapbox:verification';
const PEPE_FETCH_INSTANCE_SUCCESS = 'PEPE_FETCH_INSTANCE_SUCCESS';
const FETCH_CHALLENGES_SUCCESS = 'FETCH_CHALLENGES_SUCCESS';
const FETCH_TOKEN_SUCCESS = 'FETCH_TOKEN_SUCCESS';
const SET_NEXT_CHALLENGE = 'SET_NEXT_CHALLENGE';
const SET_CHALLENGES_COMPLETE = 'SET_CHALLENGES_COMPLETE';
const SET_LOADING = 'SET_LOADING';
const ChallengeTypes = {
EMAIL: 'email',
SMS: 'sms',
AGE: 'age',
};
/**
* Fetch the state of the user's verification in local storage.
*
* @returns {object}
* {
* token: String,
* challenges: {
* email: Number (0 = incomplete, 1 = complete),
* sms: Number,
* age: Number
* }
* }
*/
function fetchStoredVerification() {
try {
return JSON.parse(localStorage.getItem(LOCAL_STORAGE_VERIFICATION_KEY));
} catch {
return null;
}
}
/**
* Remove the state of the user's verification from local storage.
*/
function removeStoredVerification() {
localStorage.removeItem(LOCAL_STORAGE_VERIFICATION_KEY);
}
/**
* Fetch and return the Registration token for Pepe.
* @returns {string}
*/
function fetchStoredToken() {
try {
const verification = fetchStoredVerification();
return verification.token;
} catch {
return null;
}
}
/**
* Fetch and return the state of the verification challenges.
* @returns {object}
* {
* challenges: {
* email: Number (0 = incomplete, 1 = complete),
* sms: Number,
* age: Number
* }
* }
*/
function fetchStoredChallenges() {
try {
const verification = fetchStoredVerification();
return verification.challenges;
} catch {
return null;
}
}
/**
* Update the verification object in local storage.
*
* @param {*} verification object
*/
function updateStorage({ ...updatedVerification }) {
const verification = fetchStoredVerification();
localStorage.setItem(
LOCAL_STORAGE_VERIFICATION_KEY,
JSON.stringify({ ...verification, ...updatedVerification }),
);
}
/**
* Fetch Pepe challenges and registration token
* @returns {promise}
*/
function fetchVerificationConfig() {
return async(dispatch) => {
await dispatch(fetchPepeInstance());
dispatch(fetchRegistrationToken());
};
}
/**
* Save the challenges in localStorage.
*
* - If the API removes a challenge after the client has stored it, remove that
* challenge from localStorage.
* - If the API adds a challenge after the client has stored it, add that
* challenge to localStorage.
* - Don't overwrite a challenge that has already been completed.
* - Update localStorage to the new set of challenges.
*
* @param {array} challenges - ['age', 'sms', 'email']
*/
function saveChallenges(challenges) {
const currentChallenges = fetchStoredChallenges() || {};
const challengesToRemove = Object.keys(currentChallenges).filter((currentChallenge) => !challenges.includes(currentChallenge));
challengesToRemove.forEach((challengeToRemove) => delete currentChallenges[challengeToRemove]);
for (let i = 0; i < challenges.length; i++) {
const challengeName = challenges[i];
if (typeof currentChallenges[challengeName] !== 'number') {
currentChallenges[challengeName] = 0;
}
}
updateStorage({ challenges: currentChallenges });
}
/**
* Finish a challenge.
* @param {string} challenge - "sms" or "email" or "age"
*/
function finishChallenge(challenge) {
const currentChallenges = fetchStoredChallenges() || {};
// Set challenge to "complete"
currentChallenges[challenge] = 1;
updateStorage({ challenges: currentChallenges });
}
/**
* Fetch the next challenge
* @returns {string} challenge - "sms" or "email" or "age"
*/
function fetchNextChallenge() {
const currentChallenges = fetchStoredChallenges() || {};
return Object.keys(currentChallenges).find((challenge) => currentChallenges[challenge] === 0);
}
/**
* Dispatch the next challenge or set to complete if all challenges are completed.
*/
function dispatchNextChallenge(dispatch) {
const nextChallenge = fetchNextChallenge();
if (nextChallenge) {
dispatch({ type: SET_NEXT_CHALLENGE, challenge: nextChallenge });
} else {
dispatch({ type: SET_CHALLENGES_COMPLETE });
}
}
/**
* Fetch the challenges and age mininum from Pepe
* @returns {promise}
*/
function fetchPepeInstance() {
return (dispatch, getState) => {
dispatch({ type: SET_LOADING });
return api(getState).get('/api/v1/pepe/instance').then(response => {
const { challenges, age_minimum: ageMinimum } = response.data;
saveChallenges(challenges);
const currentChallenge = fetchNextChallenge();
dispatch({ type: PEPE_FETCH_INSTANCE_SUCCESS, instance: { isReady: true, ...response.data } });
dispatch({
type: FETCH_CHALLENGES_SUCCESS,
ageMinimum,
currentChallenge,
isComplete: !currentChallenge,
});
})
.finally(() => dispatch({ type: SET_LOADING, value: false }));
};
}
/**
* Fetch the regristration token from Pepe unless it's already been stored locally
* @returns {promise}
*/
function fetchRegistrationToken() {
return (dispatch, getState) => {
dispatch({ type: SET_LOADING });
const token = fetchStoredToken();
if (token) {
dispatch({
type: FETCH_TOKEN_SUCCESS,
value: token,
});
return null;
}
return api(getState).post('/api/v1/pepe/registrations')
.then(response => {
updateStorage({ token: response.data.access_token });
return dispatch({
type: FETCH_TOKEN_SUCCESS,
value: response.data.access_token,
});
})
.finally(() => dispatch({ type: SET_LOADING, value: false }));
};
}
function checkEmailAvailability(email) {
return (dispatch, getState) => {
dispatch({ type: SET_LOADING });
const token = fetchStoredToken();
return api(getState).get(`/api/v1/pepe/account/exists?email=${email}`, {
headers: { Authorization: `Bearer ${token}` },
}).finally(() => dispatch({ type: SET_LOADING, value: false }));
};
}
/**
* Send the user's email to Pepe to request confirmation
* @param {string} email
* @returns {promise}
*/
function requestEmailVerification(email) {
return (dispatch, getState) => {
dispatch({ type: SET_LOADING });
const token = fetchStoredToken();
return api(getState).post('/api/v1/pepe/verify_email/request', { email }, {
headers: { Authorization: `Bearer ${token}` },
})
.finally(() => dispatch({ type: SET_LOADING, value: false }));
};
}
function checkEmailVerification() {
return (dispatch, getState) => {
const token = fetchStoredToken();
return api(getState).get('/api/v1/pepe/verify_email', {
headers: { Authorization: `Bearer ${token}` },
});
};
}
/**
* Confirm the user's email with Pepe
* @param {string} emailToken
* @returns {promise}
*/
function confirmEmailVerification(emailToken) {
return (dispatch, getState) => {
dispatch({ type: SET_LOADING });
const token = fetchStoredToken();
return api(getState).post('/api/v1/pepe/verify_email/confirm', { token: emailToken }, {
headers: { Authorization: `Bearer ${token}` },
})
.then(() => {
finishChallenge(ChallengeTypes.EMAIL);
dispatchNextChallenge(dispatch);
})
.finally(() => dispatch({ type: SET_LOADING, value: false }));
};
}
function postEmailVerification() {
return (dispatch, getState) => {
finishChallenge(ChallengeTypes.EMAIL);
dispatchNextChallenge(dispatch);
};
}
/**
* Send the user's phone number to Pepe to request confirmation
* @param {string} phone
* @returns {promise}
*/
function requestPhoneVerification(phone) {
return (dispatch, getState) => {
dispatch({ type: SET_LOADING });
const token = fetchStoredToken();
return api(getState).post('/api/v1/pepe/verify_sms/request', { phone }, {
headers: { Authorization: `Bearer ${token}` },
})
.finally(() => dispatch({ type: SET_LOADING, value: false }));
};
}
/**
* Confirm the user's phone number with Pepe
* @param {string} code
* @returns {promise}
*/
function confirmPhoneVerification(code) {
return (dispatch, getState) => {
dispatch({ type: SET_LOADING });
const token = fetchStoredToken();
return api(getState).post('/api/v1/pepe/verify_sms/confirm', { code }, {
headers: { Authorization: `Bearer ${token}` },
})
.then(() => {
finishChallenge(ChallengeTypes.SMS);
dispatchNextChallenge(dispatch);
})
.finally(() => dispatch({ type: SET_LOADING, value: false }));
};
}
/**
* Confirm the user's age with Pepe
* @param {date} birthday
* @returns {promise}
*/
function verifyAge(birthday) {
return (dispatch, getState) => {
dispatch({ type: SET_LOADING });
const token = fetchStoredToken();
return api(getState).post('/api/v1/pepe/verify_age/confirm', { birthday }, {
headers: { Authorization: `Bearer ${token}` },
})
.then(() => {
finishChallenge(ChallengeTypes.AGE);
dispatchNextChallenge(dispatch);
})
.finally(() => dispatch({ type: SET_LOADING, value: false }));
};
}
/**
* Create the user's account with Pepe
* @param {string} username
* @param {string} password
* @returns {promise}
*/
function createAccount(username, password) {
return (dispatch, getState) => {
dispatch({ type: SET_LOADING });
const token = fetchStoredToken();
return api(getState).post('/api/v1/pepe/accounts', { username, password }, {
headers: { Authorization: `Bearer ${token}` },
}).finally(() => dispatch({ type: SET_LOADING, value: false }));
};
}
export {
PEPE_FETCH_INSTANCE_SUCCESS,
FETCH_CHALLENGES_SUCCESS,
FETCH_TOKEN_SUCCESS,
LOCAL_STORAGE_VERIFICATION_KEY,
SET_CHALLENGES_COMPLETE,
SET_LOADING,
SET_NEXT_CHALLENGE,
checkEmailAvailability,
confirmEmailVerification,
confirmPhoneVerification,
createAccount,
fetchStoredChallenges,
fetchVerificationConfig,
fetchRegistrationToken,
removeStoredVerification,
requestEmailVerification,
checkEmailVerification,
postEmailVerification,
requestPhoneVerification,
verifyAge,
};