2016-12-09 14:02:36 +01:00
|
|
|
/* eslint react/no-multi-comp:off */
|
|
|
|
import React, { Component, PureComponent } from 'react';
|
2018-02-04 14:32:33 +01:00
|
|
|
import PropTypes from 'prop-types';
|
2016-12-05 15:15:01 +01:00
|
|
|
|
2018-08-06 22:16:36 +02:00
|
|
|
import { Link } from 'react-router-dom';
|
2016-12-13 20:54:53 +01:00
|
|
|
import {
|
2020-09-23 23:18:53 +02:00
|
|
|
Button,
|
2017-08-28 19:15:47 +02:00
|
|
|
Grid,
|
|
|
|
Cell,
|
|
|
|
Card,
|
2020-09-23 23:18:53 +02:00
|
|
|
CardActions,
|
2017-08-28 19:15:47 +02:00
|
|
|
CardTitle,
|
|
|
|
CardText,
|
|
|
|
CardMenu,
|
|
|
|
List,
|
|
|
|
ListItem,
|
|
|
|
ListItemContent,
|
|
|
|
Textfield,
|
|
|
|
Icon,
|
|
|
|
ProgressBar,
|
|
|
|
Tabs,
|
|
|
|
Tab,
|
2016-12-13 22:46:56 +01:00
|
|
|
Switch,
|
2016-12-13 20:54:53 +01:00
|
|
|
} from 'react-mdl';
|
2017-02-14 12:11:18 +01:00
|
|
|
import { IconLink, shorten, styles as commonStyles } from '../common';
|
2018-02-08 11:23:07 +01:00
|
|
|
import { formatFullDateTimeWithLocale } from '../common/util';
|
2018-12-19 14:54:52 +01:00
|
|
|
import { CREATE_FEATURE, CREATE_STRATEGY, UPDATE_APPLICATION } from '../../permissions';
|
2020-09-23 21:47:07 +02:00
|
|
|
import icons from './icon-names';
|
|
|
|
import MySelect from '../common/select';
|
2016-12-05 15:15:01 +01:00
|
|
|
|
2016-12-09 14:02:36 +01:00
|
|
|
class StatefulTextfield extends Component {
|
2018-02-04 14:32:33 +01:00
|
|
|
static propTypes = {
|
|
|
|
value: PropTypes.string,
|
|
|
|
label: PropTypes.string,
|
|
|
|
rows: PropTypes.number,
|
|
|
|
onBlur: PropTypes.func.isRequired,
|
|
|
|
};
|
|
|
|
|
2017-08-28 19:15:47 +02:00
|
|
|
constructor(props) {
|
2016-12-09 14:02:36 +01:00
|
|
|
super(props);
|
|
|
|
this.state = { value: props.value };
|
2017-08-28 19:15:47 +02:00
|
|
|
this.setValue = function setValue(e) {
|
2016-12-09 14:02:36 +01:00
|
|
|
this.setState({ value: e.target.value });
|
|
|
|
}.bind(this);
|
|
|
|
}
|
|
|
|
|
2017-08-28 19:15:47 +02:00
|
|
|
render() {
|
|
|
|
return (
|
|
|
|
<Textfield
|
|
|
|
style={{ width: '100%' }}
|
|
|
|
label={this.props.label}
|
|
|
|
floatingLabel
|
|
|
|
rows={this.props.rows}
|
|
|
|
value={this.state.value}
|
|
|
|
onChange={this.setValue}
|
|
|
|
onBlur={this.props.onBlur}
|
|
|
|
/>
|
2016-12-09 14:02:36 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-13 20:54:53 +01:00
|
|
|
class ClientApplications extends PureComponent {
|
2018-02-04 14:32:33 +01:00
|
|
|
static propTypes = {
|
|
|
|
fetchApplication: PropTypes.func.isRequired,
|
|
|
|
appName: PropTypes.string,
|
|
|
|
application: PropTypes.object,
|
2018-02-08 11:23:07 +01:00
|
|
|
location: PropTypes.object,
|
2018-02-04 14:32:33 +01:00
|
|
|
storeApplicationMetaData: PropTypes.func.isRequired,
|
2020-09-23 23:18:53 +02:00
|
|
|
deleteApplication: PropTypes.func.isRequired,
|
2019-01-16 10:39:58 +01:00
|
|
|
hasPermission: PropTypes.func.isRequired,
|
2020-09-23 23:18:53 +02:00
|
|
|
history: PropTypes.object.isRequired,
|
2018-02-04 14:32:33 +01:00
|
|
|
};
|
|
|
|
|
2017-08-28 19:15:47 +02:00
|
|
|
constructor(props) {
|
2016-12-13 20:54:53 +01:00
|
|
|
super(props);
|
|
|
|
this.state = { activeTab: 0 };
|
|
|
|
}
|
|
|
|
|
2017-08-28 19:15:47 +02:00
|
|
|
componentDidMount() {
|
2016-12-05 15:15:01 +01:00
|
|
|
this.props.fetchApplication(this.props.appName);
|
|
|
|
}
|
2018-02-08 11:23:07 +01:00
|
|
|
formatFullDateTime(v) {
|
|
|
|
return formatFullDateTimeWithLocale(v, this.props.location.locale);
|
|
|
|
}
|
2020-09-23 23:18:53 +02:00
|
|
|
|
|
|
|
deleteApplication = async evt => {
|
|
|
|
evt.preventDefault();
|
|
|
|
const { deleteApplication, appName } = this.props;
|
|
|
|
await deleteApplication(appName);
|
|
|
|
this.props.history.push('/applications');
|
|
|
|
};
|
|
|
|
|
2017-08-28 19:15:47 +02:00
|
|
|
render() {
|
2016-12-05 15:15:01 +01:00
|
|
|
if (!this.props.application) {
|
2016-12-10 14:19:52 +01:00
|
|
|
return <ProgressBar indeterminate />;
|
2016-12-05 15:15:01 +01:00
|
|
|
}
|
2019-01-16 10:39:58 +01:00
|
|
|
const { application, storeApplicationMetaData, hasPermission } = this.props;
|
2017-08-28 21:40:44 +02:00
|
|
|
const { appName, instances, strategies, seenToggles, url, description, icon = 'apps', color } = application;
|
2016-12-05 16:01:09 +01:00
|
|
|
|
2017-08-28 19:15:47 +02:00
|
|
|
const content =
|
|
|
|
this.state.activeTab === 0 ? (
|
|
|
|
<Grid style={{ margin: 0 }}>
|
|
|
|
<Cell col={6} tablet={4} phone={12}>
|
|
|
|
<h6> Toggles</h6>
|
|
|
|
<hr />
|
|
|
|
<List>
|
2019-10-09 19:58:49 +02:00
|
|
|
{seenToggles.map(({ name, description, enabled, notFound }, i) =>
|
|
|
|
notFound ? (
|
|
|
|
<ListItem twoLine key={i}>
|
|
|
|
{hasPermission(CREATE_FEATURE) ? (
|
|
|
|
<ListItemContent icon={'report'} subtitle={'Missing, want to create?'}>
|
|
|
|
<Link to={`/features/create?name=${name}`}>{name}</Link>
|
2017-08-28 19:15:47 +02:00
|
|
|
</ListItemContent>
|
2019-10-09 19:58:49 +02:00
|
|
|
) : (
|
|
|
|
<ListItemContent icon={'report'} subtitle={'Missing'}>
|
|
|
|
{name}
|
|
|
|
</ListItemContent>
|
|
|
|
)}
|
|
|
|
</ListItem>
|
|
|
|
) : (
|
|
|
|
<ListItem twoLine key={i}>
|
|
|
|
<ListItemContent
|
|
|
|
icon={
|
|
|
|
<span>
|
|
|
|
<Switch disabled checked={!!enabled} />
|
|
|
|
</span>
|
|
|
|
}
|
|
|
|
subtitle={shorten(description, 60)}
|
|
|
|
>
|
|
|
|
<Link to={`/features/view/${name}`}>{shorten(name, 50)}</Link>
|
|
|
|
</ListItemContent>
|
|
|
|
</ListItem>
|
|
|
|
)
|
2017-08-28 19:15:47 +02:00
|
|
|
)}
|
|
|
|
</List>
|
|
|
|
</Cell>
|
|
|
|
<Cell col={6} tablet={4} phone={12}>
|
|
|
|
<h6>Implemented strategies</h6>
|
|
|
|
<hr />
|
|
|
|
<List>
|
2019-10-09 19:58:49 +02:00
|
|
|
{strategies.map(({ name, description, notFound }, i) =>
|
|
|
|
notFound ? (
|
|
|
|
<ListItem twoLine key={`${name}-${i}`}>
|
|
|
|
{hasPermission(CREATE_STRATEGY) ? (
|
|
|
|
<ListItemContent icon={'report'} subtitle={'Missing, want to create?'}>
|
|
|
|
<Link to={`/strategies/create?name=${name}`}>{name}</Link>
|
|
|
|
</ListItemContent>
|
|
|
|
) : (
|
|
|
|
<ListItemContent icon={'report'} subtitle={'Missing'}>
|
|
|
|
{name}
|
2017-08-28 19:15:47 +02:00
|
|
|
</ListItemContent>
|
2019-10-09 19:58:49 +02:00
|
|
|
)}
|
|
|
|
</ListItem>
|
|
|
|
) : (
|
|
|
|
<ListItem twoLine key={`${name}-${i}`}>
|
|
|
|
<ListItemContent icon={'extension'} subtitle={shorten(description, 60)}>
|
|
|
|
<Link to={`/strategies/view/${name}`}>{shorten(name, 50)}</Link>
|
|
|
|
</ListItemContent>
|
|
|
|
</ListItem>
|
|
|
|
)
|
2017-08-28 19:15:47 +02:00
|
|
|
)}
|
|
|
|
</List>
|
|
|
|
</Cell>
|
|
|
|
<Cell col={12} tablet={12}>
|
|
|
|
<h6>{instances.length} Instances registered</h6>
|
|
|
|
<hr />
|
|
|
|
<List>
|
2017-08-28 21:40:44 +02:00
|
|
|
{instances.map(({ instanceId, clientIp, lastSeen, sdkVersion }, i) => (
|
|
|
|
<ListItem key={i} twoLine>
|
|
|
|
<ListItemContent
|
|
|
|
icon="timeline"
|
|
|
|
subtitle={
|
|
|
|
<span>
|
2018-02-08 12:57:36 +01:00
|
|
|
{clientIp} last seen at{' '}
|
|
|
|
<small>{this.formatFullDateTime(lastSeen)}</small>
|
2017-08-28 21:40:44 +02:00
|
|
|
</span>
|
|
|
|
}
|
|
|
|
>
|
|
|
|
{instanceId} {sdkVersion ? `(${sdkVersion})` : ''}
|
|
|
|
</ListItemContent>
|
|
|
|
</ListItem>
|
|
|
|
))}
|
2017-08-28 19:15:47 +02:00
|
|
|
</List>
|
|
|
|
</Cell>
|
|
|
|
</Grid>
|
|
|
|
) : (
|
|
|
|
<Grid>
|
|
|
|
<Cell col={6} tablet={12}>
|
|
|
|
<StatefulTextfield
|
|
|
|
value={url}
|
|
|
|
label="URL"
|
2020-09-23 21:47:07 +02:00
|
|
|
type="url"
|
2017-08-28 21:40:44 +02:00
|
|
|
onBlur={e => storeApplicationMetaData(appName, 'url', e.target.value)}
|
2017-08-28 19:15:47 +02:00
|
|
|
/>
|
|
|
|
<br />
|
|
|
|
<StatefulTextfield
|
|
|
|
value={description}
|
|
|
|
label="Description"
|
|
|
|
rows={5}
|
2017-08-28 21:40:44 +02:00
|
|
|
onBlur={e => storeApplicationMetaData(appName, 'description', e.target.value)}
|
2017-08-28 19:15:47 +02:00
|
|
|
/>
|
|
|
|
</Cell>
|
|
|
|
<Cell col={6} tablet={12}>
|
2020-09-23 21:47:07 +02:00
|
|
|
<MySelect
|
|
|
|
label="Icon"
|
2020-09-23 23:18:53 +02:00
|
|
|
options={icons.map(v => ({ key: v, label: v }))}
|
2017-08-28 19:15:47 +02:00
|
|
|
value={icon}
|
2020-09-23 21:47:07 +02:00
|
|
|
onChange={e => storeApplicationMetaData(appName, 'icon', e.target.value)}
|
|
|
|
filled
|
2017-08-28 19:15:47 +02:00
|
|
|
/>
|
|
|
|
<StatefulTextfield
|
|
|
|
value={color}
|
2020-09-23 21:47:07 +02:00
|
|
|
label="Color"
|
2017-08-28 21:40:44 +02:00
|
|
|
onBlur={e => storeApplicationMetaData(appName, 'color', e.target.value)}
|
2017-08-28 19:15:47 +02:00
|
|
|
/>
|
|
|
|
</Cell>
|
|
|
|
</Grid>
|
|
|
|
);
|
2016-12-13 20:54:53 +01:00
|
|
|
|
|
|
|
return (
|
2017-02-14 12:11:18 +01:00
|
|
|
<Card shadow={0} className={commonStyles.fullwidth}>
|
2017-08-28 21:40:44 +02:00
|
|
|
<CardTitle style={{ paddingTop: '24px', paddingRight: '64px', wordBreak: 'break-all' }}>
|
2020-09-23 23:18:53 +02:00
|
|
|
<Icon name={icon || 'apps'} />
|
2020-09-23 21:47:07 +02:00
|
|
|
{appName}
|
2017-02-14 12:11:18 +01:00
|
|
|
</CardTitle>
|
2017-08-28 19:15:47 +02:00
|
|
|
{description && <CardText>{description}</CardText>}
|
|
|
|
{url && (
|
|
|
|
<CardMenu>
|
|
|
|
<IconLink url={url} icon="link" />
|
|
|
|
</CardMenu>
|
|
|
|
)}
|
2019-01-16 10:39:58 +01:00
|
|
|
{hasPermission(UPDATE_APPLICATION) ? (
|
2020-09-23 23:18:53 +02:00
|
|
|
<div>
|
|
|
|
<CardActions
|
|
|
|
border
|
|
|
|
style={{
|
|
|
|
display: 'flex',
|
|
|
|
alignItems: 'center',
|
|
|
|
justifyContent: 'space-between',
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<span />
|
|
|
|
<Button accent title="Delete application" onClick={this.deleteApplication}>
|
|
|
|
Delete
|
|
|
|
</Button>
|
|
|
|
</CardActions>
|
|
|
|
<hr />
|
|
|
|
<Tabs
|
|
|
|
activeTab={this.state.activeTab}
|
|
|
|
onChange={tabId => this.setState({ activeTab: tabId })}
|
|
|
|
ripple
|
|
|
|
tabBarProps={{ style: { width: '100%' } }}
|
|
|
|
className="mdl-color--grey-100"
|
|
|
|
>
|
|
|
|
<Tab>Details</Tab>
|
|
|
|
<Tab>Edit</Tab>
|
|
|
|
</Tabs>
|
|
|
|
</div>
|
2019-01-16 10:39:58 +01:00
|
|
|
) : (
|
|
|
|
''
|
|
|
|
)}
|
2016-12-09 14:02:36 +01:00
|
|
|
|
2017-02-14 12:11:18 +01:00
|
|
|
{content}
|
|
|
|
</Card>
|
2016-12-05 15:15:01 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-13 20:54:53 +01:00
|
|
|
export default ClientApplications;
|