diff --git a/frontend/src/component/app.jsx b/frontend/src/component/app.jsx index a905082e31..b7272d7dda 100644 --- a/frontend/src/component/app.jsx +++ b/frontend/src/component/app.jsx @@ -22,6 +22,7 @@ import AuthenticationContainer from './user/authentication-container'; import ShowUserContainer from './user/show-user-container'; import ShowApiDetailsContainer from './api/show-api-details-container'; import { ScrollContainer } from 'react-router-scroll'; +import { logoutUser } from '../store/user/actions'; function replace(input, params) { if (!params) { @@ -112,7 +113,7 @@ export default class App extends Component { return [0, 0]; } }; - const createListItem = (path, caption, icon, isDrawerNavigation = false, isAnchor = false) => { + const createListItem = (path, caption, icon, isDrawerNavigation = false) => { const linkColor = isDrawerNavigation && this.context.router.isActive(path) ? 'mdl-color-text--black' @@ -127,15 +128,7 @@ export default class App extends Component { className={isDrawerNavigation ? [styles.navigationIcon, iconColor].join(' ') : undefined} /> ); - return isAnchor ? ( - - {icon && renderIcon} - {caption} - - ) : ( + return (
@@ -197,7 +190,7 @@ export default class App extends Component { {createListItem('/history', 'Event History', '')} {createListItem('/archive', 'Archived Toggles', '')} {createListItem('/applications', 'Applications', '')} - Sign out + {createListItem('/logout', 'Sign out', '')} diff --git a/frontend/src/component/feature/list-component.jsx b/frontend/src/component/feature/list-component.jsx index 3b38518ddc..c86c7bf976 100644 --- a/frontend/src/component/feature/list-component.jsx +++ b/frontend/src/component/feature/list-component.jsx @@ -1,9 +1,8 @@ import React from 'react'; import PropTypes from 'prop-types'; import Feature from './feature-list-item-component'; -import { Link } from 'react-router'; +import { hashHistory, Link } from 'react-router'; import { Icon, FABButton, Textfield, Menu, MenuItem, Card, CardActions, List } from 'react-mdl'; - import { MenuItemWithIcon, DropdownButton, styles as commonStyles } from '../common'; import styles from './feature.scss'; @@ -13,6 +12,7 @@ export default class FeatureListComponent extends React.Component { featureMetrics: PropTypes.object.isRequired, fetchFeatureToggles: PropTypes.func, fetchArchive: PropTypes.func, + logoutUser: PropTypes.func, revive: PropTypes.func, updateSetting: PropTypes.func.isRequired, toggleFeature: PropTypes.func, @@ -24,6 +24,10 @@ export default class FeatureListComponent extends React.Component { }; componentDidMount() { + if (this.props.logout) { + this.props.logoutUser(); + hashHistory.push(`/`); + } if (this.props.fetchFeatureToggles) { this.props.fetchFeatureToggles(); } else { diff --git a/frontend/src/component/feature/list-container.jsx b/frontend/src/component/feature/list-container.jsx index 3ca68741f1..65b9128ff2 100644 --- a/frontend/src/component/feature/list-container.jsx +++ b/frontend/src/component/feature/list-container.jsx @@ -3,6 +3,7 @@ import { toggleFeature, fetchFeatureToggles } from '../../store/feature-actions' import { updateSettingForGroup } from '../../store/settings/actions'; import FeatureListComponent from './list-component'; +import { logoutUser } from '../../store/user/actions'; export const mapStateToPropsConfigurable = isFeature => state => { const featureMetrics = state.featureMetrics.toJS(); @@ -71,6 +72,7 @@ export const mapStateToPropsConfigurable = isFeature => state => { }; const mapStateToProps = mapStateToPropsConfigurable(true); const mapDispatchToProps = { + logoutUser, toggleFeature, fetchFeatureToggles, updateSetting: updateSettingForGroup('feature'), diff --git a/frontend/src/data/user-api.js b/frontend/src/data/user-api.js index e2671ce8dc..ccab844390 100644 --- a/frontend/src/data/user-api.js +++ b/frontend/src/data/user-api.js @@ -2,6 +2,12 @@ import { throwIfNotSuccess, headers } from './helper'; const URI = 'api/admin/user'; +function logoutUser() { + return fetch(`${URI}/logout`, { method: 'GET', credentials: 'include' }) + .then(throwIfNotSuccess) + .then(response => response.json()); +} + function fetchUser() { return fetch(URI, { credentials: 'include' }) .then(throwIfNotSuccess) @@ -17,4 +23,5 @@ function unsecureLogin(path, user) { export default { fetchUser, unsecureLogin, + logoutUser, }; diff --git a/frontend/src/index.jsx b/frontend/src/index.jsx index 8e2000a9cd..188f331cce 100644 --- a/frontend/src/index.jsx +++ b/frontend/src/index.jsx @@ -26,6 +26,7 @@ import Archive from './page/archive'; import ShowArchive from './page/archive/show'; import Applications from './page/applications'; import ApplicationView from './page/applications/view'; +import LogoutFeatures from './page/user/logout'; let composeEnhancers; @@ -76,6 +77,9 @@ ReactDOM.render( + + + , diff --git a/frontend/src/page/user/logout.js b/frontend/src/page/user/logout.js new file mode 100644 index 0000000000..d2b3217548 --- /dev/null +++ b/frontend/src/page/user/logout.js @@ -0,0 +1,6 @@ +import React from 'react'; +import FeatureListContainer from './../../component/feature/list-container'; + +const render = () => ; + +export default render; diff --git a/frontend/src/store/user/actions.js b/frontend/src/store/user/actions.js index c52ca50c27..ba8cf0a852 100644 --- a/frontend/src/store/user/actions.js +++ b/frontend/src/store/user/actions.js @@ -36,3 +36,7 @@ export function unsecureLogin(path, user) { .catch(handleError); }; } + +export function logoutUser() { + return () => api.logoutUser().catch(handleError); +}