"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[108],{3905:(e,t,n)=>{n.d(t,{Zo:()=>u,kt:()=>d});var r=n(7294);function i(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function o(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,r)}return n}function a(e){for(var t=1;t=0||(i[n]=e[n]);return i}(e,t);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(i[n]=e[n])}return i}var s=r.createContext({}),c=function(e){var t=r.useContext(s),n=t;return e&&(n="function"==typeof e?e(t):a(a({},t),e)),n},u=function(e){var t=c(e.components);return r.createElement(s.Provider,{value:t},e.children)},p="mdxType",m={inlineCode:"code",wrapper:function(e){var t=e.children;return r.createElement(r.Fragment,{},t)}},g=r.forwardRef((function(e,t){var n=e.components,i=e.mdxType,o=e.originalType,s=e.parentName,u=l(e,["components","mdxType","originalType","parentName"]),p=c(n),g=i,d=p["".concat(s,".").concat(g)]||p[g]||m[g]||o;return n?r.createElement(d,a(a({ref:t},u),{},{components:n})):r.createElement(d,a({ref:t},u))}));function d(e,t){var n=arguments,i=t&&t.mdxType;if("string"==typeof e||i){var o=n.length,a=new Array(o);a[0]=g;var l={};for(var s in t)hasOwnProperty.call(t,s)&&(l[s]=t[s]);l.originalType=e,l[p]="string"==typeof e?e:i,a[1]=l;for(var c=2;c{n.r(t),n.d(t,{assets:()=>s,contentTitle:()=>a,default:()=>p,frontMatter:()=>o,metadata:()=>l,toc:()=>c});var r=n(7462),i=(n(7294),n(3905));const o={sidebar_position:20},a="Configuring Metrics",l={unversionedId:"guides/self-hosting/metrics-and-limits/configuring-metrics",id:"guides/self-hosting/metrics-and-limits/configuring-metrics",title:"Configuring Metrics",description:"A fully configured, production-scale zrok service instance looks like this:",source:"@site/../docs/guides/self-hosting/metrics-and-limits/configuring-metrics.md",sourceDirName:"guides/self-hosting/metrics-and-limits",slug:"/guides/self-hosting/metrics-and-limits/configuring-metrics",permalink:"/docs/guides/self-hosting/metrics-and-limits/configuring-metrics",draft:!1,editUrl:"https://github.com/openziti/zrok/blob/main/docs/../docs/guides/self-hosting/metrics-and-limits/configuring-metrics.md",tags:[],version:"current",sidebarPosition:20,frontMatter:{sidebar_position:20},sidebar:"tutorialSidebar",previous:{title:"Metrics and Limits",permalink:"/docs/category/metrics-and-limits"},next:{title:"Configuring Limits",permalink:"/docs/guides/self-hosting/metrics-and-limits/configuring-limits"}},s={},c=[{value:"Configuring the OpenZiti Controller",id:"configuring-the-openziti-controller",level:2},{value:"Configuring the zrok Metrics Bridge",id:"configuring-the-zrok-metrics-bridge",level:2},{value:"RabbitMQ",id:"rabbitmq",level:3},{value:"Configuring zrok Metrics",id:"configuring-zrok-metrics",level:2},{value:"Testing Metrics",id:"testing-metrics",level:2}],u={toc:c};function p(e){let{components:t,...o}=e;return(0,i.kt)("wrapper",(0,r.Z)({},u,o,{components:t,mdxType:"MDXLayout"}),(0,i.kt)("h1",{id:"configuring-metrics"},"Configuring Metrics"),(0,i.kt)("p",null,"A fully configured, production-scale ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok")," service instance looks like this:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"zrok Metrics Architecture",src:n(4755).Z,width:"381",height:"492"})),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"zrok")," metrics builds on top of the ",(0,i.kt)("inlineCode",{parentName:"p"},"fabric.usage")," event type from OpenZiti. The OpenZiti controller has a number of way to emit events. The ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok")," controller has several ways to consume ",(0,i.kt)("inlineCode",{parentName:"p"},"fabric.usage")," events. Smaller installations could be configured in these ways:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"zrok simplified metrics architecture",src:n(3521).Z,width:"522",height:"322"})),(0,i.kt)("p",null,"Environments that horizontally scale the ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok"),' control plane with multiple controllers should use an AMQP-based queue to "fan out" the metrics workload across the entire control plane. Simpler installations that use a single ',(0,i.kt)("inlineCode",{parentName:"p"},"zrok")," controller can collect ",(0,i.kt)("inlineCode",{parentName:"p"},"fabric.usage"),' events from the OpenZiti controller by "tailing" the events log file, or collecting them from the OpenZiti controller\'s websocket implementation.'),(0,i.kt)("h2",{id:"configuring-the-openziti-controller"},"Configuring the OpenZiti Controller"),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"This requires a version of OpenZiti with a ",(0,i.kt)("inlineCode",{parentName:"p"},"fabric")," dependency of ",(0,i.kt)("inlineCode",{parentName:"p"},"v0.22.52")," or newer, which is satisfed by the ",(0,i.kt)("inlineCode",{parentName:"p"},"v0.27.6")," release of OpenZiti Controller.")),(0,i.kt)("p",null,"Emitting ",(0,i.kt)("inlineCode",{parentName:"p"},"fabric.usage")," events to a file is currently the most reliable mechanism to capture usage events into ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok"),". We're going to configure the OpenZiti controller to append ",(0,i.kt)("inlineCode",{parentName:"p"},"fabric.usage")," events to a file, by adding this stanza to the OpenZiti controller configuration:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-yaml"},"events:\n jsonLogger:\n subscriptions:\n - type: fabric.usage\n version: 3\n handler:\n type: file\n format: json\n path: /tmp/fabric-usage.json\n")),(0,i.kt)("p",null,"You'll want to adjust the ",(0,i.kt)("inlineCode",{parentName:"p"},"events/jsonLogger/handler/path")," to wherever you would like to send these events for ingestion into ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok"),". There are additional OpenZiti options that control file rotation. Be sure to consult the OpenZiti docs to tune these settings to be appropriate for your environment."),(0,i.kt)("p",null,"By default, the OpenZiti events infrastructure reports and batches events in 1 minute buckets. 1 minute is too large of an interval to provide a snappy ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok")," metrics experience. So, let's increase the frequency to every 5 seconds. Add this to the ",(0,i.kt)("inlineCode",{parentName:"p"},"network")," stanza of your OpenZiti controller's configuration:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-yaml"},"network:\n intervalAgeThreshold: 5s \n metricsReportInterval: 5s\n")),(0,i.kt)("p",null,"And you'll want to add this stanza to the tail-end of the router configuration for every router on your OpenZiti network:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-yaml"},"metrics:\n reportInterval: 5s\n intervalAgeThreshold: 5s\n")),(0,i.kt)("p",null,"Be sure to restart all of the components of your OpenZiti network after making these configuration changes."),(0,i.kt)("h2",{id:"configuring-the-zrok-metrics-bridge"},"Configuring the zrok Metrics Bridge"),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"zrok"),' currently uses a "metrics bridge" component (running as a separate process) to consume the ',(0,i.kt)("inlineCode",{parentName:"p"},"fabric.usage")," events from the OpenZiti controller, and publish them onto an AMQP queue. Add a stanza like the following to your ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok")," controller configuration:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-yaml"},"bridge:\n source:\n type: fileSource\n path: /tmp/fabric-usage.json\n sink:\n type: amqpSink\n url: amqp://guest:guest@localhost:5672\n queue_name: events\n")),(0,i.kt)("p",null,"This configuration consumes the ",(0,i.kt)("inlineCode",{parentName:"p"},"fabric.usage")," events from the file we previously specified in our OpenZiti controller configuration, and publishes them onto an AMQP queue. "),(0,i.kt)("h3",{id:"rabbitmq"},"RabbitMQ"),(0,i.kt)("p",null,"For this example, we're going to use RabbitMQ as our AMQP implementation. The stock, default RabbitMQ configuration, launched as a ",(0,i.kt)("inlineCode",{parentName:"p"},"docker")," container will work just fine:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre"},"$ docker run -it --rm --name rabbitmq -p 5672:5672 -p 15672:15672 rabbitmq:3.11-management\n")),(0,i.kt)("p",null,"Once RabbitMQ is running, you can start the ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok")," metrics bridge by pointing it at your ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok")," controller configuration, like this:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre"},"$ zrok ctrl metrics bridge \n")),(0,i.kt)("h2",{id:"configuring-zrok-metrics"},"Configuring zrok Metrics"),(0,i.kt)("p",null,"Configure the ",(0,i.kt)("inlineCode",{parentName:"p"},"metrics")," section of your ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok")," controller. Here is an example:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-yaml"},'metrics:\n agent:\n source:\n type: amqpSource\n url: amqp://guest:guest@localhost:5672\n queue_name: events\n influx:\n url: "http://127.0.0.1:8086"\n bucket: zrok # the bucket and org must be\n org: zrok # created in advance in InfluxDB\n token: ""\n')),(0,i.kt)("p",null,"This configures the ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok")," controller to consume usage events from the AMQP queue, and configures the InfluxDB metrics store. The InfluxDB organization and bucket must be created in advance. The ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok")," controller will not create these for you."),(0,i.kt)("h2",{id:"testing-metrics"},"Testing Metrics"),(0,i.kt)("p",null,"With all of the components configured and running, either use ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok test loop")," or manually create share(s) to generate traffic on the ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok")," instance. If everything is working correctly, you should see log messages from the controller like the following, which indicate that that the controller is processing OpenZiti usage events, and generating ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok")," metrics:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre"},"[5339.658] INFO zrok/controller/metrics.(*influxWriter).Handle: share: 736z80mr4syu, circuit: Ad1V-6y48 backend {rx: 4.5 kB, tx: 4.6 kB} frontend {rx: 4.6 kB, tx: 4.5 kB}\n[5349.652] INFO zrok/controller/metrics.(*influxWriter).Handle: share: 736z80mr4syu, circuit: Ad1V-6y48 backend {rx: 2.5 kB, tx: 2.6 kB} frontend {rx: 2.6 kB, tx: 2.5 kB}\n[5354.657] INFO zrok/controller/metrics.(*influxWriter).Handle: share: 5a4u7lqxb7pa, circuit: iG1--6H4S backend {rx: 13.2 kB, tx: 13.3 kB} frontend {rx: 13.3 kB, tx: 13.2 kB}\n")),(0,i.kt)("p",null,"The ",(0,i.kt)("inlineCode",{parentName:"p"},"zrok")," web console should also be showing activity for your share(s) like the following:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"zrok web console activity",src:n(804).Z,width:"1920",height:"1230"})),(0,i.kt)("p",null,"With metrics configured, you might be interested in ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/self-hosting/metrics-and-limits/configuring-limits"},"configuring limits"),"."))}p.isMDXComponent=!0},3521:(e,t,n)=>{n.d(t,{Z:()=>r});const r=n.p+"assets/images/metrics-architecture-simple-15902678f75b6a41fc6d26c1b6165d48.png"},4755:(e,t,n)=>{n.d(t,{Z:()=>r});const r=n.p+"assets/images/metrics-architecture-abc07e1548198a0d2176bbfea70521fa.png"},804:(e,t,n)=>{n.d(t,{Z:()=>r});const r=n.p+"assets/images/zrok-console-activity-5e11ec1f9f3e58b54566c42a213e4ee5.png"}}]);