mirror of https://github.com/hashicorp/consul
Merge pull request #10002 from hashicorp/ui/feature/banners-and-labels-for-tproxy-changes
commit
daf897f1a9
|
@ -0,0 +1,3 @@
|
||||||
|
```ui:enhancement
|
||||||
|
Transparent Proxy - Service mesh visualization updates
|
||||||
|
```
|
|
@ -0,0 +1,3 @@
|
||||||
|
<span class="consul-transparent-proxy">
|
||||||
|
{{t "components.consul.transparent-proxy"}}
|
||||||
|
</span>
|
|
@ -2,6 +2,9 @@
|
||||||
& {
|
& {
|
||||||
min-width: 190px;
|
min-width: 190px;
|
||||||
}
|
}
|
||||||
|
&.documentation {
|
||||||
|
min-width: 270px;
|
||||||
|
}
|
||||||
> div {
|
> div {
|
||||||
padding: 1rem;
|
padding: 1rem;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,11 @@
|
||||||
<a class="card"
|
{{#if (eq @item.Name '* (All Services)')}}
|
||||||
|
<a class="topology-metrics-card" href={{href-to 'dc.services.index'}}>
|
||||||
|
<p class="empty">
|
||||||
|
{{@item.Name}}
|
||||||
|
</p>
|
||||||
|
</a>
|
||||||
|
{{else}}
|
||||||
|
<a class="topology-metrics-card"
|
||||||
href={{if
|
href={{if
|
||||||
(and (env 'CONSUL_NSPACES_ENABLED') (not-eq @item.Namespace @service.Namespace))
|
(and (env 'CONSUL_NSPACES_ENABLED') (not-eq @item.Namespace @service.Namespace))
|
||||||
(href-to "nspace.dc.services.show.index" (concat '~' @item.Namespace) @item.Datacenter @item.Name)
|
(href-to "nspace.dc.services.show.index" (concat '~' @item.Namespace) @item.Datacenter @item.Name)
|
||||||
|
@ -10,6 +17,9 @@
|
||||||
<p>
|
<p>
|
||||||
{{@item.Name}}
|
{{@item.Name}}
|
||||||
</p>
|
</p>
|
||||||
|
{{#if (eq @item.Source 'proxy-registration')}}
|
||||||
|
<TopologyMetrics::SourceType />
|
||||||
|
{{/if}}
|
||||||
<div class="details">
|
<div class="details">
|
||||||
{{#if (and (env 'CONSUL_NSPACES_ENABLED') (not-eq @item.Namespace @service.Namespace))}}
|
{{#if (and (env 'CONSUL_NSPACES_ENABLED') (not-eq @item.Namespace @service.Namespace))}}
|
||||||
<dl class="nspace">
|
<dl class="nspace">
|
||||||
|
@ -53,4 +63,5 @@
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</div>
|
</div>
|
||||||
{{yield}}
|
{{yield}}
|
||||||
</a>
|
</a>
|
||||||
|
{{/if}}
|
|
@ -0,0 +1,81 @@
|
||||||
|
#upstream-container .topology-metrics-card:not(:last-child),
|
||||||
|
#downstream-container .topology-metrics-card:not(:last-child) {
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
#upstream-container .topology-metrics-card,
|
||||||
|
#downstream-container .topology-metrics-card {
|
||||||
|
display: block;
|
||||||
|
color: $gray-700;
|
||||||
|
overflow: hidden;
|
||||||
|
background-color: $white;
|
||||||
|
border-radius: $decor-radius-100;
|
||||||
|
border: 1px solid $gray-200;
|
||||||
|
p {
|
||||||
|
padding: 12px 12px 0 12px;
|
||||||
|
font-size: $typo-size-500;
|
||||||
|
font-weight: $typo-weight-semibold;
|
||||||
|
margin-bottom: 0 !important;
|
||||||
|
}
|
||||||
|
p.empty {
|
||||||
|
padding: 12px !important;
|
||||||
|
}
|
||||||
|
div {
|
||||||
|
display: inline-flex;
|
||||||
|
dl {
|
||||||
|
display: inline-flex;
|
||||||
|
margin-right: 8px;
|
||||||
|
}
|
||||||
|
dd {
|
||||||
|
color: $gray-700;
|
||||||
|
}
|
||||||
|
span {
|
||||||
|
margin-right: 8px;
|
||||||
|
}
|
||||||
|
span::before,
|
||||||
|
dt::before {
|
||||||
|
margin-right: 4px;
|
||||||
|
}
|
||||||
|
.nspace dt::before,
|
||||||
|
.health dt::before {
|
||||||
|
margin-top: 2px;
|
||||||
|
}
|
||||||
|
.nspace dt::before {
|
||||||
|
@extend %with-folder-outline-mask, %as-pseudo;
|
||||||
|
}
|
||||||
|
.health dt::before {
|
||||||
|
@extend %with-help-circle-outline-mask, %as-pseudo;
|
||||||
|
}
|
||||||
|
.nspace dt::before {
|
||||||
|
@extend %with-folder-outline-mask, %as-pseudo;
|
||||||
|
}
|
||||||
|
.health dt::before {
|
||||||
|
@extend %with-help-circle-outline-mask, %as-pseudo;
|
||||||
|
}
|
||||||
|
.nspace dt::before,
|
||||||
|
.health dt::before {
|
||||||
|
background-color: $gray-500;
|
||||||
|
}
|
||||||
|
.passing::before {
|
||||||
|
@extend %with-check-circle-fill-color-mask, %as-pseudo;
|
||||||
|
background-color: $green-500;
|
||||||
|
}
|
||||||
|
.warning::before {
|
||||||
|
@extend %with-alert-triangle-color-mask, %as-pseudo;
|
||||||
|
background-color: $orange-500;
|
||||||
|
}
|
||||||
|
.critical::before {
|
||||||
|
@extend %with-cancel-square-fill-color-mask, %as-pseudo;
|
||||||
|
background-color: $red-500;
|
||||||
|
}
|
||||||
|
.empty::before {
|
||||||
|
@extend %with-minus-square-fill-mask, %as-pseudo;
|
||||||
|
color: $gray-500;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.details {
|
||||||
|
padding: 0 12px 12px 12px;
|
||||||
|
}
|
||||||
|
div.stats {
|
||||||
|
border-top: 1px solid $gray-200;
|
||||||
|
}
|
||||||
|
}
|
|
@ -90,6 +90,13 @@
|
||||||
@item={{item}}
|
@item={{item}}
|
||||||
@oncreate={{action @oncreate item @service}}
|
@oncreate={{action @oncreate item @service}}
|
||||||
/>
|
/>
|
||||||
|
{{else if (and item.Intention.Allowed (not item.TransparentProxy) (eq item.Source 'specific-intention'))}}
|
||||||
|
<TopologyMetrics::Popover
|
||||||
|
@type='not-defined'
|
||||||
|
@position={{find-by 'id' (concat this.guid item.Namespace item.Name) this.iconPositions}}
|
||||||
|
@item={{item}}
|
||||||
|
@oncreate={{action @oncreate item @service}}
|
||||||
|
/>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
{{/each}}
|
{{/each}}
|
||||||
|
|
||||||
|
|
|
@ -81,15 +81,17 @@
|
||||||
@oncreate={{action @oncreate}}
|
@oncreate={{action @oncreate}}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
{{#if (gt @topology.Upstreams.length 0)}}
|
{{#if (gt this.upstreams.length 0)}}
|
||||||
<div id="upstream-column">
|
<div id="upstream-column">
|
||||||
{{#each-in (group-by "Datacenter" @topology.Upstreams) as |dc upstreams|}}
|
{{#each-in (group-by "Datacenter" this.upstreams) as |dc upstreams|}}
|
||||||
<div
|
<div
|
||||||
id="upstream-container"
|
id="upstream-container"
|
||||||
{{did-insert this.setHeight 'upstream-lines'}}
|
{{did-insert this.setHeight 'upstream-lines'}}
|
||||||
{{did-update this.setHeight 'upstream-lines' @topology.Upstreams}}
|
{{did-update this.setHeight 'upstream-lines' this.upstreams}}
|
||||||
>
|
>
|
||||||
|
{{#if dc}}
|
||||||
<p>{{dc}}</p>
|
<p>{{dc}}</p>
|
||||||
|
{{/if}}
|
||||||
{{#each upstreams as |item|}}
|
{{#each upstreams as |item|}}
|
||||||
<TopologyMetrics::Card
|
<TopologyMetrics::Card
|
||||||
@dc={{@dc}}
|
@dc={{@dc}}
|
||||||
|
@ -119,7 +121,7 @@
|
||||||
@view={{this.upView}}
|
@view={{this.upView}}
|
||||||
@center={{this.centerDimensions}}
|
@center={{this.centerDimensions}}
|
||||||
@lines={{this.upLines}}
|
@lines={{this.upLines}}
|
||||||
@items={{@topology.Upstreams}}
|
@items={{this.upstreams}}
|
||||||
@oncreate={{action @oncreate}}
|
@oncreate={{action @oncreate}}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import Component from '@glimmer/component';
|
import Component from '@glimmer/component';
|
||||||
import { tracked } from '@glimmer/tracking';
|
import { tracked } from '@glimmer/tracking';
|
||||||
import { action } from '@ember/object';
|
import { action, get } from '@ember/object';
|
||||||
|
|
||||||
export default class TopologyMetrics extends Component {
|
export default class TopologyMetrics extends Component {
|
||||||
// =attributes
|
// =attributes
|
||||||
|
@ -66,6 +66,24 @@ export default class TopologyMetrics extends Component {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get upstreams() {
|
||||||
|
const upstreams = get(this.args.topology, 'Upstreams') || [];
|
||||||
|
const items = [...upstreams];
|
||||||
|
const defaultAllow = get(this.args.topology, 'DefaultAllow');
|
||||||
|
const wildcardIntention = get(this.args.topology, 'WildcardIntention');
|
||||||
|
if (defaultAllow || wildcardIntention) {
|
||||||
|
items.push({
|
||||||
|
Name: '* (All Services)',
|
||||||
|
Datacenter: '',
|
||||||
|
Namespace: '',
|
||||||
|
Intention: {
|
||||||
|
Allowed: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return items;
|
||||||
|
}
|
||||||
|
|
||||||
// =actions
|
// =actions
|
||||||
@action
|
@action
|
||||||
setHeight(el, item) {
|
setHeight(el, item) {
|
||||||
|
@ -89,12 +107,22 @@ export default class TopologyMetrics extends Component {
|
||||||
|
|
||||||
// Calculate viewBox dimensions
|
// Calculate viewBox dimensions
|
||||||
this.downView = document.getElementById('downstream-lines').getBoundingClientRect();
|
this.downView = document.getElementById('downstream-lines').getBoundingClientRect();
|
||||||
this.upView = document.getElementById('upstream-lines').getBoundingClientRect();
|
const upstreamLines = document.getElementById('upstream-lines').getBoundingClientRect();
|
||||||
|
const upstreamColumn = document.getElementById('upstream-column').getBoundingClientRect();
|
||||||
|
|
||||||
|
this.upView = {
|
||||||
|
x: upstreamLines.x,
|
||||||
|
y: upstreamLines.y,
|
||||||
|
width: upstreamLines.width,
|
||||||
|
height: upstreamColumn.height,
|
||||||
|
};
|
||||||
|
|
||||||
// Get Card elements positions
|
// Get Card elements positions
|
||||||
const downCards = [...document.querySelectorAll('#downstream-container .card')];
|
const downCards = [
|
||||||
|
...document.querySelectorAll('#downstream-container .topology-metrics-card'),
|
||||||
|
];
|
||||||
const grafanaCard = document.querySelector('.metrics-header');
|
const grafanaCard = document.querySelector('.metrics-header');
|
||||||
const upCards = [...document.querySelectorAll('#upstream-column .card')];
|
const upCards = [...document.querySelectorAll('#upstream-column .topology-metrics-card')];
|
||||||
|
|
||||||
// Set center positioning points
|
// Set center positioning points
|
||||||
this.centerDimensions = grafanaCard.getBoundingClientRect();
|
this.centerDimensions = grafanaCard.getBoundingClientRect();
|
||||||
|
|
|
@ -49,44 +49,6 @@
|
||||||
#upstream-column #upstream-container:not(:last-child) {
|
#upstream-column #upstream-container:not(:last-child) {
|
||||||
margin-bottom: 8px;
|
margin-bottom: 8px;
|
||||||
}
|
}
|
||||||
#upstream-container .card:not(:last-child),
|
|
||||||
#downstream-container .card:not(:last-child) {
|
|
||||||
margin-bottom: 8px;
|
|
||||||
}
|
|
||||||
#upstream-container .card,
|
|
||||||
#downstream-container .card {
|
|
||||||
display: block;
|
|
||||||
color: $gray-700;
|
|
||||||
overflow: hidden;
|
|
||||||
p {
|
|
||||||
padding: 12px 12px 0 12px;
|
|
||||||
font-size: 16px;
|
|
||||||
font-weight: 600;
|
|
||||||
margin-bottom: 0 !important;
|
|
||||||
}
|
|
||||||
div {
|
|
||||||
display: inline-flex;
|
|
||||||
dl {
|
|
||||||
display: inline-flex;
|
|
||||||
margin-right: 8px;
|
|
||||||
}
|
|
||||||
span {
|
|
||||||
margin-right: 8px;
|
|
||||||
}
|
|
||||||
span::before,
|
|
||||||
dt::before {
|
|
||||||
margin-right: 4px;
|
|
||||||
}
|
|
||||||
.nspace dt::before,
|
|
||||||
.health dt::before {
|
|
||||||
margin-top: 2px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.details {
|
|
||||||
padding: 0 12px 12px 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// Metrics Container
|
// Metrics Container
|
||||||
#metrics-container {
|
#metrics-container {
|
||||||
|
|
|
@ -0,0 +1,31 @@
|
||||||
|
<Notice
|
||||||
|
class="topology-metrics-notice"
|
||||||
|
...attributes
|
||||||
|
@type={{@type}}
|
||||||
|
as |notice|>
|
||||||
|
<notice.Header>
|
||||||
|
<h3>
|
||||||
|
{{t (concat "components.consul.topology-metrics.notice." @for ".header")}}
|
||||||
|
</h3>
|
||||||
|
</notice.Header>
|
||||||
|
<notice.Body>
|
||||||
|
<p>
|
||||||
|
{{t (concat "components.consul.topology-metrics.notice." @for ".body")}}
|
||||||
|
</p>
|
||||||
|
</notice.Body>
|
||||||
|
{{#if @action}}
|
||||||
|
<notice.Footer>
|
||||||
|
<p>
|
||||||
|
{{#if @internal}}
|
||||||
|
<Action @href={{href-to (t (concat "components.consul.topology-metrics.notice." @for ".footer.URL"))}}>
|
||||||
|
{{t (concat "components.consul.topology-metrics.notice." @for ".footer.name")}}
|
||||||
|
</Action>
|
||||||
|
{{else}}
|
||||||
|
<Action @href={{@link}} @external={{true}}>
|
||||||
|
{{t (concat "components.consul.topology-metrics.notice." @for ".footer")}}
|
||||||
|
</Action>
|
||||||
|
{{/if}}
|
||||||
|
</p>
|
||||||
|
</notice.Footer>
|
||||||
|
{{/if}}
|
||||||
|
</Notice>
|
|
@ -1,16 +0,0 @@
|
||||||
<Notice
|
|
||||||
class="topology-metrics-notice-limited-access"
|
|
||||||
...attributes
|
|
||||||
@type={{or @type "info"}}
|
|
||||||
as |notice|>
|
|
||||||
<notice.Header>
|
|
||||||
<h3>
|
|
||||||
Limited Access
|
|
||||||
</h3>
|
|
||||||
</notice.Header>
|
|
||||||
<notice.Body>
|
|
||||||
<p>
|
|
||||||
This service may have dependencies you won’t see because you don’t have access to them.
|
|
||||||
</p>
|
|
||||||
</notice.Body>
|
|
||||||
</Notice>
|
|
|
@ -4,22 +4,21 @@
|
||||||
>
|
>
|
||||||
|
|
||||||
{{#if (eq @type 'deny')}}
|
{{#if (eq @type 'deny')}}
|
||||||
|
|
||||||
<InformedAction
|
<InformedAction
|
||||||
class="dangerous"
|
class="dangerous"
|
||||||
{{did-insert (set this 'popover')}}
|
{{did-insert (set this 'popover')}}
|
||||||
>
|
>
|
||||||
<:header>
|
<:header>
|
||||||
<h3>
|
<h3>
|
||||||
Connection Denied
|
{{t "components.consul.topology-metrics.popover.deny.header"}}
|
||||||
</h3>
|
</h3>
|
||||||
</:header>
|
</:header>
|
||||||
<:body>
|
<:body>
|
||||||
<p>
|
<p>
|
||||||
{{#if @item.Intention.HasExact}}
|
{{#if @item.Intention.HasExact}}
|
||||||
Change the action of this intention to allow.
|
{{t "components.consul.topology-metrics.popover.deny.body.isExact"}}
|
||||||
{{else}}
|
{{else}}
|
||||||
Add an intention that allows these two services to connect.
|
{{t "components.consul.topology-metrics.popover.deny.body.notExact"}}
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</p>
|
</p>
|
||||||
</:body>
|
</:body>
|
||||||
|
@ -31,9 +30,9 @@
|
||||||
type="button"
|
type="button"
|
||||||
>
|
>
|
||||||
{{#if @item.Intention.HasExact}}
|
{{#if @item.Intention.HasExact}}
|
||||||
Allow
|
{{t "components.consul.topology-metrics.popover.deny.action.isExact"}}
|
||||||
{{else}}
|
{{else}}
|
||||||
Create
|
{{t "components.consul.topology-metrics.popover.deny.action.notExact"}}
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</button>
|
</button>
|
||||||
</Actions.Action>
|
</Actions.Action>
|
||||||
|
@ -48,29 +47,57 @@
|
||||||
</Actions.Action>
|
</Actions.Action>
|
||||||
</:actions>
|
</:actions>
|
||||||
</InformedAction>
|
</InformedAction>
|
||||||
|
{{else if (eq @type 'not-defined')}}
|
||||||
{{else}}
|
|
||||||
|
|
||||||
<InformedAction
|
<InformedAction
|
||||||
class="info"
|
class="warning documentation"
|
||||||
{{did-insert (set this 'popover')}}
|
{{did-insert (set this 'popover')}}
|
||||||
>
|
>
|
||||||
<:header>
|
<:header>
|
||||||
<h3>
|
<h3>
|
||||||
Layer 7 permissions
|
{{t "components.consul.topology-metrics.popover.not-defined.header"}}
|
||||||
</h3>
|
</h3>
|
||||||
</:header>
|
</:header>
|
||||||
<:body>
|
<:body>
|
||||||
<p>
|
<p>
|
||||||
Certain HTTP request info must be identified.
|
{{t "components.consul.topology-metrics.popover.not-defined.body"}}
|
||||||
</p>
|
</p>
|
||||||
</:body>
|
</:body>
|
||||||
<:actions as |Actions|>
|
<:actions as |Actions|>
|
||||||
<Actions.Action class="action">
|
<Actions.Action class="action">
|
||||||
<a
|
<a href="{{env 'CONSUL_DOCS_URL'}}/connect/registration/service-registration#upstreams" rel="noopener noreferrer" target="_blank">
|
||||||
href={{href-to 'dc.services.show.intentions.edit' (concat @item.Intention.ID)}}
|
{{t "components.consul.topology-metrics.popover.not-defined.action"}}
|
||||||
>
|
</a>
|
||||||
View
|
</Actions.Action>
|
||||||
|
<Actions.Action>
|
||||||
|
<button
|
||||||
|
{{on 'click' (fn (optional this.popoverController.hide))}}
|
||||||
|
class="cancel"
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
Close
|
||||||
|
</button>
|
||||||
|
</Actions.Action>
|
||||||
|
</:actions>
|
||||||
|
</InformedAction>
|
||||||
|
{{else}}
|
||||||
|
<InformedAction
|
||||||
|
class="info"
|
||||||
|
{{did-insert (set this 'popover')}}
|
||||||
|
>
|
||||||
|
<:header>
|
||||||
|
<h3>
|
||||||
|
{{t "components.consul.topology-metrics.popover.l7.header"}}
|
||||||
|
</h3>
|
||||||
|
</:header>
|
||||||
|
<:body>
|
||||||
|
<p>
|
||||||
|
{{t "components.consul.topology-metrics.popover.l7.body"}}
|
||||||
|
</p>
|
||||||
|
</:body>
|
||||||
|
<:actions as |Actions|>
|
||||||
|
<Actions.Action class="action">
|
||||||
|
<a href={{href-to 'dc.services.show.intentions.edit' (concat @item.Intention.ID)}}>
|
||||||
|
{{t "components.consul.topology-metrics.popover.l7.action"}}
|
||||||
</a>
|
</a>
|
||||||
</Actions.Action>
|
</Actions.Action>
|
||||||
<Actions.Action>
|
<Actions.Action>
|
||||||
|
@ -84,7 +111,6 @@
|
||||||
</Actions.Action>
|
</Actions.Action>
|
||||||
</:actions>
|
</:actions>
|
||||||
</InformedAction>
|
</InformedAction>
|
||||||
|
|
||||||
{{/if}}
|
{{/if}}
|
||||||
<button
|
<button
|
||||||
{{with-overlay
|
{{with-overlay
|
||||||
|
|
|
@ -27,4 +27,9 @@
|
||||||
@extend %with-layers-mask, %as-pseudo;
|
@extend %with-layers-mask, %as-pseudo;
|
||||||
background-color: $gray-300;
|
background-color: $gray-300;
|
||||||
}
|
}
|
||||||
|
&.not-defined > button::before,
|
||||||
|
&.not-defined .tippy-arrow::after {
|
||||||
|
@extend %with-alert-triangle-mask, %as-pseudo;
|
||||||
|
color: $yellow-500;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,52 +20,6 @@
|
||||||
background-color: $gray-500;
|
background-color: $gray-500;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#upstream-container .card,
|
|
||||||
#downstream-container .card {
|
|
||||||
background-color: $white;
|
|
||||||
border-radius: $decor-radius-100;
|
|
||||||
border: 1px solid $gray-200;
|
|
||||||
div {
|
|
||||||
dd {
|
|
||||||
color: $gray-700;
|
|
||||||
}
|
|
||||||
.nspace dt::before {
|
|
||||||
@extend %with-folder-outline-mask, %as-pseudo;
|
|
||||||
}
|
|
||||||
.health dt::before {
|
|
||||||
@extend %with-help-circle-outline-mask, %as-pseudo;
|
|
||||||
}
|
|
||||||
.nspace dt::before {
|
|
||||||
@extend %with-folder-outline-mask, %as-pseudo;
|
|
||||||
}
|
|
||||||
.health dt::before {
|
|
||||||
@extend %with-help-circle-outline-mask, %as-pseudo;
|
|
||||||
}
|
|
||||||
.nspace dt::before,
|
|
||||||
.health dt::before {
|
|
||||||
background-color: $gray-500;
|
|
||||||
}
|
|
||||||
.passing::before {
|
|
||||||
@extend %with-check-circle-fill-color-mask, %as-pseudo;
|
|
||||||
background-color: $green-500;
|
|
||||||
}
|
|
||||||
.warning::before {
|
|
||||||
@extend %with-alert-triangle-color-mask, %as-pseudo;
|
|
||||||
background-color: $orange-500;
|
|
||||||
}
|
|
||||||
.critical::before {
|
|
||||||
@extend %with-cancel-square-fill-color-mask, %as-pseudo;
|
|
||||||
background-color: $red-500;
|
|
||||||
}
|
|
||||||
.empty::before {
|
|
||||||
@extend %with-minus-square-fill-mask, %as-pseudo;
|
|
||||||
color: $gray-500;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
div:nth-child(3) {
|
|
||||||
border-top: 1px solid $gray-200;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Metrics Container
|
// Metrics Container
|
||||||
#metrics-container {
|
#metrics-container {
|
||||||
|
@ -90,9 +44,6 @@
|
||||||
@extend %with-docs-mask, %as-pseudo;
|
@extend %with-docs-mask, %as-pseudo;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
div:nth-child(3) {
|
|
||||||
border-top: 1px solid $gray-200;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// SVG Line styling
|
// SVG Line styling
|
||||||
|
@ -114,6 +65,9 @@
|
||||||
stroke: $gray-300;
|
stroke: $gray-300;
|
||||||
stroke-width: 2;
|
stroke-width: 2;
|
||||||
}
|
}
|
||||||
|
path[data-permission='not-defined'] {
|
||||||
|
stroke-dasharray: 4;
|
||||||
|
}
|
||||||
path[data-permission='deny'] {
|
path[data-permission='deny'] {
|
||||||
stroke: $red-500;
|
stroke: $red-500;
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,6 @@
|
||||||
|
<span
|
||||||
|
class="topology-metrics-source-type"
|
||||||
|
{{tooltip (t "components.consul.topology-metrics.source-type.tooltip")}}
|
||||||
|
>
|
||||||
|
{{t "components.consul.topology-metrics.source-type.text"}}
|
||||||
|
</span>
|
|
@ -0,0 +1,4 @@
|
||||||
|
.topology-metrics-source-type {
|
||||||
|
margin: 6px 0 6px 12px;
|
||||||
|
display: table;
|
||||||
|
}
|
|
@ -3,12 +3,15 @@ import { helper } from '@ember/component/helper';
|
||||||
export default helper(function serviceIntentionPermissions([params] /*, hash*/) {
|
export default helper(function serviceIntentionPermissions([params] /*, hash*/) {
|
||||||
const hasPermissions = params.Intention.HasPermissions;
|
const hasPermissions = params.Intention.HasPermissions;
|
||||||
const allowed = params.Intention.Allowed;
|
const allowed = params.Intention.Allowed;
|
||||||
|
const notExplicitlyDefined = params.Source === 'specific-intention' && !params.TransparentProxy;
|
||||||
|
|
||||||
switch (true) {
|
switch (true) {
|
||||||
case hasPermissions:
|
case hasPermissions:
|
||||||
return 'allow';
|
return 'allow';
|
||||||
case !allowed && !hasPermissions:
|
case !allowed && !hasPermissions:
|
||||||
return 'deny';
|
return 'deny';
|
||||||
|
case allowed && notExplicitlyDefined:
|
||||||
|
return 'not-defined';
|
||||||
default:
|
default:
|
||||||
return 'allow';
|
return 'allow';
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import Model, { attr } from '@ember-data/model';
|
import Model, { attr } from '@ember-data/model';
|
||||||
|
import { computed } from '@ember/object';
|
||||||
|
|
||||||
export const PRIMARY_KEY = 'uid';
|
export const PRIMARY_KEY = 'uid';
|
||||||
export const SLUG_KEY = 'ServiceName';
|
export const SLUG_KEY = 'ServiceName';
|
||||||
|
@ -11,7 +12,30 @@ export default class Topology extends Model {
|
||||||
@attr('string') Namespace;
|
@attr('string') Namespace;
|
||||||
@attr('string') Protocol;
|
@attr('string') Protocol;
|
||||||
@attr('boolean') FilteredByACLs;
|
@attr('boolean') FilteredByACLs;
|
||||||
|
@attr('boolean') TransparentProxy;
|
||||||
|
@attr('boolean') DefaultAllow;
|
||||||
|
@attr('boolean') WildcardIntention;
|
||||||
@attr() Upstreams; // Service[]
|
@attr() Upstreams; // Service[]
|
||||||
@attr() Downstreams; // Service[],
|
@attr() Downstreams; // Service[],
|
||||||
@attr() meta; // {}
|
@attr() meta; // {}
|
||||||
|
|
||||||
|
@computed('Upstreams', 'Downstreams')
|
||||||
|
get undefinedIntention() {
|
||||||
|
let undefinedUpstream = false;
|
||||||
|
let undefinedDownstream = false;
|
||||||
|
|
||||||
|
undefinedUpstream =
|
||||||
|
this.Upstreams.filter(
|
||||||
|
item =>
|
||||||
|
item.Source === 'specific-intention' && !item.TransparentProxy && item.Intention.Allowed
|
||||||
|
).length !== 0;
|
||||||
|
|
||||||
|
undefinedDownstream =
|
||||||
|
this.Downstreams.filter(
|
||||||
|
item =>
|
||||||
|
item.Source === 'specific-intention' && !item.TransparentProxy && item.Intention.Allowed
|
||||||
|
).length !== 0;
|
||||||
|
|
||||||
|
return undefinedUpstream || undefinedDownstream;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -79,6 +79,8 @@
|
||||||
|
|
||||||
@import 'consul-ui/components/role-selector';
|
@import 'consul-ui/components/role-selector';
|
||||||
@import 'consul-ui/components/topology-metrics';
|
@import 'consul-ui/components/topology-metrics';
|
||||||
|
@import 'consul-ui/components/topology-metrics/card';
|
||||||
|
@import 'consul-ui/components/topology-metrics/source-type';
|
||||||
@import 'consul-ui/components/topology-metrics/popover';
|
@import 'consul-ui/components/topology-metrics/popover';
|
||||||
@import 'consul-ui/components/topology-metrics/series';
|
@import 'consul-ui/components/topology-metrics/series';
|
||||||
@import 'consul-ui/components/topology-metrics/stats';
|
@import 'consul-ui/components/topology-metrics/stats';
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
span.policy-service-identity,
|
span.policy-service-identity,
|
||||||
span.policy-node-identity,
|
span.policy-node-identity,
|
||||||
.leader,
|
.leader,
|
||||||
.consul-auth-method-type {
|
.consul-auth-method-type,
|
||||||
|
.topology-metrics-source-type,
|
||||||
|
.consul-transparent-proxy {
|
||||||
@extend %pill-200, %frame-gray-600;
|
@extend %pill-200, %frame-gray-600;
|
||||||
}
|
}
|
||||||
span.policy-service-identity::before,
|
span.policy-service-identity::before,
|
||||||
|
|
|
@ -58,6 +58,9 @@ as |route|>
|
||||||
</h1>
|
</h1>
|
||||||
<Consul::ExternalSource @item={{item}} />
|
<Consul::ExternalSource @item={{item}} />
|
||||||
<Consul::Kind @item={{item}} @withInfo={{true}} />
|
<Consul::Kind @item={{item}} @withInfo={{true}} />
|
||||||
|
{{#if (eq proxy.ServiceProxy.Mode 'transparent')}}
|
||||||
|
<Consul::TransparentProxy />
|
||||||
|
{{/if}}
|
||||||
</BlockSlot>
|
</BlockSlot>
|
||||||
<BlockSlot @name="nav">
|
<BlockSlot @name="nav">
|
||||||
<dl>
|
<dl>
|
||||||
|
|
|
@ -23,7 +23,36 @@ as |route|>
|
||||||
</EmptyState>
|
</EmptyState>
|
||||||
{{else}}
|
{{else}}
|
||||||
{{#if topology.FilteredByACLs}}
|
{{#if topology.FilteredByACLs}}
|
||||||
<TopologyMetrics::Notice::LimitedAccess />
|
<TopologyMetrics::Notice
|
||||||
|
@type="info"
|
||||||
|
@for="limited-access"
|
||||||
|
@action={{false}}
|
||||||
|
/>
|
||||||
|
{{/if}}
|
||||||
|
{{#if topology.DefaultAllow}}
|
||||||
|
<TopologyMetrics::Notice
|
||||||
|
@type="warning"
|
||||||
|
@for="default-allow"
|
||||||
|
@internal={{true}}
|
||||||
|
@action={{true}}
|
||||||
|
/>
|
||||||
|
{{/if}}
|
||||||
|
{{#if topology.WildcardIntention}}
|
||||||
|
<TopologyMetrics::Notice
|
||||||
|
@type="warning"
|
||||||
|
@for="wildcard-intention"
|
||||||
|
@internal={{true}}
|
||||||
|
@action={{true}}
|
||||||
|
/>
|
||||||
|
{{/if}}
|
||||||
|
{{#if topology.undefinedIntention}}
|
||||||
|
<TopologyMetrics::Notice
|
||||||
|
@type="warning"
|
||||||
|
@for="not-defined-intention"
|
||||||
|
@link="{{env 'CONSUL_DOCS_URL'}}/connect/registration/service-registration#upstreams"
|
||||||
|
@internal={{false}}
|
||||||
|
@action={{true}}
|
||||||
|
/>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
<TopologyMetrics
|
<TopologyMetrics
|
||||||
@nspace={{nspace}}
|
@nspace={{nspace}}
|
||||||
|
|
|
@ -49,6 +49,7 @@ ${range(env('CONSUL_EXPOSED_COUNT', 3)).map((i) => `
|
||||||
`)}
|
`)}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
"Mode": "${fake.helpers.randomize(['', 'direct', 'transparent'])}",
|
||||||
"DestinationServiceName": "${location.pathname.slice(4)}"
|
"DestinationServiceName": "${location.pathname.slice(4)}"
|
||||||
${ location.pathname.slice(4) === "service-0" ? `
|
${ location.pathname.slice(4) === "service-0" ? `
|
||||||
,
|
,
|
||||||
|
|
|
@ -52,10 +52,18 @@ ${
|
||||||
}
|
}
|
||||||
fake.seed(index);
|
fake.seed(index);
|
||||||
|
|
||||||
|
|
||||||
|
// Randomly make permissive intentions
|
||||||
|
const defaultAllow = fake.random.boolean();
|
||||||
|
const wildcardIntention = defaultAllow ? false : fake.random.boolean();
|
||||||
|
|
||||||
return `
|
return `
|
||||||
{
|
{
|
||||||
"Protocol": "${serviceProto}",
|
"Protocol": "${serviceProto}",
|
||||||
"FilteredByACLs": ${fake.random.boolean()},
|
"FilteredByACLs": ${fake.random.boolean()},
|
||||||
|
"TransparentProxy": ${fake.random.boolean()},
|
||||||
|
"DefaultAllow": ${defaultAllow},
|
||||||
|
"WildcardIntention": ${wildcardIntention},
|
||||||
"Upstreams": [
|
"Upstreams": [
|
||||||
${
|
${
|
||||||
upstreams.map((item, i) => {
|
upstreams.map((item, i) => {
|
||||||
|
@ -70,11 +78,14 @@ ${
|
||||||
"ChecksPassing":${fake.random.number({min: 1, max: env('CONSUL_CHECK_COUNT', fake.random.number(10))})},
|
"ChecksPassing":${fake.random.number({min: 1, max: env('CONSUL_CHECK_COUNT', fake.random.number(10))})},
|
||||||
"ChecksWarning":${fake.random.number({min: 0, max: env('CONSUL_CHECK_COUNT', fake.random.number(10))})},
|
"ChecksWarning":${fake.random.number({min: 0, max: env('CONSUL_CHECK_COUNT', fake.random.number(10))})},
|
||||||
"ChecksCritical":${fake.random.number({min: 0, max: env('CONSUL_CHECK_COUNT', fake.random.number(10))})},
|
"ChecksCritical":${fake.random.number({min: 0, max: env('CONSUL_CHECK_COUNT', fake.random.number(10))})},
|
||||||
|
"Source": "${fake.helpers.randomize(['proxy-registration', 'default-allow', 'wildcard-intention'])}",
|
||||||
|
"TransparentProxy": ${fake.random.boolean()},
|
||||||
"Intention": {
|
"Intention": {
|
||||||
"Allowed": ${allowed},
|
"Allowed": ${allowed},
|
||||||
"HasPermissions": ${hasPerms},
|
"HasPermissions": ${hasPerms},
|
||||||
"ExternalSource": "${fake.helpers.randomize(['nomad', 'kubernetes', ''])}",
|
"ExternalSource": "${fake.helpers.randomize(['nomad', 'kubernetes', ''])}",
|
||||||
"HasExact": ${fake.random.boolean()}
|
"HasExact": ${fake.random.boolean()},
|
||||||
|
"DefaultAllow": ${fake.random.boolean()}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
`})}
|
`})}
|
||||||
|
@ -93,11 +104,14 @@ ${
|
||||||
"ChecksPassing":${fake.random.number({min: 1, max: env('CONSUL_CHECK_COUNT', fake.random.number(10))})},
|
"ChecksPassing":${fake.random.number({min: 1, max: env('CONSUL_CHECK_COUNT', fake.random.number(10))})},
|
||||||
"ChecksWarning":${fake.random.number({min: 0, max: env('CONSUL_CHECK_COUNT', fake.random.number(10))})},
|
"ChecksWarning":${fake.random.number({min: 0, max: env('CONSUL_CHECK_COUNT', fake.random.number(10))})},
|
||||||
"ChecksCritical":${fake.random.number({min: 0, max: env('CONSUL_CHECK_COUNT', fake.random.number(10))})},
|
"ChecksCritical":${fake.random.number({min: 0, max: env('CONSUL_CHECK_COUNT', fake.random.number(10))})},
|
||||||
|
"Source": "${fake.helpers.randomize(['proxy-registration', 'specific-intention', 'default-allow', 'wildcard-intention'])}",
|
||||||
|
"TransparentProxy": ${fake.random.boolean()},
|
||||||
"Intention": {
|
"Intention": {
|
||||||
"Allowed": ${allowed},
|
"Allowed": ${allowed},
|
||||||
"HasPermissions": ${hasPerms},
|
"HasPermissions": ${hasPerms},
|
||||||
"ExternalSource": "${fake.helpers.randomize(['nomad', 'kubernetes', ''])}",
|
"ExternalSource": "${fake.helpers.randomize(['nomad', 'kubernetes', ''])}",
|
||||||
"HasExact": ${fake.random.boolean()}
|
"HasExact": ${fake.random.boolean()},
|
||||||
|
"DefaultAllow": ${fake.random.boolean()}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
`})}
|
`})}
|
||||||
|
|
|
@ -184,6 +184,51 @@ components:
|
||||||
name: Precedence
|
name: Precedence
|
||||||
asc: Ascending
|
asc: Ascending
|
||||||
desc: Descending
|
desc: Descending
|
||||||
|
transparent-proxy: Transparent Proxy
|
||||||
|
topology-metrics:
|
||||||
|
source-type:
|
||||||
|
tooltip: This connection was defined in a proxy registration.
|
||||||
|
text: Defined in proxy registration
|
||||||
|
notice:
|
||||||
|
limited-access:
|
||||||
|
header: Limited Access
|
||||||
|
body: This service may have dependencies you won’t see because you don’t have access to them.
|
||||||
|
default-allow:
|
||||||
|
header: Intentions are set to default allow
|
||||||
|
body: Your Intention settings are currently set to default allow. This means that this view will show connections to every service in your cluster. We recommend changing your Intention settings to default deny and creating specific Intentions for upstream and downstream services for this view to be useful.
|
||||||
|
footer:
|
||||||
|
name: Edit intentions
|
||||||
|
URL: dc.services.show.intentions
|
||||||
|
not-defined-intention:
|
||||||
|
header: Connections are not explicitly defined
|
||||||
|
body: There appears to be an Intention defining traffic, but the services are unable to communicate until that connection is explicitly defined as a downstream or Transparent Proxy mode is turned on.
|
||||||
|
footer: Read the documentation
|
||||||
|
wildcard-intention:
|
||||||
|
header: Permissive Intention
|
||||||
|
body: One or more of your Intentions are set to allow traffic to and/or from all other services in a namespace. This Topology view will show all of those connections if that remains unchanged. We recommend setting more specific Intentions for upstream and downstream services to make this vizualization more useful.
|
||||||
|
footer:
|
||||||
|
name: Edit intentions
|
||||||
|
URL: dc.services.show.intentions
|
||||||
|
popover:
|
||||||
|
l7:
|
||||||
|
header: Layer 7 permissions
|
||||||
|
body: Certain HTTP request info must be identified.
|
||||||
|
action: View
|
||||||
|
deny:
|
||||||
|
header: Connection Denied
|
||||||
|
body:
|
||||||
|
isExact: Change the action of this intention to allow.
|
||||||
|
notExact: Add an intention that allows these two services to connect.
|
||||||
|
action:
|
||||||
|
isExact: Allow
|
||||||
|
notExact: Create
|
||||||
|
not-defined:
|
||||||
|
header: No traffic
|
||||||
|
body: Add the current service as an explicit upstream or turn on Transparent Proxy mode to initiate traffic.
|
||||||
|
action: Documentation
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
models:
|
models:
|
||||||
auth-method:
|
auth-method:
|
||||||
|
|
Loading…
Reference in New Issue