Merge pull request #133 from status-im/bug_fix/double-auth

Don't authenticate twice
This commit is contained in:
Iuri Matias 2018-10-26 16:58:08 +02:00 committed by GitHub
commit b02cafe672
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 93 additions and 37 deletions

View File

@ -5,5 +5,5 @@ export const LIGHT_THEME = 'light';
export const DEPLOYMENT_PIPELINES = {
injectedWeb3: 'injectedWeb3',
embark: 'embark'
}
export const DEFAULT_HOST = 'localhost:8000';
};
export const DEFAULT_HOST = process.env.NODE_ENV === 'development' ? 'localhost:8000' : window.location.host;

View File

@ -6,6 +6,7 @@ import routes from '../routes';
import Login from '../components/Login';
import Layout from "../components/Layout";
import {DEFAULT_HOST} from '../constants';
import {getQueryToken, stripQueryToken} from '../utils/utils';
import {
authenticate, fetchCredentials, logout,
@ -18,42 +19,57 @@ import {
import {LIGHT_THEME, DARK_THEME} from '../constants';
import { getCredentials, getAuthenticationError, getProcesses, getTheme } from '../reducers/selectors';
const qs = require('qs');
import {
getCredentials, getAuthenticationError, getProcesses, getTheme
} from '../reducers/selectors';
class AppContainer extends Component {
constructor (props) {
super(props);
this.queryStringAuthenticate();
}
queryStringAuthenticate() {
const token = qs.parse(this.props.location.search, {ignoreQueryPrefix: true}).token;
if (!token) {
return;
}
const host = process.env.NODE_ENV === 'development' ? DEFAULT_HOST : window.location.host;
if (token === this.props.credentials.token && this.props.credentials.host === host) {
return;
}
this.props.authenticate(host, token);
}
componentDidMount() {
this.props.fetchCredentials();
this.props.fetchTheme();
}
doAuthenticate() {
let {host, token} = this.props.credentials;
const queryToken = getQueryToken(this.props.location);
if (queryToken) {
host = DEFAULT_HOST;
token = queryToken;
}
this.props.authenticate(host, token);
}
requireAuthentication() {
return this.props.credentials.token && this.props.credentials.host && !this.props.credentials.authenticated;
if (this.props.credentials.authenticating) {
return false;
}
const queryToken = getQueryToken(this.props.location);
if (queryToken && !(queryToken === this.props.credentials.token &&
this.props.credentials.host === DEFAULT_HOST)) {
return true;
}
if (!this.props.credentials.authenticated &&
this.props.credentials.host &&
this.props.credentials.token) {
return true;
}
return false;
}
componentDidUpdate(){
if (this.requireAuthentication()) {
this.props.authenticate(this.props.credentials.host, this.props.credentials.token);
this.doAuthenticate();
}
if (getQueryToken(this.props.location) &&
(!this.props.credentials.authenticating ||
this.props.credentials.authenticated)) {
this.props.history.replace(stripQueryToken(this.props.location));
}
if (this.props.credentials.authenticated && !this.props.initialized) {
@ -64,7 +80,8 @@ class AppContainer extends Component {
}
shouldRenderLogin() {
return this.props.authenticationError || !this.props.credentials.authenticated;
return this.props.authenticationError ||
!(this.props.credentials.authenticated || this.props.credentials.authenticating);
}
toggleTheme() {
@ -77,7 +94,13 @@ class AppContainer extends Component {
renderBody() {
if (this.shouldRenderLogin()) {
return <Login credentials={this.props.credentials} authenticate={this.props.authenticate} error={this.props.authenticationError} />;
return (
<Login credentials={this.props.credentials}
authenticate={this.props.authenticate}
error={this.props.authenticationError} />
);
} else if (this.props.credentials.authenticating) {
return <React.Fragment/>;
}
return (
<Layout location={this.props.location}
@ -90,7 +113,11 @@ class AppContainer extends Component {
}
render() {
return <div className={(this.props.theme) + "-theme"}>{this.renderBody()}</div>;
return (
<div className={(this.props.theme) + "-theme"}>
{this.renderBody()}
</div>
);
}
}

View File

@ -200,7 +200,13 @@ function compilingContract(state = false, action) {
return state;
}
const DEFAULT_CREDENTIALS_STATE = {host: DEFAULT_HOST, token: '', authenticated: false};
const DEFAULT_CREDENTIALS_STATE = {
host: DEFAULT_HOST,
token: '',
authenticated: false,
authenticating: false,
error: null
};
function credentials(state = DEFAULT_CREDENTIALS_STATE, action) {
if (action.type === LOGOUT[SUCCESS]) {
@ -208,17 +214,21 @@ function credentials(state = DEFAULT_CREDENTIALS_STATE, action) {
}
if (action.type === AUTHENTICATE[FAILURE]) {
return {error: action.error, authenticated: false};
return {error: action.error, ...DEFAULT_CREDENTIALS_STATE};
}
if (action.type === AUTHENTICATE[SUCCESS]) {
return {...state, ...{authenticated: true, token: action.token, host: action.host, error: null}};
return {...state, ...{authenticated: true, authenticating: false, token: action.token, host: action.host, error: null}};
}
if (action.type === FETCH_CREDENTIALS[SUCCESS]) {
return {...state, ...{token: action.token, host: action.host}};
}
if (action.type === AUTHENTICATE[REQUEST]) {
return {...state, ...{authenticating: true, error: null}};
}
return state;
}

View File

@ -273,6 +273,10 @@ export function *watchAuthenticateSuccess() {
yield takeEvery(actions.AUTHENTICATE[actions.SUCCESS], saveCredentials);
}
export function *watchAuthenticateFailure() {
yield takeEvery(actions.AUTHENTICATE[actions.FAILURE], logout);
}
export function *watchFetchCredentials() {
yield takeEvery(actions.FETCH_CREDENTIALS[actions.REQUEST], fetchCredentials);
}
@ -518,6 +522,7 @@ export default function *root() {
fork(watchToggleBreakpoint),
fork(watchAuthenticate),
fork(watchAuthenticateSuccess),
fork(watchAuthenticateFailure),
fork(watchLogout),
fork(watchExplorerSearch),
fork(watchFetchTheme),

View File

@ -1,4 +1,5 @@
const Convert = require('ansi-to-html');
import Convert from 'ansi-to-html';
import qs from 'qs';
export function last(array) {
return array && array.length ? array[array.length - 1] : undefined;
@ -17,5 +18,18 @@ export function hashCode(str) {
export function ansiToHtml(text) {
const convert = new Convert();
return convert.toHtml(text.replace(/\n/g,'<br>'))
return convert.toHtml(text.replace(/\n/g,'<br>'));
}
export function getQueryToken(location) {
return qs.parse(location.search, {ignoreQueryPrefix: true}).token;
}
export function stripQueryToken(location) {
const _location = Object.assign({}, location);
_location.search = _location.search.replace(
/(\?|&?)(token=[\w-]*)(&?)/,
(_, p1, p2, p3) => (p2 ? (p3 === '&' ? p1 : '') : '')
);
return _location;
}