diff options
-rwxr-xr-x | doc/analyticsBlog-docinfo.html | 0 | ||||
-rwxr-xr-x | doc/analyticsBlog.asciidoc | 306 | ||||
-rwxr-xr-x | doc/images/blog/figure1.jpg | bin | 0 -> 53340 bytes | |||
-rwxr-xr-x | doc/images/blog/figure10.JPG | bin | 0 -> 52220 bytes | |||
-rwxr-xr-x | doc/images/blog/figure11.jpg | bin | 0 -> 56349 bytes | |||
-rwxr-xr-x | doc/images/blog/figure12.png | bin | 0 -> 22723 bytes | |||
-rwxr-xr-x | doc/images/blog/figure13.JPG | bin | 0 -> 82198 bytes | |||
-rwxr-xr-x | doc/images/blog/figure14.JPG | bin | 0 -> 69046 bytes | |||
-rwxr-xr-x | doc/images/blog/figure15.jpg | bin | 0 -> 87012 bytes | |||
-rwxr-xr-x | doc/images/blog/figure16.JPG | bin | 0 -> 50636 bytes | |||
-rwxr-xr-x | doc/images/blog/figure17.JPG | bin | 0 -> 52517 bytes | |||
-rwxr-xr-x | doc/images/blog/figure18.JPG | bin | 0 -> 18761 bytes | |||
-rwxr-xr-x | doc/images/blog/figure19.JPG | bin | 0 -> 101549 bytes | |||
-rwxr-xr-x | doc/images/blog/figure2.JPG | bin | 0 -> 59920 bytes | |||
-rwxr-xr-x | doc/images/blog/figure20.png | bin | 0 -> 140990 bytes | |||
-rwxr-xr-x | doc/images/blog/figure21.png | bin | 0 -> 33714 bytes | |||
-rwxr-xr-x | doc/images/blog/figure22.png | bin | 0 -> 112572 bytes | |||
-rwxr-xr-x | doc/images/blog/figure3.JPG | bin | 0 -> 129869 bytes | |||
-rwxr-xr-x | doc/images/blog/figure4.JPG | bin | 0 -> 20269 bytes | |||
-rwxr-xr-x | doc/images/blog/figure5.JPG | bin | 0 -> 105792 bytes | |||
-rwxr-xr-x | doc/images/blog/figure6.JPG | bin | 0 -> 81691 bytes | |||
-rwxr-xr-x | doc/images/blog/figure7.JPG | bin | 0 -> 33888 bytes | |||
-rwxr-xr-x | doc/images/blog/figure8.JPG | bin | 0 -> 85129 bytes | |||
-rwxr-xr-x | doc/images/blog/figure9.png | bin | 0 -> 105620 bytes |
24 files changed, 306 insertions, 0 deletions
diff --git a/doc/analyticsBlog-docinfo.html b/doc/analyticsBlog-docinfo.html new file mode 100755 index 00000000..e69de29b --- /dev/null +++ b/doc/analyticsBlog-docinfo.html diff --git a/doc/analyticsBlog.asciidoc b/doc/analyticsBlog.asciidoc new file mode 100755 index 00000000..b9b75b6e --- /dev/null +++ b/doc/analyticsBlog.asciidoc @@ -0,0 +1,306 @@ +TRex
+====
+:author: itraviv
+:email: <itraviv@cisco.com>
+:revnumber: 1.0
+:revdate: 2017-03-23
+:quotes.++:
+:numbered:
+:web_server_url: http://trex-tgn.cisco.com/trex
+:local_web_server_url: csi-wiki-01:8181/trex
+:toclevels: 4
+
+include::trex_ga.asciidoc[]
+
+// PDF version - image width variable
+ifdef::backend-docbook[]
+:p_width: 450
+endif::backend-docbook[]
+
+// HTML version - image width variable
+ifdef::backend-xhtml11[]
+:p_width: 800
+endif::backend-xhtml11[]
+
+
+= How do we track TRex performance Using ElasticSearch, Grafana and Pandas
+The ability to monitor TRex performance on many setups/configurations on a daily basis may have a large impact on our ability to identify TRex performance degradation.
+For a long time our monitoring method was based on hard coded boundaries, we have defined maximum and minimum values for each test/'s result and any exeception triggered a notification.
+This monitoring method had a lot of false positives which in turn increased the investigation time. Moreover this method introduced more complexity through the need to maintain the golden results for many test cases on various platforms.
+
+A new method was required which would enable us to:
+1) Identify performance breakage automatically
+2) Report the performance trend on a time basis
+3) Explore/query old performance data in a simple manner
+4) Collect and extend new fields from our regression setup in a simple manner.
+
+The solution that we chose was to send all the performance results into an elastic search database and analyse it offline using Kibana and Grafana.
+
+This overtime analysis and tracking solution allows us to view the performance results and easily point out abnormal activity, not relying on hard coded boundaries.
+
+
+
+
+
+
+Figure 1 contains a block diagram of the new solution. +
+
+image:images/blog/figure1.jpg[title="figure1",align="left",width={p_width}, link="images/blog/figure1.jpg"]
+
+[small]#Figure 1#
+
+The following section will describe how we reached the chosen solution.
+
+== Early stage: Google Analytics and Python Pandas
+
+image:images/blog/figure2.jpg[title="figure2",align="left",width={p_width}, link="images/blog/figure2.jpg"]
+
+[small]#Figure 2# +
+Prior to using elastic search we tried to use Google Analytics (GA) as a database to store performance results. +
+GA account gives you an environment for collecting and analyzing data, mostly for marketing, e-commerce and internet traffic usage. +
+We've created a property (GA name for a separated section to collect and analyze data, one account can hold several properties) for collecting the results of our running tests and setups.
+
+
+
+
+
+
+=== A word on GA platform
+GA monitors user activity with websites and applications by collecting various parameters like 'pageviews' (of web pages), referrals (to your website), amount of sessions, users and downloads etc.
+There's a general parameter called 'Events', which lets the user define his points of interest as events. We created some custom 'metrics' and 'dimensions' for our performance tracking.
+
+=== Sending Data to Google Analytics
+Sending data to GA is done through an 'http' request to a collection server. +
+This can easily be automated, therefore we've created a python module that handles sending data to GA, Figure 3 depicts a snippet of the python module.
+
+image:images/blog/figure3.jpg[title="figure3",align="left",width={p_width}, link="images/blog/figure3.jpg"]
+
+[small]#Figure 3# +
+https://github.com/cisco-system-traffic-generator/trex-core/blob/master/scripts/automation/trex_control_plane/stl/trex_stl_lib/utils/GAObjClass.py/[View this on GitHub] +
+
+
+The payload is a string which will be sent as the payload of the http request.
+You can construct payloads and try sending data to GA like shown in figure 4:
+(more on this: [1])
+
+image:images/blog/figure4.jpg[title="figure4",align="left",width={p_width}, link="images/blog/figure4.jpg"]
+
+[small]#Figure 4# +
+
+https://github.com/cisco-system-traffic-generator/trex-core/blob/master/scripts/automation/trex_control_plane/stl/trex_stl_lib/utils/GAObjClass.py/[View this on GitHub] +
+
+(We are using "batched reporting", you can batch up to 20 payloads and report all at once)
+Each test result is reported to GA using the above code. +
+More about sending data to GA: [2]
+
+=== Getting your data from GA
+You can use the guides in resource [3] in order to create a request.
+The response is simply a JSON that contains the data requested with the applied filters.
+After getting the required data, we parse it in order to bring it to a more "comfortable" form for Analysis.
+Figure 5 shows an example for a Connection API to GA from a Python script: +
+
+image:images/blog/figure5.jpg[title="figure5",align="left",width={p_width}, link="images/blog/figure5.jpg"]
+
+[small]#Figure 5# +
+
+https://github.com/cisco-system-traffic-generator/trex-core/blob/master/doc/AnalyticsConnect.py/[View this on GitHub] +
+More on: https://developers.google.com/analytics/devguides/reporting/core/v3/quickstart/service-py/[Google API on Documantation] [4]
+
+== Analysis using python pandas
+=== A word on Pandas
+[From website] "Pandas is an open source, BSD-licensed library providing high-performance, easy-to-use data structures and data analysis tools for the Python programming language"
+
+In a way it is a nice programmable interface that replaces Excel and it includes a few nice Python packages like matplotlib & SciPy.
+
+We've created the pandas analysis module to be independent from querying and collecting the data, meaning this module can be used with any other DB for querying. The analysis module uses an input in JSON format to parse and analyze, that's it.
+
+
+=== Test analysis
+Test results are placed in pandas DataFrame ordered by date.
+DataFrame supports many calculations on the data, and this how we calculate the average, min, max and standard deviation values for every test run.
+Figure 6 holds an example for such calculations: +
+
+image:images/blog/figure6.jpg[title="figure6",align="left",width={p_width}, link="images/blog/figure6.jpg"]
+
+[small]#Figure 6# +
+
+https://github.com/cisco-system-traffic-generator/trex-core/blob/master/doc/TRexDataAnalysis.py/[View this on GitHub] +
+We also take the latest test results to publish.
+
+
+
+=== Setup analysis
+After analyzing all the tests for a given setup, we merge all tests into a single DataFrame which has a column for each test results, like shown in Figure 7.
+
+image:images/blog/figure7.jpg[title="figure7",align="left",width={p_width}, link="images/blog/figure7.jpg"]
+
+[small]#Figure 7# +
+
+
+Using the timestamp for each test results, we create a timeline for the plot_date function.
+We use pandas pyplot plot_date function to plot the results over time +
+
+image:images/blog/figure8.jpg[title="figure8",align="left",width={p_width}, link="images/blog/figure8.jpg"]
+
+[small]#Figure 8 - plotting results over time# +
+
+The code in figure 8 turns into this: +
+
+image:images/blog/figure9.png[title="figure9",align="left",width={p_width}, link="images/blog/figure9.png"]
+
+[small]#Figure 9# +
+
+Pandas supports exporting the data to csv, in figure 10 you can see the table you see below every graph in our https://trex-tgn.cisco.com/trex/doc/trex_analytics.html/[webpage]. +
+This table shows the data we rely on when plotting the trend-line graph (figure 9):
+
+image:images/blog/figure10.jpg[title="figure10",align="left",width={p_width}, link="images/blog/figure10.jpg"]
+
+[small]#Figure 10#
+
+We also plot the latest results we collected from each test as a bar chart (figure 11): +
+
+image:images/blog/figure11.jpg[title="figure11",align="left",width={p_width}, link="images/blog/figure11.jpg"]
+
+[small]#Figure 11 - plotting latest results as bar chart# +
+
+
+Figure 12 shows the plot we receive after running the code shown in Figure 11.
+
+image:images/blog/figure12.png[title="figure12",align="left",width={p_width}, link="images/blog/figure12.png"]
+
+[small]#Figure 12#
+
+Same as before, we export the data to a csv table using the "to_csv" api of pandas (as shown in Figure 8). +
+
+== Generating a report
+The analysis script we just described is generating all the graphs and tables for the asciidoc parser.
+In Figure 13 we see the asciidoc source file, from which the reports are made. The embedded graphs and tables are circled.
+
+image:images/blog/figure13.jpg[title="figure13",align="left",width={p_width}, link="images/blog/figure13.jpg"]
+
+[small]#Figure 13#
+
+
+The asciidoc parser creates the report (Figure 14):
+
+image:images/blog/figure14.jpg[title="figure14",align="left",width={p_width}, link="images/blog/figure14.jpg"]
+
+[small]#Figure 14#
+
+full report can be found https://trex-tgn.cisco.com/trex/doc/trex_analytics.html/[here] ([5]) +
+
+== Final stage using ElasticSearch instead of Google Analytics
+We have decided to use the elasticsearch suite: +
+
+image:images/blog/figure15.jpg[title="figure15",align="left",width={p_width}, link="images/blog/figure15.jpg"]
+
+[small]#Figure 15: current setup of the analytic module#
+
+
+=== What is Elasticsearch?
+[From website]" Elasticsearch is a distributed, RESTful search and analytics engine capable of solving a growing number of use cases. As the heart of the Elastic Stack, it centrally stores your data so you can discover the expected and uncover the unexpected"
+
+=== What is the motivation of using Elastic Search instead of GA?
+While GA mostly serves the tracking of user interactions with an application or a website, focusing on e-commerce and marketing, ELK(ElasticSearch/Kibana suit of products) is a data oriented engine designed for speed and simplicity of querying (JSON based). ELK also has a module for online analysis, querying and visualizing the data (called Kibana) and Grafana.
+Another advantage is that we could extend the fields we send to the ES(ElasticSearch engine) without changing the schema.
+
+
+=== Installation of Elasticsearch and Grafana
+Installing is as easy as following these instructions: +
+ELK [6] +
+Grafana [7]
+
+=== Sending data to ELK
+For each test a Performance report is created with the results and test parameters. +
+Figure 16 shows such a report.
+
+image:images/blog/figure16.jpg[title="figure16",align="left",width={p_width}, link="images/blog/figure16.jpg"]
+
+[small]#Figure 16#
+
+
+This class has a method for sending data to ELK using all the parameters, as you can see in figure 17.
+
+image:images/blog/figure17.jpg[title="figure17",align="left",width={p_width}, link="images/blog/figure17.jpg"]
+
+[small]#Figure 17#
+
+push_data is a method of a class that encapsulates elk_api. es.index is the "add" directive of ELK
+
+image:images/blog/figure18.jpg[title="figure18",align="left",width={p_width}, link="images/blog/figure18.jpg"]
+
+[small]#Figure 18 - elk api# +
+ +
+https://github.com/cisco-system-traffic-generator/trex-core/blob/master/scripts/automation/regression/trex_elk.py/[View this on GitHub] +
+
+
+=== Integration with the pandas analysis module
+As mentioned, the pandas module requires the data in some structure using JSON format.
+The new ELK module just queries the DB and parses the response in order to deliver it to the pandas analysis module. +
+
+image:images/blog/figure19.jpg[title="figure19",align="left",width={p_width}, link="images/blog/figure19.jpg"]
+
+[small]#Figure 19 - elk quering#
+
+=== Connect ElasticSearch To Grafana
+Why Grafana and not Kibana?
+While Kibana is good for generic analytics and querying information from ES, Grafana gives you a dashboard of time series streams (performance per setup) which is more suitable ans is easier to use from feature perspective.
+
+Dashboard for setups/test/ performance
+
+image:images/blog/figure20.png[title="figure20",align="left",width={p_width}, link="images/blog/figure20.png"]
+
+[small]#Figure 20 - dashboard examples#
+
+Dashboard for performance- zoom in into one setup +
+
+image:images/blog/figure21.png[title="figure21",align="left",width={p_width}, link="images/blog/figure21.png"]
+
+[small]#Figure 21- zoom on a setup#
+
+
+Dashboard for setups/test/ latency
+
+image:images/blog/figure22.png[title="figure22",align="left",width={p_width}, link="images/blog/figure22.png"]
+
+[small]#Figure 22 - additional information about a setup#
+
+
+Let's see if we answered the requirements using the described chosen solution +
+[qanda]
+Identify performance breakage automatically::
+ Not yet, but we could do it with Pandas in nightly script
+Report the performance trend on a time basis::
+ Yes
+Explore/query old performance data in a simple manner::
+ Yes, using Kibana (raw data) and Grafana (time-series)
+Collect and extend new fields from our regression setup in a simple manner::
+ Yes, using simple python ES API
+
+
+
+=== Resources
+
+[1] Google Analytics hit builder: +
+https://ga-dev-tools.appspot.com/hit-builder/ +
+[2] Sending data to Google Analytics https://developers.google.com/analytics/devguides/collection/analyticsjs/sending-hits +
+https://developers.google.com/analytics/devguides/collection/analyticsjs/events#overview +
+[3] Guides for creating a request to Google Analytics +
+https://developers.google.com/analytics/devguides/reporting/core/v4/rest/v4/reports/batchGet#MetricType +
+https://developers.google.com/analytics/devguides/reporting/core/v4/basics +
+[4] Google API Documentation https://developers.google.com/analytics/devguides/reporting/core/v3/quickstart/service-py +
+[5] TRex website: published reports +
+https://trex-tgn.cisco.com/trex/doc/trex_analytics.html +
+[6] ELK installation +
+https://www.elastic.co/downloads/elasticsearch +
+[7] Grafana installation +
+http://docs.grafana.org/ +
+[8] All the above code can be found in out github repository: +
+https://github.com/cisco-system-traffic-generator/trex-core/tree/master/doc +
+[9] pandas doc: +
+http://pandas.pydata.org/ +
+[10] Our analytic reports on our website: +
+https://trex-tgn.cisco.com/trex/doc/trex_analytics.html +
+
+
+
diff --git a/doc/images/blog/figure1.jpg b/doc/images/blog/figure1.jpg Binary files differnew file mode 100755 index 00000000..ae330331 --- /dev/null +++ b/doc/images/blog/figure1.jpg diff --git a/doc/images/blog/figure10.JPG b/doc/images/blog/figure10.JPG Binary files differnew file mode 100755 index 00000000..b121ec6a --- /dev/null +++ b/doc/images/blog/figure10.JPG diff --git a/doc/images/blog/figure11.jpg b/doc/images/blog/figure11.jpg Binary files differnew file mode 100755 index 00000000..fef300df --- /dev/null +++ b/doc/images/blog/figure11.jpg diff --git a/doc/images/blog/figure12.png b/doc/images/blog/figure12.png Binary files differnew file mode 100755 index 00000000..54c89511 --- /dev/null +++ b/doc/images/blog/figure12.png diff --git a/doc/images/blog/figure13.JPG b/doc/images/blog/figure13.JPG Binary files differnew file mode 100755 index 00000000..ff69be16 --- /dev/null +++ b/doc/images/blog/figure13.JPG diff --git a/doc/images/blog/figure14.JPG b/doc/images/blog/figure14.JPG Binary files differnew file mode 100755 index 00000000..f07359aa --- /dev/null +++ b/doc/images/blog/figure14.JPG diff --git a/doc/images/blog/figure15.jpg b/doc/images/blog/figure15.jpg Binary files differnew file mode 100755 index 00000000..ba36ad39 --- /dev/null +++ b/doc/images/blog/figure15.jpg diff --git a/doc/images/blog/figure16.JPG b/doc/images/blog/figure16.JPG Binary files differnew file mode 100755 index 00000000..7e1f94dd --- /dev/null +++ b/doc/images/blog/figure16.JPG diff --git a/doc/images/blog/figure17.JPG b/doc/images/blog/figure17.JPG Binary files differnew file mode 100755 index 00000000..9cba49cc --- /dev/null +++ b/doc/images/blog/figure17.JPG diff --git a/doc/images/blog/figure18.JPG b/doc/images/blog/figure18.JPG Binary files differnew file mode 100755 index 00000000..9236bdae --- /dev/null +++ b/doc/images/blog/figure18.JPG diff --git a/doc/images/blog/figure19.JPG b/doc/images/blog/figure19.JPG Binary files differnew file mode 100755 index 00000000..0f537811 --- /dev/null +++ b/doc/images/blog/figure19.JPG diff --git a/doc/images/blog/figure2.JPG b/doc/images/blog/figure2.JPG Binary files differnew file mode 100755 index 00000000..97a42d03 --- /dev/null +++ b/doc/images/blog/figure2.JPG diff --git a/doc/images/blog/figure20.png b/doc/images/blog/figure20.png Binary files differnew file mode 100755 index 00000000..4f57f426 --- /dev/null +++ b/doc/images/blog/figure20.png diff --git a/doc/images/blog/figure21.png b/doc/images/blog/figure21.png Binary files differnew file mode 100755 index 00000000..42990d5a --- /dev/null +++ b/doc/images/blog/figure21.png diff --git a/doc/images/blog/figure22.png b/doc/images/blog/figure22.png Binary files differnew file mode 100755 index 00000000..9e8206e4 --- /dev/null +++ b/doc/images/blog/figure22.png diff --git a/doc/images/blog/figure3.JPG b/doc/images/blog/figure3.JPG Binary files differnew file mode 100755 index 00000000..094b0387 --- /dev/null +++ b/doc/images/blog/figure3.JPG diff --git a/doc/images/blog/figure4.JPG b/doc/images/blog/figure4.JPG Binary files differnew file mode 100755 index 00000000..3154b035 --- /dev/null +++ b/doc/images/blog/figure4.JPG diff --git a/doc/images/blog/figure5.JPG b/doc/images/blog/figure5.JPG Binary files differnew file mode 100755 index 00000000..11ff7e56 --- /dev/null +++ b/doc/images/blog/figure5.JPG diff --git a/doc/images/blog/figure6.JPG b/doc/images/blog/figure6.JPG Binary files differnew file mode 100755 index 00000000..903b4162 --- /dev/null +++ b/doc/images/blog/figure6.JPG diff --git a/doc/images/blog/figure7.JPG b/doc/images/blog/figure7.JPG Binary files differnew file mode 100755 index 00000000..29732411 --- /dev/null +++ b/doc/images/blog/figure7.JPG diff --git a/doc/images/blog/figure8.JPG b/doc/images/blog/figure8.JPG Binary files differnew file mode 100755 index 00000000..b431ebd5 --- /dev/null +++ b/doc/images/blog/figure8.JPG diff --git a/doc/images/blog/figure9.png b/doc/images/blog/figure9.png Binary files differnew file mode 100755 index 00000000..d5f64ab7 --- /dev/null +++ b/doc/images/blog/figure9.png |