f0ef92614d
Add a persistent (per session) flag to skip any actions sparked by future renders after it has already been authenticated.
142 lines
4.2 KiB
JavaScript
142 lines
4.2 KiB
JavaScript
// @flow
|
|
import * as PAGES from 'constants/pages';
|
|
import React, { useState } from 'react';
|
|
import { withRouter } from 'react-router';
|
|
import Page from 'component/page';
|
|
import ReCAPTCHA from 'react-google-recaptcha';
|
|
import Button from 'component/button';
|
|
import { Lbryio } from 'lbryinc';
|
|
import I18nMessage from 'component/i18nMessage';
|
|
import Card from 'component/common/card';
|
|
|
|
type Props = {
|
|
history: { push: (string) => void, location: { search: string } },
|
|
doToast: ({}) => void,
|
|
};
|
|
|
|
let authenticationCompleted = false;
|
|
|
|
function SignInVerifyPage(props: Props) {
|
|
const {
|
|
history: { push, location },
|
|
doToast,
|
|
} = props;
|
|
const [isAuthenticationSuccess, setIsAuthenticationSuccess] = useState(authenticationCompleted);
|
|
const [showCaptchaMessage, setShowCaptchaMessage] = useState(false);
|
|
const [captchaLoaded, setCaptchaLoaded] = useState(false);
|
|
const urlParams = new URLSearchParams(location.search);
|
|
const authToken = urlParams.get('auth_token');
|
|
const userSubmittedEmail = urlParams.get('email');
|
|
const verificationToken = urlParams.get('verification_token');
|
|
const needsRecaptcha = urlParams.get('needs_recaptcha') === 'true';
|
|
|
|
function onAuthError(message) {
|
|
doToast({
|
|
message: message || __('Authentication failure.'),
|
|
isError: true,
|
|
});
|
|
push(`/$/${PAGES.AUTH}`);
|
|
}
|
|
|
|
React.useEffect(() => {
|
|
if (!authToken || !userSubmittedEmail || !verificationToken) {
|
|
onAuthError(__('Invalid or expired sign-in link.'));
|
|
}
|
|
}, [authToken, userSubmittedEmail, verificationToken, doToast, push]);
|
|
|
|
React.useEffect(() => {
|
|
if (!needsRecaptcha && !isAuthenticationSuccess) {
|
|
verifyUser();
|
|
}
|
|
}, []);
|
|
|
|
React.useEffect(() => {
|
|
let captchaTimeout;
|
|
|
|
if (needsRecaptcha && !captchaLoaded) {
|
|
captchaTimeout = setTimeout(() => {
|
|
setShowCaptchaMessage(true);
|
|
}, 2000);
|
|
}
|
|
|
|
return () => {
|
|
if (captchaTimeout) {
|
|
clearTimeout(captchaTimeout);
|
|
}
|
|
};
|
|
}, [needsRecaptcha, captchaLoaded]);
|
|
|
|
function onCaptchaChange(value) {
|
|
verifyUser(value);
|
|
}
|
|
|
|
function onCaptchaReady() {
|
|
setCaptchaLoaded(true);
|
|
}
|
|
|
|
function verifyUser(captchaValue) {
|
|
Lbryio.call('user_email', 'confirm', {
|
|
auth_token: authToken,
|
|
email: userSubmittedEmail,
|
|
verification_token: verificationToken,
|
|
...(captchaValue ? { recaptcha: captchaValue } : {}),
|
|
})
|
|
.then(() => {
|
|
authenticationCompleted = true;
|
|
setIsAuthenticationSuccess(true);
|
|
})
|
|
.catch(() => {
|
|
onAuthError(__('Invalid captcha response or other authentication error.'));
|
|
});
|
|
}
|
|
|
|
return (
|
|
<Page authPage noFooter>
|
|
<div className="main__sign-up">
|
|
<Card
|
|
title={isAuthenticationSuccess ? __('Log in success!') : __('Log in')}
|
|
subtitle={
|
|
<React.Fragment>
|
|
<p>
|
|
{isAuthenticationSuccess
|
|
? __('You can now close this tab.')
|
|
: needsRecaptcha
|
|
? null
|
|
: __('Welcome back! You are automatically being signed in.')}
|
|
</p>
|
|
{showCaptchaMessage && !isAuthenticationSuccess && (
|
|
<p>
|
|
<I18nMessage
|
|
tokens={{
|
|
refresh: (
|
|
<Button button="link" label={__('refreshing')} onClick={() => window.location.reload()} />
|
|
),
|
|
}}
|
|
>
|
|
Not seeing a captcha? Check your ad blocker or try %refresh%.
|
|
</I18nMessage>
|
|
</p>
|
|
)}
|
|
</React.Fragment>
|
|
}
|
|
actions={
|
|
!isAuthenticationSuccess &&
|
|
needsRecaptcha && (
|
|
<div className="section__actions">
|
|
<ReCAPTCHA
|
|
sitekey="6LePsJgUAAAAAFTuWOKRLnyoNKhm0HA4C3elrFMG"
|
|
onChange={onCaptchaChange}
|
|
asyncScriptOnLoad={onCaptchaReady}
|
|
onExpired={onAuthError}
|
|
onErrored={onAuthError}
|
|
/>
|
|
</div>
|
|
)
|
|
}
|
|
/>
|
|
</div>
|
|
</Page>
|
|
);
|
|
}
|
|
|
|
export default withRouter(SignInVerifyPage);
|