Ori.livneh has uploaded a new change for review.

  https://gerrit.wikimedia.org/r/179791

Change subject: Add 'xenon' module for aggregating ext_xenon-produced traces
......................................................................

Add 'xenon' module for aggregating ext_xenon-produced traces

Provisions a simple Python daemon that subscribes to Xenon PHP traces published
on a Redis server and writes them to disk, plus a cron job that uses
<https://github.com/brendangregg/FlameGraph> to generate flame graphs from the
data.

Change-Id: I09926c8c26da29baedcf5ab3ef219770690be30a
---
A modules/xenon/files/flamegraph.pl
A modules/xenon/files/xenon-generate-svgs
A modules/xenon/files/xenon-log
A modules/xenon/files/xenon-log.conf
A modules/xenon/manifests/init.pp
5 files changed, 1,092 insertions(+), 0 deletions(-)


  git pull ssh://gerrit.wikimedia.org:29418/operations/puppet 
refs/changes/91/179791/1

diff --git a/modules/xenon/files/flamegraph.pl 
b/modules/xenon/files/flamegraph.pl
new file mode 100755
index 0000000..def5ed0
--- /dev/null
+++ b/modules/xenon/files/flamegraph.pl
@@ -0,0 +1,846 @@
+#!/usr/bin/perl -w
+#
+# flamegraph.pl                flame stack grapher.
+#
+# This takes stack samples and renders a call graph, allowing hot functions
+# and codepaths to be quickly identified.  Stack samples can be generated using
+# tools such as DTrace, perf, SystemTap, and Instruments.
+#
+# USAGE: ./flamegraph.pl [options] input.txt > graph.svg
+#
+#        grep funcA input.txt | ./flamegraph.pl [options] > graph.svg
+#
+# Options are listed in the usage message (--help).
+#
+# The input is stack frames and sample counts formatted as single lines.  Each
+# frame in the stack is semicolon separated, with a space and count at the end
+# of the line.  These can be generated using DTrace with stackcollapse.pl,
+# and other tools using the stackcollapse variants.
+#
+# An optional extra column of counts can be provided to generate a differential
+# flame graph of the counts, colored red for more, and blue for less.  This
+# can be useful when using flame graphs for non-regression testing.
+# See the header comment in the difffolded.pl program for instructions.
+#
+# The output graph shows relative presence of functions in stack samples.  The
+# ordering on the x-axis has no meaning; since the data is samples, time order
+# of events is not known.  The order used sorts function names alphabetically.
+#
+# While intended to process stack samples, this can also process stack traces.
+# For example, tracing stacks for memory allocation, or resource usage.  You
+# can use --title to set the title to reflect the content, and --countname
+# to change "samples" to "bytes" etc.
+#
+# There are a few different palettes, selectable using --color.  By default,
+# the colors are selected at random (except for differentials).  Functions
+# called "-" will be printed gray, which can be used for stack separators (eg,
+# between user and kernel stacks).
+#
+# HISTORY
+#
+# This was inspired by Neelakanth Nadgir's excellent function_call_graph.rb
+# program, which visualized function entry and return trace events.  As Neel
+# wrote: "The output displayed is inspired by Roch's CallStackAnalyzer which
+# was in turn inspired by the work on vftrace by Jan Boerhout".  See:
+# https://blogs.oracle.com/realneel/entry/visualizing_callstacks_via_dtrace_and
+#
+# Copyright 2011 Joyent, Inc.  All rights reserved.
+# Copyright 2011 Brendan Gregg.  All rights reserved.
+#
+# CDDL HEADER START
+#
+# The contents of this file are subject to the terms of the
+# Common Development and Distribution License (the "License").
+# You may not use this file except in compliance with the License.
+#
+# You can obtain a copy of the license at docs/cddl1.txt or
+# http://opensource.org/licenses/CDDL-1.0.
+# See the License for the specific language governing permissions
+# and limitations under the License.
+#
+# When distributing Covered Code, include this CDDL HEADER in each
+# file and include the License file at docs/cddl1.txt.
+# If applicable, add the following below this CDDL HEADER, with the
+# fields enclosed by brackets "[]" replaced with your own identifying
+# information: Portions Copyright [yyyy] [name of copyright owner]
+#
+# CDDL HEADER END
+#
+# 21-Nov-2013   Shawn Sterling  Added consistent palette file option
+# 17-Mar-2013   Tim Bunce       Added options and more tunables.
+# 15-Dec-2011  Dave Pacheco    Support for frames with whitespace.
+# 10-Sep-2011  Brendan Gregg   Created this.
+
+use strict;
+
+use Getopt::Long;
+
+# tunables
+my $encoding;
+my $fonttype = "Verdana";
+my $imagewidth = 1200;          # max width, pixels
+my $frameheight = 16;           # max height is dynamic
+my $fontsize = 12;              # base text size
+my $fontwidth = 0.59;           # avg width relative to fontsize
+my $minwidth = 0.1;             # min function width, pixels
+my $nametype = "Function:";     # what are the names in the data?
+my $countname = "samples";      # what are the counts in the data?
+my $colors = "hot";             # color theme
+my $bgcolor1 = "#eeeeee";       # background color gradient start
+my $bgcolor2 = "#eeeeb0";       # background color gradient stop
+my $nameattrfile;               # file holding function attributes
+my $timemax;                    # (override the) sum of the counts
+my $factor = 1;                 # factor to scale counts by
+my $hash = 0;                   # color by function name
+my $palette = 0;                # if we use consistent palettes (default off)
+my %palette_map;                # palette map hash
+my $pal_file = "palette.map";   # palette map file name
+my $stackreverse = 0;           # reverse stack order, switching merge end
+my $inverted = 0;               # icicle graph
+my $negate = 0;                 # switch differential hues
+my $titletext = "";             # centered heading
+my $titledefault = "Flame Graph";      # overwritten by --title
+my $titleinverted = "Icicle Graph";    #   "    "
+
+GetOptions(
+       'fonttype=s'  => \$fonttype,
+       'width=i'     => \$imagewidth,
+       'height=i'    => \$frameheight,
+       'encoding=s'  => \$encoding,
+       'fontsize=f'  => \$fontsize,
+       'fontwidth=f' => \$fontwidth,
+       'minwidth=f'  => \$minwidth,
+       'title=s'     => \$titletext,
+       'nametype=s'  => \$nametype,
+       'countname=s' => \$countname,
+       'nameattr=s'  => \$nameattrfile,
+       'total=s'     => \$timemax,
+       'factor=f'    => \$factor,
+       'colors=s'    => \$colors,
+       'hash'        => \$hash,
+       'cp'          => \$palette,
+       'reverse'     => \$stackreverse,
+       'inverted'    => \$inverted,
+       'negate'      => \$negate,
+) or die <<USAGE_END;
+USAGE: $0 [options] infile > outfile.svg\n
+       --title       # change title text
+       --width       # width of image (default 1200)
+       --height      # height of each frame (default 16)
+       --minwidth    # omit smaller functions (default 0.1 pixels)
+       --fonttype    # font type (default "Verdana")
+       --fontsize    # font size (default 12)
+       --countname   # count type label (default "samples")
+       --nametype    # name type label (default "Function:")
+       --colors      # set color palette. choices are: hot (default), mem, io,
+                     # java, js, red, green, blue, yellow, purple, orange
+       --hash        # colors are keyed by function name hash
+       --cp          # use consistent palette (palette.map)
+       --reverse     # generate stack-reversed flame graph
+       --inverted    # icicle graph
+       --negate      # switch differential hues (blue<->red)
+
+       eg,
+       $0 --title="Flame Graph: malloc()" trace.txt > graph.svg
+USAGE_END
+
+# internals
+my $ypad1 = $fontsize * 4;      # pad top, include title
+my $ypad2 = $fontsize * 2 + 10; # pad bottom, include labels
+my $xpad = 10;                  # pad lefm and right
+my $framepad = 1;              # vertical padding for frames
+my $depthmax = 0;
+my %Events;
+my %nameattr;
+
+if ($titletext eq "") {
+       unless ($inverted) {
+               $titletext = $titledefault;
+       } else {
+               $titletext = $titleinverted;
+       }
+}
+
+if ($nameattrfile) {
+       # The name-attribute file format is a function name followed by a tab 
then
+       # a sequence of tab separated name=value pairs.
+       open my $attrfh, $nameattrfile or die "Can't read $nameattrfile: $!\n";
+       while (<$attrfh>) {
+               chomp;
+               my ($funcname, $attrstr) = split /\t/, $_, 2;
+               die "Invalid format in $nameattrfile" unless defined $attrstr;
+               $nameattr{$funcname} = { map { split /=/, $_, 2 } split /\t/, 
$attrstr };
+       }
+}
+
+if ($colors eq "mem") { $bgcolor1 = "#eeeeee"; $bgcolor2 = "#e0e0ff"; }
+if ($colors eq "io")  { $bgcolor1 = "#f8f8f8"; $bgcolor2 = "#e8e8e8"; }
+
+# SVG functions
+{ package SVG;
+       sub new {
+               my $class = shift;
+               my $self = {};
+               bless ($self, $class);
+               return $self;
+       }
+
+       sub header {
+               my ($self, $w, $h) = @_;
+               my $enc_attr = '';
+               if (defined $encoding) {
+                       $enc_attr = qq{ encoding="$encoding"};
+               }
+               $self->{svg} .= <<SVG;
+<?xml version="1.0"$enc_attr standalone="no"?>
+<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" 
"http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd";>
+<svg version="1.1" width="$w" height="$h" onload="init(evt)" viewBox="0 0 $w 
$h" xmlns="http://www.w3.org/2000/svg"; 
xmlns:xlink="http://www.w3.org/1999/xlink";>
+SVG
+       }
+
+       sub include {
+               my ($self, $content) = @_;
+               $self->{svg} .= $content;
+       }
+
+       sub colorAllocate {
+               my ($self, $r, $g, $b) = @_;
+               return "rgb($r,$g,$b)";
+       }
+
+       sub group_start {
+               my ($self, $attr) = @_;
+
+               my @g_attr = map {
+                       exists $attr->{$_} ? sprintf(qq/$_="%s"/, $attr->{$_}) 
: ()
+               } qw(class style onmouseover onmouseout onclick);
+               push @g_attr, $attr->{g_extra} if $attr->{g_extra};
+               $self->{svg} .= sprintf qq/<g %s>\n/, join(' ', @g_attr);
+
+               $self->{svg} .= sprintf qq/<title>%s<\/title>/, $attr->{title}
+                       if $attr->{title}; # should be first element within g 
container
+
+               if ($attr->{href}) {
+                       my @a_attr;
+                       push @a_attr, sprintf qq/xlink:href="%s"/, 
$attr->{href} if $attr->{href};
+                       # default target=_top else links will open within SVG 
<object>
+                       push @a_attr, sprintf qq/target="%s"/, $attr->{target} 
|| "_top";
+                       push @a_attr, $attr->{a_extra}                          
 if $attr->{a_extra};
+                       $self->{svg} .= sprintf qq/<a %s>/, join(' ', @a_attr);
+               }
+       }
+
+       sub group_end {
+               my ($self, $attr) = @_;
+               $self->{svg} .= qq/<\/a>\n/ if $attr->{href};
+               $self->{svg} .= qq/<\/g>\n/;
+       }
+
+       sub filledRectangle {
+               my ($self, $x1, $y1, $x2, $y2, $fill, $extra) = @_;
+               $x1 = sprintf "%0.1f", $x1;
+               $x2 = sprintf "%0.1f", $x2;
+               my $w = sprintf "%0.1f", $x2 - $x1;
+               my $h = sprintf "%0.1f", $y2 - $y1;
+               $extra = defined $extra ? $extra : "";
+               $self->{svg} .= qq/<rect x="$x1" y="$y1" width="$w" height="$h" 
fill="$fill" $extra \/>\n/;
+       }
+
+       sub stringTTF {
+               my ($self, $color, $font, $size, $angle, $x, $y, $str, $loc, 
$extra) = @_;
+               $x = sprintf "%0.2f", $x;
+               $loc = defined $loc ? $loc : "left";
+               $extra = defined $extra ? $extra : "";
+               $self->{svg} .= qq/<text text-anchor="$loc" x="$x" y="$y" 
font-size="$size" font-family="$font" fill="$color" $extra >$str<\/text>\n/;
+       }
+
+       sub svg {
+               my $self = shift;
+               return "$self->{svg}</svg>\n";
+       }
+       1;
+}
+
+sub namehash {
+       # Generate a vector hash for the name string, weighting early over
+       # later characters. We want to pick the same colors for function
+       # names across different flame graphs.
+       my $name = shift;
+       my $vector = 0;
+       my $weight = 1;
+       my $max = 1;
+       my $mod = 10;
+       # if module name present, trunc to 1st char
+       $name =~ s/.(.*?)`//;
+       foreach my $c (split //, $name) {
+               my $i = (ord $c) % $mod;
+               $vector += ($i / ($mod++ - 1)) * $weight;
+               $max += 1 * $weight;
+               $weight *= 0.70;
+               last if $mod > 12;
+       }
+       return (1 - $vector / $max)
+}
+
+sub color {
+       my ($type, $hash, $name) = @_;
+       my ($v1, $v2, $v3);
+
+       if ($hash) {
+               $v1 = namehash($name);
+               $v2 = $v3 = namehash(scalar reverse $name);
+       } else {
+               $v1 = rand(1);
+               $v2 = rand(1);
+               $v3 = rand(1);
+       }
+
+       # theme palettes
+       if (defined $type and $type eq "hot") {
+               my $r = 205 + int(50 * $v3);
+               my $g = 0 + int(230 * $v1);
+               my $b = 0 + int(55 * $v2);
+               return "rgb($r,$g,$b)";
+       }
+       if (defined $type and $type eq "mem") {
+               my $r = 0;
+               my $g = 190 + int(50 * $v2);
+               my $b = 0 + int(210 * $v1);
+               return "rgb($r,$g,$b)";
+       }
+       if (defined $type and $type eq "io") {
+               my $r = 80 + int(60 * $v1);
+               my $g = $r;
+               my $b = 190 + int(55 * $v2);
+               return "rgb($r,$g,$b)";
+       }
+
+       # multi palettes
+       if (defined $type and $type eq "java") {
+               if ($name =~ /::/) {            # C++
+                       $type = "yellow";
+               } elsif ($name =~ m:/:) {       # Java (match "/" in path)
+                       $type = "green"
+               } else {                        # system
+                       $type = "red";
+               }
+               # fall-through to color palettes
+       }
+       if (defined $type and $type eq "js") {
+               if ($name =~ /::/) {            # C++
+                       $type = "yellow";
+               } elsif ($name =~ m:/:) {       # JavaScript (match "/" in path)
+                       $type = "green"
+               } elsif ($name =~ m/:/) {       # JavaScript (match ":" in 
builtin)
+                       $type = "aqua"
+               } elsif ($name =~ m/^ $/) {     # Missing symbol
+                       $type = "green"
+               } else {                        # system
+                       $type = "red";
+               }
+               # fall-through to color palettes
+       }
+
+       # color palettes
+       if (defined $type and $type eq "red") {
+               my $r = 200 + int(55 * $v1);
+               my $x = 50 + int(80 * $v1);
+               return "rgb($r,$x,$x)";
+       }
+       if (defined $type and $type eq "green") {
+               my $g = 200 + int(55 * $v1);
+               my $x = 50 + int(60 * $v1);
+               return "rgb($x,$g,$x)";
+       }
+       if (defined $type and $type eq "blue") {
+               my $b = 205 + int(50 * $v1);
+               my $x = 80 + int(60 * $v1);
+               return "rgb($x,$x,$b)";
+       }
+       if (defined $type and $type eq "yellow") {
+               my $x = 175 + int(55 * $v1);
+               my $b = 50 + int(20 * $v1);
+               return "rgb($x,$x,$b)";
+       }
+       if (defined $type and $type eq "purple") {
+               my $x = 190 + int(65 * $v1);
+               my $g = 80 + int(60 * $v1);
+               return "rgb($x,$g,$x)";
+       }
+       if (defined $type and $type eq "aqua") {
+               my $r = 50 + int(60 * $v1);
+               my $g = 165 + int(55 * $v1);
+               my $b = 165 + int(55 * $v1);
+               return "rgb($r,$g,$b)";
+       }
+       if (defined $type and $type eq "orange") {
+               my $r = 190 + int(65 * $v1);
+               my $g = 90 + int(65 * $v1);
+               return "rgb($r,$g,0)";
+       }
+
+       return "rgb(0,0,0)";
+}
+
+sub color_scale {
+       my ($value, $max) = @_;
+       my ($r, $g, $b) = (255, 255, 255);
+       $value = -$value if $negate;
+       if ($value > 0) {
+               $g = $b = int(210 * ($max - $value) / $max);
+       } elsif ($value < 0) {
+               $r = $g = int(210 * ($max + $value) / $max);
+       }
+       return "rgb($r,$g,$b)";
+}
+
+sub color_map {
+       my ($colors, $func) = @_;
+       if (exists $palette_map{$func}) {
+               return $palette_map{$func};
+       } else {
+               $palette_map{$func} = color($colors);
+               return $palette_map{$func};
+       }
+}
+
+sub write_palette {
+       open(FILE, ">$pal_file");
+       foreach my $key (sort keys %palette_map) {
+               print FILE $key."->".$palette_map{$key}."\n";
+       }
+       close(FILE);
+}
+
+sub read_palette {
+       if (-e $pal_file) {
+       open(FILE, $pal_file) or die "can't open file $pal_file: $!";
+       while ( my $line = <FILE>) {
+               chomp($line);
+               (my $key, my $value) = split("->",$line);
+               $palette_map{$key}=$value;
+       }
+       close(FILE)
+       }
+}
+
+my %Node;      # Hash of merged frame data
+my %Tmp;
+
+# flow() merges two stacks, storing the merged frames and value data in %Node.
+sub flow {
+       my ($last, $this, $v, $d) = @_;
+
+       my $len_a = @$last - 1;
+       my $len_b = @$this - 1;
+
+       my $i = 0;
+       my $len_same;
+       for (; $i <= $len_a; $i++) {
+               last if $i > $len_b;
+               last if $last->[$i] ne $this->[$i];
+       }
+       $len_same = $i;
+
+       for ($i = $len_a; $i >= $len_same; $i--) {
+               my $k = "$last->[$i];$i";
+               # a unique ID is constructed from "func;depth;etime";
+               # func-depth isn't unique, it may be repeated later.
+               $Node{"$k;$v"}->{stime} = delete $Tmp{$k}->{stime};
+               if (defined $Tmp{$k}->{delta}) {
+                       $Node{"$k;$v"}->{delta} = delete $Tmp{$k}->{delta};
+               }
+               delete $Tmp{$k};
+       }
+
+       for ($i = $len_same; $i <= $len_b; $i++) {
+               my $k = "$this->[$i];$i";
+               $Tmp{$k}->{stime} = $v;
+               if (defined $d) {
+                       $Tmp{$k}->{delta} += $i == $len_b ? $d : 0;
+               }
+       }
+
+        return $this;
+}
+
+# parse input
+my @Data;
+my $last = [];
+my $time = 0;
+my $delta = undef;
+my $ignored = 0;
+my $line;
+my $maxdelta = 1;
+
+# reverse if needed
+foreach (<>) {
+       chomp;
+       $line = $_;
+       if ($stackreverse) {
+               # there may be an extra samples column for differentials
+               # XXX todo: redo these REs as one. It's repeated below.
+               my ($stack, $samples) = (/^(.*)\s+?(\d+(?:\.\d*)?)$/);
+               my $samples2 = undef;
+               if ($stack =~ /^(.*)\s+?(\d+(?:\.\d*)?)$/) {
+                       $samples2 = $samples;
+                       ($stack, $samples) = $stack =~ 
(/^(.*)\s+?(\d+(?:\.\d*)?)$/);
+                       unshift @Data, join(";", reverse split(";", $stack)) . 
" $samples $samples2";
+               } else {
+                       unshift @Data, join(";", reverse split(";", $stack)) . 
" $samples";
+               }
+       } else {
+               unshift @Data, $line;
+       }
+}
+
+# process and merge frames
+foreach (sort @Data) {
+       chomp;
+       # process: folded_stack count
+       # eg: func_a;func_b;func_c 31
+       my ($stack, $samples) = (/^(.*)\s+?(\d+(?:\.\d*)?)$/);
+       unless (defined $samples and defined $stack) {
+               ++$ignored;
+               next;
+       }
+
+       # there may be an extra samples column for differentials:
+       my $samples2 = undef;
+       if ($stack =~ /^(.*)\s+?(\d+(?:\.\d*)?)$/) {
+               $samples2 = $samples;
+               ($stack, $samples) = $stack =~ (/^(.*)\s+?(\d+(?:\.\d*)?)$/);
+       }
+       $delta = undef;
+       if (defined $samples2) {
+               $delta = $samples2 - $samples;
+               $maxdelta = abs($delta) if abs($delta) > $maxdelta;
+       }
+
+       $stack =~ tr/<>/()/;
+
+       # merge frames and populate %Node:
+       $last = flow($last, [ '', split ";", $stack ], $time, $delta);
+
+       if (defined $samples2) {
+               $time += $samples2;
+       } else {
+               $time += $samples;
+       }
+}
+flow($last, [], $time, $delta);
+
+warn "Ignored $ignored lines with invalid format\n" if $ignored;
+unless ($time) {
+       warn "ERROR: No stack counts found\n";
+       my $im = SVG->new();
+       # emit an error message SVG, for tools automating flamegraph use
+       my $imageheight = $fontsize * 5;
+       $im->header($imagewidth, $imageheight);
+       $im->stringTTF($im->colorAllocate(0, 0, 0), $fonttype, $fontsize + 2,
+           0.0, int($imagewidth / 2), $fontsize * 2,
+           "ERROR: No valid input provided to flamegraph.pl.", "middle");
+       print $im->svg;
+       exit 2;
+}
+if ($timemax and $timemax < $time) {
+       warn "Specified --total $timemax is less than actual total $time, so 
ignored\n"
+       if $timemax/$time > 0.02; # only warn is significant (e.g., not 
rounding etc)
+       undef $timemax;
+}
+$timemax ||= $time;
+
+my $widthpertime = ($imagewidth - 2 * $xpad) / $timemax;
+my $minwidth_time = $minwidth / $widthpertime;
+
+# prune blocks that are too narrow and determine max depth
+while (my ($id, $node) = each %Node) {
+       my ($func, $depth, $etime) = split ";", $id;
+       my $stime = $node->{stime};
+       die "missing start for $id" if not defined $stime;
+
+       if (($etime-$stime) < $minwidth_time) {
+               delete $Node{$id};
+               next;
+       }
+       $depthmax = $depth if $depth > $depthmax;
+}
+
+# draw canvas, and embed interactive JavaScript program
+my $imageheight = ($depthmax * $frameheight) + $ypad1 + $ypad2;
+my $im = SVG->new();
+$im->header($imagewidth, $imageheight);
+my $inc = <<INC;
+<defs >
+       <linearGradient id="background" y1="0" y2="1" x1="0" x2="0" >
+               <stop stop-color="$bgcolor1" offset="5%" />
+               <stop stop-color="$bgcolor2" offset="95%" />
+       </linearGradient>
+</defs>
+<style type="text/css">
+       .func_g:hover { stroke:black; stroke-width:0.5; cursor:pointer; }
+</style>
+<script type="text/ecmascript">
+<![CDATA[
+       var details, svg;
+       function init(evt) { 
+               details = document.getElementById("details").firstChild; 
+               svg = document.getElementsByTagName("svg")[0];
+       }
+       function s(info) { details.nodeValue = "$nametype " + info; }
+       function c() { details.nodeValue = ' '; }
+       function find_child(parent, name, attr) {
+               var children = parent.childNodes;
+               for (var i=0; i<children.length;i++) {
+                       if (children[i].tagName == name)
+                               return (attr != undefined) ? 
children[i].attributes[attr].value : children[i];
+               }
+               return;
+       }
+       function orig_save(e, attr, val) {
+               if (e.attributes["_orig_"+attr] != undefined) return;
+               if (e.attributes[attr] == undefined) return;
+               if (val == undefined) val = e.attributes[attr].value;
+               e.setAttribute("_orig_"+attr, val);
+       }
+       function orig_load(e, attr) {
+               if (e.attributes["_orig_"+attr] == undefined) return;
+               e.attributes[attr].value = e.attributes["_orig_"+attr].value;
+               e.removeAttribute("_orig_"+attr);
+       }
+       function update_text(e) {
+               var r = find_child(e, "rect");
+               var t = find_child(e, "text");
+               var w = parseFloat(r.attributes["width"].value) -3;
+               var txt = find_child(e, 
"title").textContent.replace(/\\([^(]*\\)/,"");
+               t.attributes["x"].value = parseFloat(r.attributes["x"].value) 
+3;
+               
+               // Smaller than this size won't fit anything
+               if (w < 2*$fontsize*$fontwidth) {
+                       t.textContent = "";
+                       return;
+               }
+               
+               t.textContent = txt;
+               // Fit in full text width
+               if (/^ *\$/.test(txt) || t.getSubStringLength(0, txt.length) < 
w)
+                       return;
+               
+               for (var x=txt.length-2; x>0; x--) {
+                       if (t.getSubStringLength(0, x+2) <= w) { 
+                               t.textContent = txt.substring(0,x) + "..";
+                               return;
+                       }
+               }
+               t.textContent = "";
+       }
+       function zoom_reset(e) {
+               if (e.attributes != undefined) {
+                       orig_load(e, "x");
+                       orig_load(e, "width");
+               }
+               if (e.childNodes == undefined) return;
+               for(var i=0, c=e.childNodes; i<c.length; i++) {
+                       zoom_reset(c[i]);
+               }
+       }
+       function zoom_child(e, x, ratio) {
+               if (e.attributes != undefined) {
+                       if (e.attributes["x"] != undefined) {
+                               orig_save(e, "x");
+                               e.attributes["x"].value = 
(parseFloat(e.attributes["x"].value) - x - $xpad) * ratio + $xpad;
+                               if(e.tagName == "text") e.attributes["x"].value 
= find_child(e.parentNode, "rect", "x") + 3;
+                       }
+                       if (e.attributes["width"] != undefined) {
+                               orig_save(e, "width");
+                               e.attributes["width"].value = 
parseFloat(e.attributes["width"].value) * ratio;
+                       }
+               }
+               
+               if (e.childNodes == undefined) return;
+               for(var i=0, c=e.childNodes; i<c.length; i++) {
+                       zoom_child(c[i], x-$xpad, ratio);
+               }
+       }
+       function zoom_parent(e) {
+               if (e.attributes) {
+                       if (e.attributes["x"] != undefined) {
+                               orig_save(e, "x");
+                               e.attributes["x"].value = $xpad;
+                       }
+                       if (e.attributes["width"] != undefined) {
+                               orig_save(e, "width");
+                               e.attributes["width"].value = 
parseInt(svg.width.baseVal.value) - ($xpad*2);
+                       }
+               }
+               if (e.childNodes == undefined) return;
+               for(var i=0, c=e.childNodes; i<c.length; i++) {
+                       zoom_parent(c[i]);
+               }
+       }
+       function zoom(node) { 
+               var attr = find_child(node, "rect").attributes;
+               var width = parseFloat(attr["width"].value);
+               var xmin = parseFloat(attr["x"].value);
+               var xmax = parseFloat(xmin + width);
+               var ymin = parseFloat(attr["y"].value);
+               var ratio = (svg.width.baseVal.value - 2*$xpad) / width;
+               
+               // XXX: Workaround for JavaScript float issues (fix me)
+               var fudge = 0.0001;
+               
+               var unzoombtn = document.getElementById("unzoom");
+               unzoombtn.style["opacity"] = "1.0";
+               
+               var el = document.getElementsByTagName("g");
+               for(var i=0;i<el.length;i++){
+                       var e = el[i];
+                       var a = find_child(e, "rect").attributes;
+                       var ex = parseFloat(a["x"].value);
+                       var ew = parseFloat(a["width"].value);
+                       // Is it an ancestor
+                       if ($inverted == 0) {
+                               var upstack = parseFloat(a["y"].value) > ymin;
+                       } else {
+                               var upstack = parseFloat(a["y"].value) < ymin;
+                       }
+                       if (upstack) {
+                               // Direct ancestor
+                               if (ex <= xmin && (ex+ew+fudge) >= xmax) {
+                                       e.style["opacity"] = "0.5";
+                                       zoom_parent(e);
+                                       e.onclick = function(e){unzoom(); 
zoom(this);};
+                                       update_text(e);
+                               }
+                               // not in current path
+                               else
+                                       e.style["display"] = "none";
+                       }
+                       // Children maybe
+                       else {
+                               // no common path
+                               if (ex < xmin || ex + fudge >= xmax) {
+                                       e.style["display"] = "none";
+                               }
+                               else {
+                                       zoom_child(e, xmin, ratio);
+                                       e.onclick = function(e){zoom(this);};
+                                       update_text(e);
+                               }
+                       }
+               }
+       }
+       function unzoom() {
+               var unzoombtn = document.getElementById("unzoom");
+               unzoombtn.style["opacity"] = "0.0";
+               
+               var el = document.getElementsByTagName("g");
+               for(i=0;i<el.length;i++) {
+                       el[i].style["display"] = "block";
+                       el[i].style["opacity"] = "1";
+                       zoom_reset(el[i]);
+                       update_text(el[i]);
+               }
+       }       
+]]>
+</script>
+INC
+$im->include($inc);
+$im->filledRectangle(0, 0, $imagewidth, $imageheight, 'url(#background)');
+my ($white, $black, $vvdgrey, $vdgrey) = (
+       $im->colorAllocate(255, 255, 255),
+       $im->colorAllocate(0, 0, 0),
+       $im->colorAllocate(40, 40, 40),
+       $im->colorAllocate(160, 160, 160),
+    );
+$im->stringTTF($black, $fonttype, $fontsize + 5, 0.0, int($imagewidth / 2), 
$fontsize * 2, $titletext, "middle");
+$im->stringTTF($black, $fonttype, $fontsize, 0.0, $xpad, $imageheight - 
($ypad2 / 2), " ", "", 'id="details"');
+$im->stringTTF($black, $fonttype, $fontsize, 0.0, $xpad, $fontsize * 2,
+    "Reset Zoom", "", 'id="unzoom" onclick="unzoom()" 
style="opacity:0.0;cursor:pointer"');
+
+if ($palette) {
+       read_palette();
+}
+
+# draw frames
+while (my ($id, $node) = each %Node) {
+       my ($func, $depth, $etime) = split ";", $id;
+       my $stime = $node->{stime};
+       my $delta = $node->{delta};
+
+       $etime = $timemax if $func eq "" and $depth == 0;
+
+       my $x1 = $xpad + $stime * $widthpertime;
+       my $x2 = $xpad + $etime * $widthpertime;
+       my ($y1, $y2);
+       unless ($inverted) {
+               $y1 = $imageheight - $ypad2 - ($depth + 1) * $frameheight + 
$framepad;
+               $y2 = $imageheight - $ypad2 - $depth * $frameheight;
+       } else {
+               $y1 = $ypad1 + $depth * $frameheight;
+               $y2 = $ypad1 + ($depth + 1) * $frameheight - $framepad;
+       }
+
+       my $samples = sprintf "%.0f", ($etime - $stime) * $factor;
+       (my $samples_txt = $samples) # add commas per perlfaq5
+               =~ s/(^[-+]?\d+?(?=(?>(?:\d{3})+)(?!\d))|\G\d{3}(?=\d))/$1,/g;
+
+       my $info;
+       if ($func eq "" and $depth == 0) {
+               $info = "all ($samples_txt $countname, 100%)";
+       } else {
+               my $pct = sprintf "%.2f", ((100 * $samples) / ($timemax * 
$factor));
+               my $escaped_func = $func;
+               $escaped_func =~ s/&/&amp;/g;
+               $escaped_func =~ s/</&lt;/g;
+               $escaped_func =~ s/>/&gt;/g;
+               unless (defined $delta) {
+                       $info = "$escaped_func ($samples_txt $countname, 
$pct%)";
+               } else {
+                       my $deltapct = sprintf "%.2f", ((100 * $delta) / 
($timemax * $factor));
+                       $deltapct = $delta > 0 ? "+$deltapct" : $deltapct;
+                       $info = "$escaped_func ($samples_txt $countname, $pct%; 
$deltapct%)";
+               }
+       }
+
+       my $nameattr = { %{ $nameattr{$func}||{} } }; # shallow clone
+       $nameattr->{class}       ||= "func_g";
+       $nameattr->{onmouseover} ||= "s('".$info."')";
+       $nameattr->{onmouseout}  ||= "c()";
+       $nameattr->{onclick}     ||= "zoom(this)";
+       $nameattr->{title}       ||= $info;
+       $im->group_start($nameattr);
+
+       my $color;
+       if ($func eq "-") {
+               $color = $vdgrey;
+       } elsif (defined $delta) {
+               $color = color_scale($delta, $maxdelta);
+       } elsif ($palette) {
+               $color = color_map($colors, $func);
+       } else {
+               $color = color($colors, $hash, $func);
+       }
+       $im->filledRectangle($x1, $y1, $x2, $y2, $color, 'rx="2" ry="2"');
+
+       my $chars = int( ($x2 - $x1) / ($fontsize * $fontwidth));
+       my $text = "";
+       if ($chars >= 3) { # room for one char plus two dots
+               $text = substr $func, 0, $chars;
+               substr($text, -2, 2) = ".." if $chars < length $func;
+               $text =~ s/&/&amp;/g;
+               $text =~ s/</&lt;/g;
+               $text =~ s/>/&gt;/g;
+       }
+       $im->stringTTF($black, $fonttype, $fontsize, 0.0, $x1 + 3, 3 + ($y1 + 
$y2) / 2, $text, "");
+
+       $im->group_end($nameattr);
+}
+
+print $im->svg;
+
+if ($palette) {
+       write_palette();
+}
+
+# vim: ts=8 sts=8 sw=8 noexpandtab
diff --git a/modules/xenon/files/xenon-generate-svgs 
b/modules/xenon/files/xenon-generate-svgs
new file mode 100755
index 0000000..d9c958b
--- /dev/null
+++ b/modules/xenon/files/xenon-generate-svgs
@@ -0,0 +1,9 @@
+#!/bin/bash
+set +C  # OK to clobber out-of-date SVGs
+for log in "/srv/xenon/logs/*/*.log"; do
+    svg="${log//log/svg}"
+    mkdir -m0755 -p "$(dirname $svg)"
+    [ ! -f "$svg" -o "$svg" -ot "$log" ] && {
+        /usr/local/bin/flamegraph.pl --minwidth=1 "$log" > "$svg"
+    }
+done
diff --git a/modules/xenon/files/xenon-log b/modules/xenon/files/xenon-log
new file mode 100755
index 0000000..73cb3c3
--- /dev/null
+++ b/modules/xenon/files/xenon-log
@@ -0,0 +1,91 @@
+#!/usr/bin/env python
+# -*- coding: utf-8 -*-
+"""
+  xenon-log
+  ~~~~~~~~~
+  
+  `xenon` is a built-in HHVM extension that periodically captures
+  stacktraces of running PHP code. This tool reads xenon-generated
+  traces via redis and logs them to disk.
+
+"""
+from __future__ import print_function
+
+import sys
+reload(sys)
+sys.setdefaultencoding('utf-8')
+
+import argparse
+import datetime
+import errno
+import logging
+import os
+import os.path
+
+import redis
+import yaml
+
+
+parser = argparse.ArgumentParser()
+parser.add_argument('config', nargs='?', default='/etc/xenon/config.yaml',
+                    type=argparse.FileType('r'))
+args = parser.parse_args()
+
+with args.config as f:
+    config = yaml.load(f)
+
+
+class TimeLog(object):
+
+    base_path = config.get('base_path', '/srv/xenon/logs')
+
+    def __init__(self, period, format, retain):
+        self.period = period
+        self.format = format
+        self.retain = retain
+        self.path = os.path.join(self.base_path, period)
+        try:
+            os.makedirs(self.path, 0755)
+        except OSError as exc:
+            if exc.errno != errno.EEXIST: raise
+
+    def get_time_from_file(self, file_name):
+        return datetime.datetime.strptime(file_name, self.format + '.log')
+
+    def get_file_from_time(self, time=None):
+        time = datetime.datetime.utcnow() if time is None else time
+        return time.strftime(self.format) + '.log'
+
+    def write(self, message, time=None):
+        file = os.path.join(self.path, self.get_file_from_time(time))
+        if not os.path.isfile(file):
+            self.prune_files()
+        with open(file, 'a') as f:
+            print(message, file=f)
+
+    def prune_files(self):
+        files = {}
+        for file in os.listdir(self.path):
+            try:
+                files[file] = self.get_time_from_file(file)
+            except ValueError:
+                continue
+        files = list(sorted(files, key=files.get, reverse=True))
+        for file in files[self.retain:]:
+            try:
+                os.remove(os.path.join(path, file))
+            except OSError:
+                continue
+
+
+
+logs = [TimeLog(**log) for log in config['logs']]
+conn = redis.Redis(config['redis'])
+pubsub = conn.pubsub()
+pubsub.subscribe('xenon')
+
+for message in pubsub.listen():
+    data = message['data']
+    time = datetime.datetime.utcnow()
+    for log in logs:
+        log.write(data, time)
diff --git a/modules/xenon/files/xenon-log.conf 
b/modules/xenon/files/xenon-log.conf
new file mode 100644
index 0000000..1858e81
--- /dev/null
+++ b/modules/xenon/files/xenon-log.conf
@@ -0,0 +1,13 @@
+# xenon-log - Log Xenon traces to disk.
+
+description "Xenon trace logger"
+
+start on (local-filesystems and net-device-up IFACE!=lo)
+
+setuid www-data
+setgid www-data
+
+respawn
+respawn limit unlimited
+
+exec /usr/bin/python /usr/local/bin/xenon-log /etc/xenon-log.yaml
diff --git a/modules/xenon/manifests/init.pp b/modules/xenon/manifests/init.pp
new file mode 100644
index 0000000..541558f
--- /dev/null
+++ b/modules/xenon/manifests/init.pp
@@ -0,0 +1,133 @@
+# == Class: xenon
+#
+# Xenon is an HHVM extension that periodically captures a stack trace of
+# PHP code. MediaWiki servers send the captured trace via Redis pub/sub.
+# This class implements an aggregator that stores captured traces to disk
+# and generates SVG flame graphs from them.
+#
+# === Parameters
+#
+# [*ensure*]
+#   Description
+#
+# [*redis_host*]
+#   Address of Redis server that is publishing Xenon traces.
+#   Default: '127.0.0.1'.
+#
+# [*redis_port*]
+#   Port of Redis server that is publishing Xenon traces.
+#   Default: 6379.
+#
+# === Examples
+#
+#  class { 'xenon':
+#      ensure     => present,
+#      redis_host => 'fluorine.eqiad.wmnet',
+#      redis_port => 6379,
+#  }
+#
+class xenon(
+    $ensure = present,
+    $redis_host = '127.0.0.1',
+    $redis_port = 6379,
+) {
+    require_package('python-redis')
+    require_package('python-yaml')
+
+    $config = {
+        base_path => '/srv/xenon/logs',
+        redis     => {
+            host => $redis_host,
+            port => $redis_port,
+        },
+        logs      => [
+            { period => 'hourly',  format => '%Y-%m-%d_%H', retain => 24 },
+            { period => 'daily',   format => '%Y-%m-%d',    retain => 30 },
+            { period => 'weekly',  format => '%Y-%W',       retain => 52 },
+            { period => 'monthly', format => '%Y-%m',       retain => 12 },
+        ],
+    }
+
+    group { 'xenon':
+        ensure => $ensure,
+    }
+
+    user { 'xenon':
+        ensure     => $ensure,
+        gid        => 'xenon',
+        shell      => '/bin/false',
+        home       => '/nonexistent',
+        system     => true,
+        managehome => false,
+    }
+
+    file { '/srv/xenon':
+        ensure => ensure_directory($ensure),
+        owner  => 'xenon',
+        group  => 'xenon',
+        mode   => '0755',
+        before => Service['xenon-log'],
+    }
+
+    file { '/etc/xenon-log.yaml':
+        ensure  => $ensure,
+        content => ordered_yaml($config),
+        owner   => 'root',
+        group   => 'root',
+        mode    => '0444',
+        notify  => Service['xenon-log'],
+    }
+
+    file { '/usr/local/bin/xenon-log':
+        ensure => $ensure,
+        source => 'puppet:///modules/xenon/xenon-log',
+        owner  => 'root',
+        group  => 'root',
+        mode   => '0555',
+        notify => Service['xenon-log'],
+    }
+
+    file { '/etc/init/xenon-log.conf':
+        ensure => $ensure,
+        source => 'puppet:///modules/xenon/xenon-log.conf',
+        notify => Service['xenon-log'],
+    },
+
+    service { 'xenon-log':
+        ensure   => ensure_service($ensure),
+        provider => 'upstart',
+    }
+
+
+    # This is the Perl script that generates flame graphs.
+    # It comes from <https://github.com/brendangregg/FlameGraph>.
+
+    file { '/usr/local/bin/flamegraph.pl':
+        ensure => $ensure,
+        source => 'puppet:///modules/xenon/flamegraph.pl',
+        owner  => 'root',
+        group  => 'root',
+        mode   => '0555',
+        notify => Service['xenon-log'],
+    }
+
+
+    # Walks /srv/xenon/logs looking for log files which do not have a
+    # corresponding SVG file and calls flamegraph.pl on each of them.
+
+    file { '/usr/local/bin/xenon-generate-svgs':
+        ensure => $ensure,
+        source => 'puppet:///modules/xenon/xenon-generate-svgs',
+        owner  => 'root',
+        group  => 'root',
+        mode   => '0555',
+        before => Cron['xenon_generate_svgs'],
+    }
+
+    cron { 'xenon_generate_svgs':
+        ensure  => $ensure,
+        command => '/usr/local/bin/xenon-generate-svgs',
+        user    => 'xenon',
+        minute  => '*/15',
+    }
+}

-- 
To view, visit https://gerrit.wikimedia.org/r/179791
To unsubscribe, visit https://gerrit.wikimedia.org/r/settings

Gerrit-MessageType: newchange
Gerrit-Change-Id: I09926c8c26da29baedcf5ab3ef219770690be30a
Gerrit-PatchSet: 1
Gerrit-Project: operations/puppet
Gerrit-Branch: production
Gerrit-Owner: Ori.livneh <[email protected]>

_______________________________________________
MediaWiki-commits mailing list
[email protected]
https://lists.wikimedia.org/mailman/listinfo/mediawiki-commits

Reply via email to