Live: improved channel state (#27672)

This commit is contained in:
Ryan McKinley
2020-09-23 08:02:01 -07:00
committed by GitHub
parent 83050b9ccc
commit 93a01eb1d9
13 changed files with 915 additions and 219 deletions

View File

@@ -1,85 +1,137 @@
import React, { PureComponent } from 'react';
import { hot } from 'react-hot-loader';
import { connect } from 'react-redux';
import { css } from 'emotion';
import { StoreState } from 'app/types';
import { getNavModel } from 'app/core/selectors/navModel';
import Page from 'app/core/components/Page/Page';
import { NavModel, SelectableValue, FeatureState } from '@grafana/data';
import {
NavModel,
SelectableValue,
FeatureState,
LiveChannelScope,
LiveChannelConfig,
LiveChannelSupport,
} from '@grafana/data';
import { LivePanel } from './LivePanel';
import { Select, Input, Button, FeatureInfoBox, Container } from '@grafana/ui';
import { getGrafanaLiveSrv } from '@grafana/runtime';
import { Select, FeatureInfoBox, Container } from '@grafana/ui';
import { getGrafanaLiveCentrifugeSrv } from '../live/live';
interface Props {
navModel: NavModel;
}
const scopes: Array<SelectableValue<LiveChannelScope>> = [
{ label: 'Grafana', value: LiveChannelScope.Grafana, description: 'Core grafana live features' },
{ label: 'Data Sources', value: LiveChannelScope.DataSource, description: 'Data sources with live support' },
{ label: 'Plugins', value: LiveChannelScope.Plugin, description: 'Plugins with live support' },
];
interface State {
channel: string;
text: string;
scope: LiveChannelScope;
namespace?: string;
path?: string;
namespaces: Array<SelectableValue<string>>;
paths: Array<SelectableValue<string>>;
support?: LiveChannelSupport;
config?: LiveChannelConfig;
}
export class LiveAdmin extends PureComponent<Props, State> {
state: State = {
channel: 'random-2s-stream',
text: '', // publish text to a channel
scope: LiveChannelScope.Grafana,
namespace: 'testdata',
path: 'random-2s-stream',
namespaces: [],
paths: [],
};
// onTextChanged: ((event: FormEvent<HTMLInputElement>) => void) | undefined;
// onPublish: ((event: MouseEvent<HTMLButtonElement, MouseEvent>) => void) | undefined;
onChannelChanged = (v: SelectableValue<string>) => {
async componentDidMount() {
const { scope, namespace, path } = this.state;
const srv = getGrafanaLiveCentrifugeSrv();
const namespaces = await srv.scopes[scope].listNamespaces();
const support = namespace ? await srv.scopes[scope].getChannelSupport(namespace) : undefined;
const paths = support ? await support.getSupportedPaths() : undefined;
const config = support && path ? await support.getChannelConfig(path) : undefined;
this.setState({
namespaces,
support,
paths: paths
? paths.map(p => ({
label: p.path,
value: p.path,
description: p.description,
}))
: [],
config,
});
}
onScopeChanged = async (v: SelectableValue<LiveChannelScope>) => {
if (v.value) {
const srv = getGrafanaLiveCentrifugeSrv();
this.setState({
channel: v.value,
scope: v.value,
namespace: undefined,
path: undefined,
namespaces: await srv.scopes[v.value!].listNamespaces(),
paths: [],
support: undefined,
config: undefined,
});
}
};
onTextChanged = (event: React.ChangeEvent<HTMLInputElement>) => {
this.setState({ text: event.target.value });
};
onNamespaceChanged = async (v: SelectableValue<string>) => {
if (v.value) {
const namespace = v.value;
const srv = getGrafanaLiveCentrifugeSrv();
const support = await srv.scopes[this.state.scope].getChannelSupport(namespace);
onPublish = () => {
const { text, channel } = this.state;
if (text) {
const msg = {
line: text,
};
const srv = getGrafanaLiveSrv();
srv.publish(channel, msg).then(v => {
console.log('PUBLISHED', text, v);
this.setState({
namespace: v.value,
paths: support!.getSupportedPaths().map(p => ({
label: p.path,
value: p.path,
description: p.description,
})),
path: undefined,
config: undefined,
});
}
};
onPathChanged = async (v: SelectableValue<string>) => {
if (v.value) {
const path = v.value;
const srv = getGrafanaLiveCentrifugeSrv();
const support = await srv.scopes[this.state.scope].getChannelSupport(this.state.namespace!);
if (!support) {
this.setState({
namespace: undefined,
paths: [],
config: undefined,
support,
});
return;
}
this.setState({
path,
support,
config: support.getChannelConfig(path),
});
}
this.setState({ text: '' });
};
render() {
const { navModel } = this.props;
const { channel, text } = this.state;
const channels: Array<SelectableValue<string>> = [
{
label: 'random-2s-stream',
value: 'random-2s-stream',
description: 'Random stream that updates every 2s',
},
{
label: 'random-flakey-stream',
value: 'random-flakey-stream',
description: 'Random stream with intermittent updates',
},
{
label: 'example-chat',
value: 'example-chat',
description: 'A channel that expects chat messages',
},
];
let current = channels.find(f => f.value === channel);
if (!current) {
current = {
label: channel,
value: channel,
};
channels.push(current);
}
const { scope, namespace, namespaces, path, paths, config } = this.state;
return (
<Page navModel={navModel}>
@@ -99,19 +151,36 @@ export class LiveAdmin extends PureComponent<Props, State> {
<br />
</Container>
<h2>Channels</h2>
<Select options={channels} value={current} onChange={this.onChannelChanged} allowCustomValue={true} />
<br />
<LivePanel channel={channel} />
<div
className={css`
width: 100%;
display: flex;
> div {
margin-right: 8px;
min-width: 150px;
}
`}
>
<div>
<h5>Scope</h5>
<Select options={scopes} value={scopes.find(s => s.value === scope)} onChange={this.onScopeChanged} />
</div>
<div>
<h5>Namespace</h5>
<Select
options={namespaces}
value={namespaces.find(s => s.value === namespace) || ''}
onChange={this.onNamespaceChanged}
/>
</div>
<div>
<h5>Path</h5>
<Select options={paths} value={paths.find(s => s.value === path) || ''} onChange={this.onPathChanged} />
</div>
</div>
<br />
<br />
<h3>Write to channel</h3>
<Input value={text} onChange={this.onTextChanged} />
<Button onClick={this.onPublish} variant={text ? 'primary' : 'secondary'}>
Publish
</Button>
{scope && namespace && path && <LivePanel scope={scope} namespace={namespace} path={path} config={config} />}
</Page.Contents>
</Page>
);

View File

@@ -1,52 +1,71 @@
import React, { PureComponent } from 'react';
import { Unsubscribable, PartialObserver } from 'rxjs';
import { getGrafanaLiveSrv } from '@grafana/runtime';
import {
AppEvents,
LiveChannel,
LiveChannelConfig,
LiveChannelConnectionState,
LiveChannelMessage,
LiveChannelScope,
LiveChannelStatus,
} from '@grafana/data';
import { Input, Button } from '@grafana/ui';
import { appEvents } from 'app/core/core';
interface Props {
channel: string;
scope: LiveChannelScope;
namespace: string;
path: string;
config?: LiveChannelConfig;
}
interface State {
connected: boolean;
channel?: LiveChannel;
status: LiveChannelStatus;
count: number;
lastTime: number;
lastBody: string;
text: string; // for publish!
}
export class LivePanel extends PureComponent<Props, State> {
state: State = {
connected: false,
status: { id: '?', state: LiveChannelConnectionState.Pending, timestamp: Date.now() },
count: 0,
lastTime: 0,
lastBody: '',
text: '',
};
subscription?: Unsubscribable;
observer: PartialObserver<any> = {
next: (msg: any) => {
this.setState({
count: this.state.count + 1,
lastTime: Date.now(),
lastBody: JSON.stringify(msg),
});
streamObserver: PartialObserver<LiveChannelMessage> = {
next: (msg: LiveChannelMessage) => {
if (msg.type === 'status') {
this.setState({ status: msg.message as LiveChannelStatus });
} else {
this.setState({
count: this.state.count + 1,
lastTime: Date.now(),
lastBody: JSON.stringify(msg),
});
}
},
};
startSubscription = () => {
if (this.subscription) {
this.subscription.unsubscribe();
this.subscription = undefined;
const { scope, namespace, path } = this.props;
const channel = getGrafanaLiveSrv().getChannel(scope, namespace, path);
if (this.state.channel === channel) {
return; // no change!
}
const srv = getGrafanaLiveSrv();
if (srv.isConnected()) {
const stream = srv.getChannelStream(this.props.channel);
this.subscription = stream.subscribe(this.observer);
this.setState({ connected: true, count: 0, lastTime: 0, lastBody: '' });
return;
if (this.subscription) {
this.subscription.unsubscribe();
}
console.log('Not yet connected... try again...');
setTimeout(this.startSubscription, 200);
this.subscription = channel.getStream().subscribe(this.streamObserver);
this.setState({ channel });
};
componentDidMount = () => {
@@ -56,21 +75,47 @@ export class LivePanel extends PureComponent<Props, State> {
componentWillUnmount() {
if (this.subscription) {
this.subscription.unsubscribe();
this.subscription = undefined;
}
}
componentDidUpdate(oldProps: Props) {
if (oldProps.channel !== this.props.channel) {
if (oldProps.config !== this.props.config) {
this.startSubscription();
}
}
onTextChanged = (event: React.ChangeEvent<HTMLInputElement>) => {
this.setState({ text: event.target.value });
};
onPublish = () => {
const { text, channel } = this.state;
if (text && channel) {
const msg = {
line: text,
};
channel.publish!(msg)
.then(v => {
console.log('PUBLISHED', text, v);
})
.catch(err => {
appEvents.emit(AppEvents.alertError, ['Publish error', `${err}`]);
});
}
this.setState({ text: '' });
};
render() {
const { lastBody, lastTime, count } = this.state;
const { lastBody, lastTime, count, status, text } = this.state;
const { config } = this.props;
const showPublish = config && config.canPublish && config.canPublish();
return (
<div>
<h5>Status: {config ? '' : '(no config)'}</h5>
<pre>{JSON.stringify(status)}</pre>
<h5>Count: {count}</h5>
{lastTime > 0 && (
<>
@@ -82,6 +127,16 @@ export class LivePanel extends PureComponent<Props, State> {
)}
</>
)}
{showPublish && (
<div>
<h3>Write to channel</h3>
<Input value={text} onChange={this.onTextChanged} />
<Button onClick={this.onPublish} variant={text ? 'primary' : 'secondary'}>
Publish
</Button>
</div>
)}
</div>
);
}