ui: Move discovery chain component (#9154)

pull/9156/head
John Cowen 2020-11-11 14:43:37 +00:00 committed by GitHub
parent 2e654a1759
commit e1d9771381
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
21 changed files with 209 additions and 210 deletions

View File

@ -0,0 +1,148 @@
<style>
{{#if selected.nodes }}
{{selected.nodes}} {
opacity: 1 !important;
background-color: var(--white);
border: var(--decor-border-100);
border-radius: var(--decor-radius-200);
border-color: var(--gray-500);
box-shadow: var(--decor-elevation-600);
}
{{/if}}
{{#if selected.edges }}
{{selected.edges}} {
opacity: 1;
}
{{/if}}
</style>
<div class="routes">
<header>
<h2>
{{chain.ServiceName}} Router
<span>
<em role="tooltip">Use routers to intercept traffic using L7 criteria such as path prefixes or http headers.</em>
</span>
</h2>
</header>
<div role="group">
{{#each routes as |item|}}
<Consul::DiscoveryChain::RouteCard
@item={{item}}
@onclick={{action "click"}}
/>
{{/each}}
</div>
</div>
<div class="splitters">
<header>
<h2>
Splitters
<span>
<em role="tooltip">Splitters are configured to split incoming requests across different services or subsets of a single service.</em>
</span>
</h2>
</header>
<div role="group">
{{#each (sort-by 'Name' splitters) as |item|}}
<Consul::DiscoveryChain::SplitterCard
@item={{item}}
@onclick={{action "click"}}
/>
{{/each}}
</div>
</div>
<div class="resolvers">
<header>
<h2>
Resolvers
<span>
<em role="tooltip">Resolvers are used to define which instances of a service should satisfy discovery requests.</em>
</span>
</h2>
</header>
<div role="group">
{{#each (sort-by 'Name' resolvers) as |item|}}
<Consul::DiscoveryChain::ResolverCard
@item={{item}}
@onclick={{action "click"}}
/>
{{/each}}
</div>
</div>
<svg width="100%" height="100%" viewBox={{concat '0 0 ' width ' ' height}} preserveAspectRatio="none">
{{#each routes as |item|}}
{{#let (dom-position (concat '#' item.ID)) as |src|}}
{{#let (dom-position (concat '#' item.NextNode)) as |destRect|}}
{{#let (tween-to (hash
x=destRect.x
y=(add destRect.y (div destRect.height 2))
) (concat item.ID)) as |dest|}}
<path
id={{concat item.ID '>' item.NextNode}}
d={{
svg-curve (hash
x=dest.x
y=dest.y
) src=(hash
x=(add src.x src.width)
y=(add src.y (div src.height 2))
)}} />
{{/let}}
{{/let}}
{{/let}}
{{/each}}
{{#each splitters as |splitter|}}
{{#let (dom-position (concat '#' splitter.ID)) as |src|}}
{{#each splitter.Splits as |item index|}}
{{#let (dom-position (concat '#' item.NextNode)) as |destRect|}}
{{#let (tween-to (hash
x=destRect.x
y=(add destRect.y (div destRect.height 2))
) (concat splitter.ID '-' index)) as |dest|}}
<path
id={{concat 'splitter:' splitter.Name '>' item.NextNode}}
class="split"
data-percentage={{or item.Weight 0}}
d={{
svg-curve (hash
x=dest.x
y=dest.y
) src=(hash
x=(add src.x src.width)
y=(add src.y (div src.height 2))
)}} />
{{/let}}
{{/let}}
{{/each}}
{{/let}}
{{/each}}
</svg>
<svg class="resolver-inlets" viewBox={{concat '0 0 10 ' height}}>
{{#each routes as |item|}}
{{#if (starts-with 'resolver:' item.NextNode) }}
{{#let (dom-position (concat '#' item.NextNode)) as |dest|}}
<circle r="2.5" cx="5" cy={{add dest.y (div dest.height 2)}} />
{{/let}}
{{/if}}
{{/each}}
{{#each splitters as |item|}}
{{#each item.Splits as |item|}}
{{#let (dom-position (concat '#' item.NextNode)) as |dest|}}
<circle r="2.5" cx="5" cy={{add dest.y (div dest.height 2)}} />
{{/let}}
{{/each}}
{{/each}}
</svg>
<svg class="splitter-inlets" viewBox={{concat '0 0 10 ' height}}>
{{#each routes as |item|}}
{{#if (starts-with 'splitter:' item.NextNode) }}
{{#let (dom-position (concat '#' item.NextNode)) as |dest|}}
<circle r="2.5" cx="5" cy={{add dest.y (div dest.height 2)}} />
{{/let}}
{{/if}}
{{/each}}
</svg>
<div class={{concat 'tooltip' (if activeTooltip ' active' '')}} style={{{ concat 'top:' y 'px;left:' x 'px;'}}}>
<span role="tooltip">{{round tooltip decimals=2}}%</span>
</div>

View File

@ -20,22 +20,6 @@ export default Component.extend({
init: function() { init: function() {
this._super(...arguments); this._super(...arguments);
this._listeners = this.dom.listeners(); this._listeners = this.dom.listeners();
this._viewportlistener = this.dom.listeners();
},
didInsertElement: function() {
this._super(...arguments);
this._viewportlistener.add(
this.dom.isInViewport(this.element, bool => {
if (get(this, 'isDisplayed') !== bool) {
set(this, 'isDisplayed', bool);
if (this.isDisplayed) {
this.addPathListeners();
} else {
this.ticker.destroy(this);
}
}
})
);
}, },
didReceiveAttrs: function() { didReceiveAttrs: function() {
this._super(...arguments); this._super(...arguments);
@ -46,7 +30,6 @@ export default Component.extend({
willDestroyElement: function() { willDestroyElement: function() {
this._super(...arguments); this._super(...arguments);
this._listeners.remove(); this._listeners.remove();
this._viewportlistener.remove();
this.ticker.destroy(this); this.ticker.destroy(this);
}, },
splitters: computed('chain.Nodes', function() { splitters: computed('chain.Nodes', function() {
@ -133,10 +116,10 @@ export default Component.extend({
edges: edges.map(item => `#${CSS.escape(item)}`), edges: edges.map(item => `#${CSS.escape(item)}`),
}; };
}), }),
width: computed('isDisplayed', 'chain.{Nodes,Targets}', function() { width: computed('chain.{Nodes,Targets}', function() {
return this.element.offsetWidth; return this.element.offsetWidth;
}), }),
height: computed('isDisplayed', 'chain.{Nodes,Targets}', function() { height: computed('chain.{Nodes,Targets}', function() {
return this.element.offsetHeight; return this.element.offsetHeight;
}), }),
// TODO(octane): ember has trouble adding mouse events to svg elements whilst giving // TODO(octane): ember has trouble adding mouse events to svg elements whilst giving

View File

@ -1,4 +1,5 @@
@import './discovery-chain/index'; @import './skin';
@import './layout';
.discovery-chain { .discovery-chain {
@extend %discovery-chain; @extend %discovery-chain;
} }

View File

@ -1,13 +1,18 @@
<div class="resolver-card"> <div
<header onclick={{onclick}} id={{concat 'resolver:' item.ID}}> class="resolver-card"
...attributes
>
<header onclick={{optional @onclick}} id={{concat 'resolver:' @item.ID}}>
<a name=""> <a name="">
<h3>{{item.Name}}</h3> <h3>{{@item.Name}}</h3>
{{#if item.Failover}} {{#if item.Failover}}
<dl class="failover"> <dl class="failover">
<dt data-tooltip={{concat item.Failover.Type ' failover'}}>{{concat item.Failover.Type ' failover'}}</dt> <dt data-tooltip={{concat @item.Failover.Type ' failover'}}>
{{concat @item.Failover.Type ' failover'}}
</dt>
<dd> <dd>
<ol> <ol>
{{#each item.Failover.Targets as |item|}} {{#each @item.Failover.Targets as |item|}}
<li> <li>
<span>{{item}}</span> <span>{{item}}</span>
</li> </li>
@ -18,10 +23,10 @@
{{/if}} {{/if}}
</a> </a>
</header> </header>
{{#if (gt item.Children.length 0)}} {{#if (gt @item.Children.length 0)}}
<ul> <ul>
{{#each item.Children as |child|}} {{#each @item.Children as |child|}}
<li onclick={{onclick}} id={{concat 'resolver:' child.ID}}> <li onclick={{optional @onclick}} id={{concat 'resolver:' child.ID}}>
<a name=""> <a name="">
{{#if child.Redirect}} {{#if child.Redirect}}
<dl class="redirect"> <dl class="redirect">

View File

@ -1,28 +1,32 @@
<a class="route-card" onclick={{onclick}} id={{item.ID}}> <a
<header class={{if (eq path.value '/') 'short'}}> class="route-card"
{{#if (gt item.Definition.Match.HTTP.Methods.length 0) }} onclick={{@onclick}}
id={{@item.ID}}
>
<header class={{if (eq this.path.value '/') 'short'}}>
{{#if (gt @item.Definition.Match.HTTP.Methods.length 0) }}
<ul class="match-methods"> <ul class="match-methods">
{{#each item.Definition.Match.HTTP.Methods as |item|}} {{#each @item.Definition.Match.HTTP.Methods as |item|}}
<li>{{item}}</li> <li>{{item}}</li>
{{/each}} {{/each}}
</ul> </ul>
{{/if}} {{/if}}
<dl> <dl>
<dt> <dt>
{{path.type}} {{this.path.type}}
</dt> </dt>
<dd> <dd>
{{path.value}} {{this.path.value}}
</dd> </dd>
</dl> </dl>
</header> </header>
{{#if (gt item.Definition.Match.HTTP.Header.length 0) }} {{#if (gt @item.Definition.Match.HTTP.Header.length 0) }}
<section class="match-headers"> <section class="match-headers">
<header data-tooltip="Header"> <header data-tooltip="Header">
<h4>Headers</h4> <h4>Headers</h4>
</header> </header>
<dl> <dl>
{{#each item.Definition.Match.HTTP.Header as |item|}} {{#each @item.Definition.Match.HTTP.Header as |item|}}
<dt> <dt>
{{item.Name}} {{item.Name}}
</dt> </dt>
@ -33,13 +37,13 @@
</dl> </dl>
</section> </section>
{{/if}} {{/if}}
{{#if (gt item.Definition.Match.HTTP.QueryParam.length 0) }} {{#if (gt @item.Definition.Match.HTTP.QueryParam.length 0) }}
<section class="match-queryparams"> <section class="match-queryparams">
<header data-tooltip="Query Params"> <header data-tooltip="Query Params">
<h4>Query Params</h4> <h4>Query Params</h4>
</header> </header>
<dl> <dl>
{{#each item.Definition.Match.HTTP.QueryParam as |item|}} {{#each @item.Definition.Match.HTTP.QueryParam as |item|}}
<dt> <dt>
{{item.Name}} {{item.Name}}
</dt> </dt>

View File

@ -1,10 +1,9 @@
import Component from '@ember/component'; import Component from '@glimmer/component';
import { get, computed } from '@ember/object'; import { get } from '@ember/object';
export default Component.extend({ export default class RouteCard extends Component {
tagName: '', get path() {
path: computed('item', function() { return Object.entries(get(this.args.item, 'Definition.Match.HTTP') || {}).reduce(
return Object.entries(get(this, 'item.Definition.Match.HTTP') || {}).reduce(
function(prev, [key, value]) { function(prev, [key, value]) {
if (key.toLowerCase().startsWith('path')) { if (key.toLowerCase().startsWith('path')) {
return { return {
@ -19,5 +18,5 @@ export default Component.extend({
value: '/', value: '/',
} }
); );
}), }
}); }

View File

@ -0,0 +1,13 @@
<div
...attributes
>
<a
id={{@item.ID}}
class="splitter-card"
onclick={{optional @onclick}}
>
<header>
<h3>{{@item.Name}}</h3>
</header>
</a>
</div>

View File

@ -1,141 +0,0 @@
{{#if isDisplayed }}
<style>
{{#if selected.nodes }}
{{selected.nodes}} {
opacity: 1 !important;
background-color: var(--white);
border: var(--decor-border-100);
border-radius: var(--decor-radius-200);
border-color: var(--gray-500);
box-shadow: var(--decor-elevation-600);
}
{{/if}}
{{#if selected.edges }}
{{selected.edges}} {
opacity: 1;
}
{{/if}}
</style>
<div class="routes">
<header>
<h2>
{{chain.ServiceName}} Router
<span>
<em role="tooltip">Use routers to intercept traffic using L7 criteria such as path prefixes or http headers.</em>
</span>
</h2>
</header>
<div role="group">
{{#each routes as |item|}}
<RouteCard @item={{item}} @onclick={{action "click"}} />
{{/each}}
</div>
</div>
<div class="splitters">
<header>
<h2>
Splitters
<span>
<em role="tooltip">Splitters are configured to split incoming requests across different services or subsets of a single service.</em>
</span>
</h2>
</header>
<div role="group">
{{#each (sort-by 'Name' splitters) as |item|}}
<SplitterCard @item={{item}} @onclick={{action "click"}} />
{{/each}}
</div>
</div>
<div class="resolvers">
<header>
<h2>
Resolvers
<span>
<em role="tooltip">Resolvers are used to define which instances of a service should satisfy discovery requests.</em>
</span>
</h2>
</header>
<div role="group">
{{#each (sort-by 'Name' resolvers) as |item|}}
<ResolverCard @item={{item}} @onclick={{action "click"}} />
{{/each}}
</div>
</div>
<svg width="100%" height="100%" viewBox={{concat '0 0 ' width ' ' height}} preserveAspectRatio="none">
{{#each routes as |item|}}
{{#let (dom-position (concat '#' item.ID)) as |src|}}
{{#let (dom-position (concat '#' item.NextNode)) as |destRect|}}
{{#let (tween-to (hash
x=destRect.x
y=(add destRect.y (div destRect.height 2))
) (concat item.ID)) as |dest|}}
<path
id={{concat item.ID '>' item.NextNode}}
d={{
svg-curve (hash
x=dest.x
y=dest.y
) src=(hash
x=(add src.x src.width)
y=(add src.y (div src.height 2))
)}} />
{{/let}}
{{/let}}
{{/let}}
{{/each}}
{{#each splitters as |splitter|}}
{{#let (dom-position (concat '#' splitter.ID)) as |src|}}
{{#each splitter.Splits as |item index|}}
{{#let (dom-position (concat '#' item.NextNode)) as |destRect|}}
{{#let (tween-to (hash
x=destRect.x
y=(add destRect.y (div destRect.height 2))
) (concat splitter.ID '-' index)) as |dest|}}
<path
id={{concat 'splitter:' splitter.Name '>' item.NextNode}}
class="split"
data-percentage={{or item.Weight 0}}
d={{
svg-curve (hash
x=dest.x
y=dest.y
) src=(hash
x=(add src.x src.width)
y=(add src.y (div src.height 2))
)}} />
{{/let}}
{{/let}}
{{/each}}
{{/let}}
{{/each}}
</svg>
<svg class="resolver-inlets" viewBox={{concat '0 0 10 ' height}}>
{{#each routes as |item|}}
{{#if (starts-with 'resolver:' item.NextNode) }}
{{#let (dom-position (concat '#' item.NextNode)) as |dest|}}
<circle r="2.5" cx="5" cy={{add dest.y (div dest.height 2)}} />
{{/let}}
{{/if}}
{{/each}}
{{#each splitters as |item|}}
{{#each item.Splits as |item|}}
{{#let (dom-position (concat '#' item.NextNode)) as |dest|}}
<circle r="2.5" cx="5" cy={{add dest.y (div dest.height 2)}} />
{{/let}}
{{/each}}
{{/each}}
</svg>
<svg class="splitter-inlets" viewBox={{concat '0 0 10 ' height}}>
{{#each routes as |item|}}
{{#if (starts-with 'splitter:' item.NextNode) }}
{{#let (dom-position (concat '#' item.NextNode)) as |dest|}}
<circle r="2.5" cx="5" cy={{add dest.y (div dest.height 2)}} />
{{/let}}
{{/if}}
{{/each}}
</svg>
<div class={{concat 'tooltip' (if activeTooltip ' active' '')}} style={{{ concat 'top:' y 'px;left:' x 'px;'}}}>
<span role="tooltip">{{round tooltip decimals=2}}%</span>
</div>
{{/if}}

View File

@ -1,5 +0,0 @@
import Component from '@ember/component';
export default Component.extend({
tagName: '',
});

View File

@ -1,5 +0,0 @@
<a class="splitter-card" onclick={{onclick}} id={{item.ID}}>
<header>
<h3>{{item.Name}}</h3>
</header>
</a>

View File

@ -1,3 +0,0 @@
import Component from '@ember/component';
export default Component.extend({});

View File

@ -29,7 +29,6 @@
@import './components/auth-form'; @import './components/auth-form';
@import './components/auth-modal'; @import './components/auth-modal';
@import './components/oidc-select'; @import './components/oidc-select';
@import './components/discovery-chain';
@import './components/empty-state'; @import './components/empty-state';
@import './components/tabular-details'; @import './components/tabular-details';
@import './components/tabular-collection'; @import './components/tabular-collection';
@ -56,6 +55,7 @@
@import 'consul-ui/components/notice'; @import 'consul-ui/components/notice';
@import 'consul-ui/components/modal-dialog'; @import 'consul-ui/components/modal-dialog';
@import 'consul-ui/components/consul/discovery-chain';
@import 'consul-ui/components/consul/exposed-path/list'; @import 'consul-ui/components/consul/exposed-path/list';
@import 'consul-ui/components/consul/external-source'; @import 'consul-ui/components/consul/external-source';
@import 'consul-ui/components/consul/kind'; @import 'consul-ui/components/consul/kind';

View File

@ -1,2 +0,0 @@
@import './skin';
@import './layout';

View File

@ -1,6 +1,8 @@
<div id="routing" class="tab-section"> <div id="routing" class="tab-section">
<div role="tabpanel"> <div role="tabpanel">
<DiscoveryChain @chain={{chain.Chain}} /> <Consul::DiscoveryChain
@chain={{chain.Chain}}
/>
</div> </div>
</div> </div>

View File

@ -1,7 +1,7 @@
import { getAlternateServices } from 'consul-ui/components/discovery-chain/utils'; import { getAlternateServices } from 'consul-ui/components/consul/discovery-chain/utils';
import { module, test } from 'qunit'; import { module, test } from 'qunit';
module('Unit | Component | discovery-chain/get-alternative-services', function() { module('Unit | Component | consul/discovery-chain/get-alternative-services', function() {
test('it guesses a different namespace', function(assert) { test('it guesses a different namespace', function(assert) {
const expected = { const expected = {
Type: 'Namespace', Type: 'Namespace',

View File

@ -1,4 +1,4 @@
import { getResolvers } from 'consul-ui/components/discovery-chain/utils'; import { getResolvers } from 'consul-ui/components/consul/discovery-chain/utils';
import { module, test } from 'qunit'; import { module, test } from 'qunit';
import { get } from 'consul-ui/tests/helpers/api'; import { get } from 'consul-ui/tests/helpers/api';
@ -7,7 +7,7 @@ const nspace = 'default';
const request = { const request = {
url: `/v1/discovery-chain/service-name?dc=${dc}`, url: `/v1/discovery-chain/service-name?dc=${dc}`,
}; };
module('Unit | Component | discovery-chain/get-resolvers', function() { module('Unit | Component | consul/discovery-chain/get-resolvers', function() {
test('it assigns Subsets correctly', function(assert) { test('it assigns Subsets correctly', function(assert) {
return get(request.url, { return get(request.url, {
headers: { headers: {

View File

@ -1,7 +1,7 @@
import { getSplitters } from 'consul-ui/components/discovery-chain/utils'; import { getSplitters } from 'consul-ui/components/consul/discovery-chain/utils';
import { module, test } from 'qunit'; import { module, test } from 'qunit';
module('Unit | Component | discovery-chain/get-splitters', function() { module('Unit | Component | consul/discovery-chain/get-splitters', function() {
test('it collects and correctly parses splitter Names', function(assert) { test('it collects and correctly parses splitter Names', function(assert) {
const actual = getSplitters({ const actual = getSplitters({
'splitter:splitter-name.default': { 'splitter:splitter-name.default': {