mirror of
				https://github.com/Telecominfraproject/wlan-lanforge-scripts.git
				synced 2025-11-03 20:27:54 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1387 lines
		
	
	
		
			46 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
			
		
		
	
	
			1387 lines
		
	
	
		
			46 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
//
 | 
						|
// LANforge-GUI Source Code
 | 
						|
// Copyright (C) 1999-2020  Candela Technologies Inc
 | 
						|
// http://www.candelatech.com
 | 
						|
//
 | 
						|
// This program is free software; you can redistribute it and/or
 | 
						|
// modify it under the terms of the GNU Library General Public License
 | 
						|
// as published by the Free Software Foundation; either version 2
 | 
						|
// of the License, or (at your option) any later version.
 | 
						|
//
 | 
						|
// This program is distributed in the hope that it will be useful,
 | 
						|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
// GNU General Public License for more details.
 | 
						|
//
 | 
						|
// You should have received a copy of the GNU Library General Public License
 | 
						|
// along with this program; if not, write to the Free Software
 | 
						|
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
 | 
						|
//
 | 
						|
// Contact:  Candela Technologies <support@candelatech.com> if you have any
 | 
						|
//           questions.
 | 
						|
//
 | 
						|
 | 
						|
import java.text.*;
 | 
						|
import java.util.concurrent.*;
 | 
						|
import java.io.*;
 | 
						|
import java.net.URL;
 | 
						|
import java.util.*;
 | 
						|
import java.nio.file.*;
 | 
						|
 | 
						|
/** Process a set of test results generated by the CI/CD process for a test type in a single testbed.
 | 
						|
 * There are currently no automated comparisons done across different testbeds.
 | 
						|
 * Generate historical graphs and links to each test run.
 | 
						|
 * Each test results directory would be packaged up by the lf_gui_report_summary.pl script,
 | 
						|
 * called by the basic_regression.bash automated regression test script.
 | 
						|
 * Example:  java kpi --dir /var/www/html/tip/testbeds/ferndale-basic-01/reports/basic
 | 
						|
 */
 | 
						|
public class kpi {
 | 
						|
   String lc_osname;
 | 
						|
   String home_dir;
 | 
						|
 | 
						|
   static final String out_sep = "\t";
 | 
						|
   static final String in_sep = "\t";
 | 
						|
   
 | 
						|
   public static int PRIORITY_IDX = 6;
 | 
						|
   public static int TEST_ID_IDX = 7;
 | 
						|
   public static int SHORT_DESC_IDX = 8;
 | 
						|
   public static int PASS_FAIL_IDX = 9;
 | 
						|
   public static int NUMERIC_SCORE_IDX = 10;
 | 
						|
   public static int NOTES_IDX = 11;
 | 
						|
   public static int UNITS_IDX = 12;
 | 
						|
   public static int GRAPH_GROUP_IDX = 13;
 | 
						|
   public static int SUBTEST_PASS_IDX = 14;
 | 
						|
   public static int SUBTEST_FAIL_IDX = 15;
 | 
						|
 | 
						|
   public static String TESTBED_TEMPLATE = "testbed_template.html";
 | 
						|
   public static String AP_AUTO_BASIC_CX = "ap_auto_basic_cx";
 | 
						|
 | 
						|
   public static String HTML_COLOR_PASS = "#bee7aa";
 | 
						|
   public static String HTML_COLOR_FAIL = "#f9c5c0";
 | 
						|
   public static String HTML_COLOR_WARNING = "#f8f6ad";
 | 
						|
 | 
						|
   public static String PF = "PASS-FAIL";
 | 
						|
 | 
						|
   public kpi() {
 | 
						|
      priv_init();
 | 
						|
   }
 | 
						|
 | 
						|
   static String yellowTd(String txt) {
 | 
						|
      return "<td><span style='background:" + HTML_COLOR_WARNING + ";color:black;'>" + txt + "</span></td>";
 | 
						|
   }
 | 
						|
 | 
						|
   static String redTd(String txt) {
 | 
						|
      return "<td><span style='background:" + HTML_COLOR_FAIL + ";color:black;'>" + txt + "</span></td>";
 | 
						|
   }
 | 
						|
 | 
						|
   static String greenTd(String txt) {
 | 
						|
      return "<td><span style='background:" + HTML_COLOR_PASS + ";color:black;'>" + txt + "</span></td>";
 | 
						|
   }
 | 
						|
 | 
						|
   public boolean is_mac() {
 | 
						|
      return lc_osname.startsWith("mac os x");
 | 
						|
   }
 | 
						|
   public boolean is_win() {
 | 
						|
      return lc_osname.startsWith("windows");
 | 
						|
   }
 | 
						|
   public boolean is_linux() {
 | 
						|
      return lc_osname.startsWith("linux");
 | 
						|
   }
 | 
						|
 | 
						|
   private void priv_init() {
 | 
						|
      lc_osname = System.getProperty("os.name").toLowerCase();
 | 
						|
      home_dir = System.getProperty("user.home");
 | 
						|
   }
 | 
						|
 | 
						|
   public static void main(String[] args) {
 | 
						|
      kpi k = new kpi();
 | 
						|
      try {
 | 
						|
         k.work(args);
 | 
						|
      }
 | 
						|
      catch (Exception ee) {
 | 
						|
         ee.printStackTrace();
 | 
						|
      }
 | 
						|
   }
 | 
						|
 | 
						|
   public static String toStringNum(double i, int precision) {
 | 
						|
      NumberFormat nf = NumberFormat.getInstance();
 | 
						|
      if (0.0 == i) {
 | 
						|
         // Don't pad precision on '0', just adds needless clutter.
 | 
						|
         return nf.format(i);
 | 
						|
      }
 | 
						|
      else {
 | 
						|
         nf = (NumberFormat)(nf.clone());
 | 
						|
         nf.setMaximumFractionDigits(precision);
 | 
						|
         nf.setMinimumFractionDigits(precision);
 | 
						|
         return nf.format(i);
 | 
						|
      }
 | 
						|
   }
 | 
						|
 | 
						|
   public void work(String[] args) {
 | 
						|
      String dir = null;
 | 
						|
      String results_url = "";
 | 
						|
      String caseid = "";
 | 
						|
      String slack_fname = "";
 | 
						|
      String testbed_name = "LANforge CICD";
 | 
						|
 | 
						|
      for (int i = 0; i<args.length; i++) {
 | 
						|
         if (args[i].equals("--dir")) {
 | 
						|
            dir = args[i+1];
 | 
						|
         }
 | 
						|
         else if (args[i].equals("--caseid")) {
 | 
						|
            caseid = args[i+1];
 | 
						|
         }
 | 
						|
         else if (args[i].equals("--results_url")) {
 | 
						|
            results_url = args[i+1];
 | 
						|
         }
 | 
						|
         else if (args[i].equals("--slack_fname")) {
 | 
						|
            slack_fname = args[i+1];
 | 
						|
         }
 | 
						|
         else if (args[i].equals("--testbed_name")) {
 | 
						|
            testbed_name = args[i+1];
 | 
						|
         }
 | 
						|
         else if (args[i].equals("--help") || args[i].equals("-h") || args[i].equals("-?")) {
 | 
						|
            System.out.println("Usage: $0 --dir /path/to/test-collection\n" +
 | 
						|
                               "--caseid [case-id]\n" +
 | 
						|
                               "--results_url [url]\n" +
 | 
						|
                               "--slack_fname [slack url webhook file name]\n" +
 | 
						|
                               "--testbed_name [My Testbed]");
 | 
						|
            System.exit(0);
 | 
						|
         }
 | 
						|
      }
 | 
						|
 | 
						|
      Hashtable<String, String> test_names = new Hashtable();
 | 
						|
      Vector<String> test_namesv = new Vector();
 | 
						|
      Vector<Run> runs = new Vector();
 | 
						|
 | 
						|
      test_names.put(PF, PF);
 | 
						|
      test_namesv.add(PF);
 | 
						|
 | 
						|
      try {
 | 
						|
         DirectoryStream<Path> stream = Files.newDirectoryStream(Paths.get(dir));
 | 
						|
         for (Path file: stream) {
 | 
						|
            File f = file.toFile(); // this is the test run dir
 | 
						|
            // Inside of it is individual tests.
 | 
						|
            if (!f.isDirectory()) {
 | 
						|
               continue;
 | 
						|
            }
 | 
						|
            //System.out.println("Checking sub-directory/file (run): " + f.getAbsolutePath());
 | 
						|
            DirectoryStream<Path> stream2 = Files.newDirectoryStream(file);
 | 
						|
            Run run = null;
 | 
						|
 | 
						|
            for (Path file2: stream2) {
 | 
						|
               File f2 = file2.toFile(); // this is the test case dir in the test run
 | 
						|
               //System.out.println("Checking test-case directory/file: " + f2.getAbsolutePath());
 | 
						|
               // Directory full of test results?
 | 
						|
               if (f2.isDirectory()) {
 | 
						|
                  DirectoryStream<Path> stream3 = Files.newDirectoryStream(file2);
 | 
						|
                  Test test = new Test(f2.getName());
 | 
						|
                  if (run == null) {
 | 
						|
                     run = new Run(f.getName());
 | 
						|
                     runs.add(run);
 | 
						|
                  }
 | 
						|
                  System.out.println("Run: " + run + " adding test: " + test);
 | 
						|
                  run.addTest(test);
 | 
						|
 | 
						|
                  // TODO:
 | 
						|
                  // Add log processing.  In particular, look in console logs for things like:
 | 
						|
                  // "ath10k_pci 0000:01:00.0: wmi command 36967 timeout, restarting hardware"
 | 
						|
                  // "WARNING: CPU: 0 PID: 8907 at backports-4.
 | 
						|
 | 
						|
                  for (Path file3: stream3) {
 | 
						|
                     File f3 = file3.toFile();
 | 
						|
                     String fname = f3.getName();
 | 
						|
                     if (fname.equals("kpi.csv")) {
 | 
						|
                        try {
 | 
						|
                           BufferedReader br = new BufferedReader(new FileReader(f3));
 | 
						|
                           if (test_names.get(f2.getName()) == null) {
 | 
						|
                              test_names.put(f2.getName(), f2.getName());
 | 
						|
                              test_namesv.add(f2.getName());
 | 
						|
                           }
 | 
						|
                           String line;
 | 
						|
                           while ((line = br.readLine()) != null) {
 | 
						|
                              test.addLine(line);
 | 
						|
                           }
 | 
						|
                        }
 | 
						|
                        catch (FileNotFoundException enf) {
 | 
						|
                           // ignore
 | 
						|
                        }
 | 
						|
                        catch (Exception e) {
 | 
						|
                           e.printStackTrace();
 | 
						|
                        }
 | 
						|
                     }// if kpi.csv
 | 
						|
                     else if (fname.startsWith("kpi-") && fname.endsWith(".png")) {
 | 
						|
                        test.addKpiImage(fname);
 | 
						|
                     }
 | 
						|
                     else if (fname.equals("logs")) {
 | 
						|
                        File logs_csv = new File(f3.getAbsolutePath() + File.separator + "logs.csv");
 | 
						|
                        if (logs_csv.exists()) {
 | 
						|
                           try {
 | 
						|
                              BufferedReader br = new BufferedReader(new FileReader(logs_csv));
 | 
						|
                              String line;
 | 
						|
                              while ((line = br.readLine()) != null) {
 | 
						|
                                 test.addLogCsv(line);
 | 
						|
                              }
 | 
						|
                           }
 | 
						|
                           catch (Exception e) {
 | 
						|
                              e.printStackTrace();
 | 
						|
                           }
 | 
						|
                        }
 | 
						|
                     }
 | 
						|
                  }// for all files in the test dir  
 | 
						|
               }
 | 
						|
            }
 | 
						|
         }
 | 
						|
      } catch (IOException | DirectoryIteratorException x) {
 | 
						|
         // IOException can never be thrown by the iteration.
 | 
						|
         // In this snippet, it can only be thrown by newDirectoryStream.
 | 
						|
         System.err.println(x);
 | 
						|
      }
 | 
						|
 | 
						|
      // Sort runs so that earliest is first.
 | 
						|
      class SortbyDate implements Comparator<Run> { 
 | 
						|
         // Used for sorting in ascending order of 
 | 
						|
         // roll number 
 | 
						|
         public int compare(Run a, Run b) { 
 | 
						|
            long c = a.getDateMs() - b.getDateMs();
 | 
						|
            if (c < 0)
 | 
						|
               return -1;
 | 
						|
            if (c > 0)
 | 
						|
               return 1;
 | 
						|
            return 0;
 | 
						|
         }
 | 
						|
      }
 | 
						|
 | 
						|
      runs.sort(new SortbyDate());
 | 
						|
 | 
						|
      // Create meta-test results for each run.
 | 
						|
      for (int i = 0; i<runs.size(); i++) {
 | 
						|
         Run run = runs.elementAt(i);
 | 
						|
         Test t = new Test(PF);
 | 
						|
         run.addTest(t);
 | 
						|
         // Build fake kpi title and add that,
 | 
						|
         // then add fake kpi results line
 | 
						|
         // Example lines from a kpi.csv
 | 
						|
         //Date	test-rig	dut-hw-version	dut-sw-version	dut-model-num	dut-serial-num	test-priority	test-id	short-description	pass/fail	numeric-score	test details	Units	Graph-Group	Subtest-Pass	Subtest-Fail
 | 
						|
         //1590190791848	Ferndale-01-Basic	Linksys-EA8300	d88ea40	Linksys-EA8300		90	AP Auto	Basic Client Connectivity Dual-Band MIN		8.0	Minimum CX time (ms)	Time (ms)	CX-Time	0	0
 | 
						|
 | 
						|
         t.addLine("Date	test-rig	dut-hw-version	dut-sw-version	dut-model-num	dut-serial-num	test-priority	test-id	short-description	pass/fail	numeric-score	test details	Units	Graph-Group	Subtest-Pass	Subtest-Fail");
 | 
						|
         t.addLine(run.getDateMs() + kpi.in_sep + run.getTestRig() + kpi.in_sep + run.getDutHwVer() + kpi.in_sep + run.getDutSwVer()
 | 
						|
                   + kpi.in_sep + run.getDutModelNum() + kpi.in_sep + run.getDutSerNum() + kpi.in_sep + "100" + kpi.in_sep + PF
 | 
						|
                   + kpi.in_sep + "Total Pass" + kpi.in_sep + kpi.in_sep + run.getPass() + kpi.in_sep + "Total passing tests and sub-tests for this run."
 | 
						|
                   + kpi.in_sep + "Number" + kpi.in_sep + PF + kpi.in_sep + "0" + kpi.in_sep + "0");
 | 
						|
         t.addLine(run.getDateMs() + kpi.in_sep + run.getTestRig() + kpi.in_sep + run.getDutHwVer() + kpi.in_sep + run.getDutSwVer()
 | 
						|
                   + kpi.in_sep + run.getDutModelNum() + kpi.in_sep + run.getDutSerNum() + kpi.in_sep + "100" + kpi.in_sep + PF
 | 
						|
                   + kpi.in_sep + "Total Failures" + kpi.in_sep + kpi.in_sep + run.getFail() + kpi.in_sep + "Total failing tests and sub-tests for this run."
 | 
						|
                   + kpi.in_sep + "Number" + kpi.in_sep + PF + kpi.in_sep + "0" + kpi.in_sep + "0");
 | 
						|
      }
 | 
						|
 | 
						|
      // Link to latest test run that has the test id
 | 
						|
      Hashtable<String, Run> test_id_links = new Hashtable();
 | 
						|
 | 
						|
      // We have read everything into memory.
 | 
						|
      // For each test, generate data over time.
 | 
						|
      Hashtable<String, History> hist_data = new Hashtable();
 | 
						|
      Vector<History> hist_datav = new Vector();
 | 
						|
      for (String tname: test_namesv) {
 | 
						|
         System.out.println("Checking testname: " + tname + " runs.size: " + runs.size());
 | 
						|
         // For each test, find all runs that have this test and consolidate data
 | 
						|
         for (int i = 0; i<runs.size(); i++) {
 | 
						|
            Run run = runs.elementAt(i);
 | 
						|
            Test t = run.findTest(tname);
 | 
						|
            System.out.println("Run [" + i + "]  Test: " + t);
 | 
						|
            if (t != null) {
 | 
						|
               test_id_links.put(tname, run);
 | 
						|
               try {
 | 
						|
                  History hist = hist_data.get(tname);
 | 
						|
                  if (hist == null) {
 | 
						|
                     hist = new History(tname);
 | 
						|
                     hist_data.put(tname, hist);
 | 
						|
                     hist_datav.add(hist);
 | 
						|
                     System.out.println("Adding history datav, tname: " + tname);
 | 
						|
                  }
 | 
						|
                  for (int z = 0; z<t.data.size(); z++) {
 | 
						|
                     Row r = t.data.elementAt(z);
 | 
						|
                     HistRow csv = hist.findRow(r.getShortDescKey());
 | 
						|
                     if (csv == null) {
 | 
						|
                        csv = new HistRow(r);
 | 
						|
                        hist.addRow(csv);
 | 
						|
                        System.out.println("Adding history row: " + r.getShortDescKey());
 | 
						|
                     }
 | 
						|
                     String score =  t.data.elementAt(z).getScore();
 | 
						|
                     csv.csv.append(i + kpi.out_sep + score + System.lineSeparator());
 | 
						|
                     csv.scores.add(Double.valueOf(score));
 | 
						|
                  }
 | 
						|
               }
 | 
						|
               catch (Exception eee) {
 | 
						|
                  eee.printStackTrace();
 | 
						|
               }
 | 
						|
            }
 | 
						|
         }
 | 
						|
      }
 | 
						|
 | 
						|
      StringBuffer scores = new StringBuffer();
 | 
						|
      StringBuffer plots = new StringBuffer();
 | 
						|
      StringBuffer groups = new StringBuffer();
 | 
						|
      StringBuffer runs_rows = new StringBuffer();
 | 
						|
 | 
						|
      // For all per-test history, print out csv files
 | 
						|
      for (History hist : hist_datav) {
 | 
						|
         String hk = hist.getName();
 | 
						|
         Hashtable<String, Vector> groupsh = new Hashtable();
 | 
						|
         Vector<String> groupsn = new Vector();
 | 
						|
         System.out.println("Checking history-data: " + hk);
 | 
						|
         for (HistRow csv : hist.csv) {
 | 
						|
            String ck = csv.getName();
 | 
						|
            String title = csv.getTitle();
 | 
						|
            String units = csv.getUnits();
 | 
						|
            String g = csv.getGraphGroup();
 | 
						|
 | 
						|
            System.out.println("csv name: " + ck);
 | 
						|
 | 
						|
            if (!(g.equals("NA") || units.equals(""))) {
 | 
						|
               Vector<HistRow> gv = groupsh.get(g);
 | 
						|
               if (gv == null) {
 | 
						|
                  gv = new Vector();
 | 
						|
                  groupsh.put(g, gv);
 | 
						|
                  groupsn.add(g);
 | 
						|
               }
 | 
						|
               gv.add(csv);
 | 
						|
            }
 | 
						|
 | 
						|
            if (units.equals("NA") || units.equals("")) {
 | 
						|
               units = "Data";
 | 
						|
            }
 | 
						|
            try {
 | 
						|
               String cf = dir + File.separator + hk + "::" + ck + ".csv";
 | 
						|
               System.out.println("Creating gnuplot csv: " + cf);
 | 
						|
               FileWriter f = new FileWriter(cf);
 | 
						|
               f.write(csv.csv.toString());
 | 
						|
               f.close();
 | 
						|
               csv.setFname(cf);
 | 
						|
 | 
						|
               ShellExec exec = new ShellExec(true, true);
 | 
						|
               int rv = exec.execute("gnuplot", null, true,
 | 
						|
                                     "-e", "set ylabel \"" + units + "\"",
 | 
						|
                                     "-e", "filename='" + cf + "'",
 | 
						|
                                     "-e", "set title '" + title + "'",
 | 
						|
                                     "default.plot");
 | 
						|
               if (rv != 0) {
 | 
						|
                  System.out.println("gnuplot for filename: " + cf + " rv: " + rv);
 | 
						|
                  System.out.println(exec.getOutput());
 | 
						|
                  System.out.println(exec.getError());
 | 
						|
               }
 | 
						|
 | 
						|
               File png = new File("plot.png");
 | 
						|
               if (!png.exists()) {
 | 
						|
                  System.out.println("ERROR:  File: " + png + " does not exist.");
 | 
						|
               }
 | 
						|
               String npng = hk + "::" + ck + ".png";
 | 
						|
               String nplot_name = dir + File.separator + npng;
 | 
						|
               System.out.println("Rename plot: " + png + " to: " + nplot_name);
 | 
						|
               File nf = new File(nplot_name);
 | 
						|
               Path nfp = Files.move(png.toPath(), nf.toPath(), StandardCopyOption.REPLACE_EXISTING);//boolean renamed = png.renameTo(nf);
 | 
						|
               if (!nf.exists()) {
 | 
						|
                  System.out.println("ERROR:  New file: " + nf + " does not exist.");
 | 
						|
               }
 | 
						|
 | 
						|
               exec = new ShellExec(true, true);
 | 
						|
               rv = exec.execute("gnuplot", null, true, "-e", "filename='" + cf + "'",
 | 
						|
                                     "mini.plot");
 | 
						|
               if (rv != 0) {
 | 
						|
                  System.out.println("mini gnuplot for filename: " + cf + " rv: " + rv);
 | 
						|
                  System.out.println(exec.getOutput());
 | 
						|
                  System.out.println(exec.getError());
 | 
						|
               }
 | 
						|
 | 
						|
               png = new File("plot.png");
 | 
						|
               String npngt = hk + "::" + ck + "-thumb.png";
 | 
						|
               Files.move(png.toPath(), new File(dir + File.separator + npngt).toPath(), StandardCopyOption.REPLACE_EXISTING);
 | 
						|
               
 | 
						|
               String hk_str = hk;
 | 
						|
               Run last = test_id_links.get(hk);
 | 
						|
               if (!hk.equals(PF)) {
 | 
						|
                  if (last != null) {
 | 
						|
                     hk_str = "<a href=\"" + last.getName() + "/" + hk + "/index.html\">" + hk + "</a>";
 | 
						|
                  }
 | 
						|
               }
 | 
						|
 | 
						|
               StringBuffer change = new StringBuffer();
 | 
						|
               Double cur = csv.scores.elementAt(csv.scores.size() - 1);
 | 
						|
               change.append("Last Value: " + toStringNum(cur, 2));
 | 
						|
               if (csv.scores.size() > 1) {
 | 
						|
                  Double prev = csv.scores.elementAt(csv.scores.size() - 2);
 | 
						|
                  change.append("<br>Delta for last run: " + toStringNum((cur - prev), 2));
 | 
						|
                  if (cur != 0) {
 | 
						|
                     change.append("<br>Percentage change: " + toStringNum(100.0 * ((cur - prev) / cur), 2) + "%");
 | 
						|
                  }
 | 
						|
               }
 | 
						|
 | 
						|
               String row_str = ("<tr><td>" + hk_str + "</td><td>" + title + "</td><td><a href=\"./" + npng + "\"><img src=\"./" + npngt + "\"></a></td><td>"
 | 
						|
                                 + change + "</td></tr>\n");
 | 
						|
               if (csv.getPriority() >= 100) {
 | 
						|
                  scores.append(row_str);
 | 
						|
               }
 | 
						|
               else {
 | 
						|
                  plots.append(row_str);
 | 
						|
               }
 | 
						|
            }
 | 
						|
            catch (Exception ee) {
 | 
						|
               ee.printStackTrace();
 | 
						|
            }
 | 
						|
         }
 | 
						|
 | 
						|
         // Graph groups
 | 
						|
         for (String g : groupsn) {
 | 
						|
            Vector<HistRow> datasets = groupsh.get(g);
 | 
						|
            if (datasets.size() < 2) {
 | 
						|
               continue; // group of 1 doesn't count
 | 
						|
            }
 | 
						|
 | 
						|
            HistRow csv0 = datasets.elementAt(0);
 | 
						|
            String title = g;
 | 
						|
            String units = csv0.getUnits();
 | 
						|
 | 
						|
            // Don't group scores
 | 
						|
            if (units.equalsIgnoreCase("Score")) {
 | 
						|
               continue;
 | 
						|
            }
 | 
						|
 | 
						|
            if (units.equals("NA") || units.equals("")) {
 | 
						|
               units = "Data";
 | 
						|
            }
 | 
						|
 | 
						|
            System.out.println("title: " + title + " units: " + units);
 | 
						|
            try {
 | 
						|
               StringBuffer plot = new StringBuffer("plot ");
 | 
						|
               StringBuffer mplot = new StringBuffer("plot ");
 | 
						|
               boolean first = true;
 | 
						|
               for (HistRow csv: datasets) {
 | 
						|
                  if (!first) {
 | 
						|
                     plot.append(", ");
 | 
						|
                     mplot.append(", ");
 | 
						|
                  }
 | 
						|
                  plot.append("\'" + csv.getFname() + "\' using 1:2 with lines title \'" + csv.getName().replace("_", " ") + "\'");
 | 
						|
                  mplot.append("\'" + csv.getFname() + "\' using 1:2 with lines notitle");
 | 
						|
                  first = false;
 | 
						|
               }
 | 
						|
 | 
						|
               BufferedReader br = new BufferedReader(new FileReader("default_group.plot"));
 | 
						|
               FileWriter f = new FileWriter("tmp.plot");
 | 
						|
               String line;
 | 
						|
               while ((line = br.readLine()) != null) {
 | 
						|
                  f.write(line);
 | 
						|
                  f.write(System.lineSeparator());
 | 
						|
               }
 | 
						|
               f.write(plot.toString());
 | 
						|
               f.write(System.lineSeparator());
 | 
						|
               f.close();
 | 
						|
               br.close();
 | 
						|
 | 
						|
               //System.out.println("group plot: " + plot);
 | 
						|
               ShellExec exec = new ShellExec(true, true);
 | 
						|
               int rv = exec.execute("gnuplot", null, true,
 | 
						|
                                     "-e", "set ylabel '" + units + "'",
 | 
						|
                                     "-e", "set title '" + title + "'",
 | 
						|
                                     "tmp.plot");
 | 
						|
               if (rv != 0) {
 | 
						|
                  System.out.println("gnuplot for group: " + title + " rv: " + rv);
 | 
						|
                  System.out.println(exec.getOutput());
 | 
						|
                  System.out.println(exec.getError());
 | 
						|
               }
 | 
						|
 | 
						|
               File png = new File("plot.png");
 | 
						|
               String npng = hk + "::" + g + ".png";
 | 
						|
               Files.move(png.toPath(), new File(dir + File.separator + npng).toPath(), StandardCopyOption.REPLACE_EXISTING);
 | 
						|
 | 
						|
               br = new BufferedReader(new FileReader("mini_group.plot"));
 | 
						|
               f = new FileWriter("tmp.plot");
 | 
						|
               while ((line = br.readLine()) != null) {
 | 
						|
                  f.write(line);
 | 
						|
                  f.write(System.lineSeparator());
 | 
						|
               }
 | 
						|
               f.write(mplot.toString());
 | 
						|
               f.write(System.lineSeparator());
 | 
						|
               f.close();
 | 
						|
               br.close();
 | 
						|
 | 
						|
               exec = new ShellExec(true, true);
 | 
						|
               rv = exec.execute("gnuplot", null, true,
 | 
						|
                                 "tmp.plot");
 | 
						|
               if (rv != 0) {
 | 
						|
                  System.out.println("mini gnuplot for group: " + title + " rv: " + rv);
 | 
						|
                  System.out.println(exec.getOutput());
 | 
						|
                  System.out.println(exec.getError());
 | 
						|
               }
 | 
						|
 | 
						|
               png = new File("plot.png");
 | 
						|
               String npngt = hk + "::" + g + "-thumb.png";
 | 
						|
               Files.move(png.toPath(), new File(dir + File.separator + npngt).toPath(), StandardCopyOption.REPLACE_EXISTING);
 | 
						|
 | 
						|
               String hk_str = hk;
 | 
						|
               Run last = test_id_links.get(hk);
 | 
						|
               if (!hk.equals(PF)) {
 | 
						|
                  if (last != null) {
 | 
						|
                     hk_str = "<a href=\"" + last.getName() + "/" + hk + "/index.html\">" + hk + "</a>";
 | 
						|
                  }
 | 
						|
               }
 | 
						|
 | 
						|
               groups.append("<tr><td>" + hk_str + "</td><td>" + title + "</td><td><a href=\"./" + npng + "\"><img src=\"./" + npngt + "\"></a></td></tr>\n");
 | 
						|
 | 
						|
            }
 | 
						|
            catch (Exception ee) {
 | 
						|
               ee.printStackTrace();
 | 
						|
            }
 | 
						|
         }
 | 
						|
         
 | 
						|
      }
 | 
						|
 | 
						|
      String test_bed = "Test Bed";
 | 
						|
      String last_run = "";
 | 
						|
      String last_run_kpi_images = "";
 | 
						|
      StringBuffer pngs = new StringBuffer();
 | 
						|
 | 
						|
      boolean cp = true;
 | 
						|
      for (int i = 0; i<runs.size(); i++) {
 | 
						|
         Run run = runs.elementAt(i);
 | 
						|
         test_bed = run.getTestRig();
 | 
						|
         StringBuffer logs_str = new StringBuffer();
 | 
						|
         int logv = run.getLogBugs();
 | 
						|
         if (logv > 0) {
 | 
						|
            logs_str.append(redTd(logv + ""));
 | 
						|
         }
 | 
						|
         else {
 | 
						|
            logs_str.append(greenTd(logv + ""));
 | 
						|
         }
 | 
						|
 | 
						|
         logv = run.getLogWarnings();
 | 
						|
         if (logv > 0) {
 | 
						|
            logs_str.append(redTd(logv + ""));
 | 
						|
         }
 | 
						|
         else {
 | 
						|
            logs_str.append(greenTd(logv + ""));
 | 
						|
         }
 | 
						|
 | 
						|
         logv = run.getLogCrashes();
 | 
						|
         if (logv > 0) {
 | 
						|
            logs_str.append(redTd(logv + ""));
 | 
						|
         }
 | 
						|
         else {
 | 
						|
            logs_str.append(greenTd(logv + ""));
 | 
						|
         }
 | 
						|
 | 
						|
         logv = run.getLogRestarting();
 | 
						|
         if (logv > 0) {
 | 
						|
            logs_str.append(redTd(logv + ""));
 | 
						|
         }
 | 
						|
         else {
 | 
						|
            logs_str.append(greenTd(logv + ""));
 | 
						|
         }
 | 
						|
 | 
						|
         String row_text = ("<tr><td>" + i + "</td><td><a href=\"" + run.getName() + "/index.html\">" + run.getName() + "</a></td><td>" + run.getDate()
 | 
						|
                            + "</td><td>" + run.getDutHwVer() + "</td><td>" + run.getDutSwVer()
 | 
						|
                            + "</td><td>" + run.getDutModelNum() + "</td>" + greenTd(run.getPass() + "") + redTd(run.getFail() + "") + logs_str + "</tr>\n");
 | 
						|
            
 | 
						|
         if (i == (runs.size() - 1)) {
 | 
						|
            // Last run
 | 
						|
            int png_row_count = 0;
 | 
						|
            boolean needs_tr = true;
 | 
						|
            last_run = row_text;
 | 
						|
            for (Test t : run.testsv) {
 | 
						|
               for (String png : t.kpi_images) {
 | 
						|
                  if (png.indexOf("-print") >= 0) {
 | 
						|
                     continue; // skip the print variants of the image.
 | 
						|
                  }
 | 
						|
                  if (needs_tr) {
 | 
						|
                     pngs.append("<tr>");
 | 
						|
                     needs_tr = false;
 | 
						|
                  }
 | 
						|
                  String img_title = "Test: " + t.getName();
 | 
						|
                  String fname = run.getName() + "/" + t.getName() + "/" + png;
 | 
						|
                  pngs.append("<td><a href=\"" + fname + "\"><img src=\"" + fname + "\" style='width:400px;max-width:400px' title=\""
 | 
						|
                              + img_title + "\"></a></td>\n");
 | 
						|
                  png_row_count++;
 | 
						|
                  if (png_row_count == 2) {
 | 
						|
                     png_row_count = 0;
 | 
						|
                     pngs.append("</tr>\n");
 | 
						|
                     needs_tr = true;
 | 
						|
                  }
 | 
						|
               }
 | 
						|
            }
 | 
						|
 | 
						|
            if ((!needs_tr) && pngs.length() > 0) {
 | 
						|
               pngs.append("</tr>\n");
 | 
						|
            }
 | 
						|
 | 
						|
            String testrails_msg = ("MSG Run: " + run.getName() + " Date: " + run.getDate() + " DUT-HW: " + run.getDutHwVer() + "DUT-SW: " + run.getDutSwVer()
 | 
						|
                                    + " Passed: " + run.getPass() + " Fail: " + run.getFail()
 | 
						|
                                    + " Log-Bugs: " + run.getLogBugs() + " Log-Warnings: " + run.getLogWarnings()
 | 
						|
                                    + " Log-Crashes: " + run.getLogCrashes() + " Log-Restarting: " + run.getLogRestarting());
 | 
						|
 | 
						|
            if (!caseid.equals("")) {
 | 
						|
               try {
 | 
						|
                  // Create file for propagating results into external tool
 | 
						|
                  String ofile = dir + File.separator + "testrails.txt";
 | 
						|
                  BufferedWriter bw = new BufferedWriter(new FileWriter(ofile));
 | 
						|
             
 | 
						|
                  bw.write("CASE_ID " + caseid);
 | 
						|
                  bw.write(System.lineSeparator());
 | 
						|
                  bw.write("RUN_ID " + i);
 | 
						|
                  bw.write(System.lineSeparator());
 | 
						|
                  bw.write(testrails_msg);
 | 
						|
                  bw.write(System.lineSeparator());
 | 
						|
                  bw.write("MSG URL: " + results_url + "/" + run.getName());
 | 
						|
                  bw.write(System.lineSeparator());
 | 
						|
 | 
						|
                  bw.close();
 | 
						|
 | 
						|
                  System.out.println("See testrails results: " + ofile);
 | 
						|
               }
 | 
						|
               catch (Exception eee) {
 | 
						|
                  eee.printStackTrace();
 | 
						|
               }
 | 
						|
            }
 | 
						|
 | 
						|
            if (!slack_fname.equals("")) {
 | 
						|
               String slack_content_type = "Content-type: application/json";
 | 
						|
               String slack_msg = "{\"text\":\"<" + results_url + "|" + testbed_name + ">"
 | 
						|
                  + " Results: <" + results_url + "/" + run.getName() + "|" + run.getName() + ">"
 | 
						|
                  + " Errors: " + (run.getLogCrashes() + run.getLogRestarting() + run.getLogBugs()) + " Warnings: " + run.getLogWarnings()
 | 
						|
                  + "\"}";
 | 
						|
               try {
 | 
						|
                  BufferedReader br = new BufferedReader(new FileReader(slack_fname));
 | 
						|
                  String slack = br.readLine();
 | 
						|
                  String cmd = "curl -X POST -H '" + slack_content_type + "' --data '" + slack_msg + "' " + slack;
 | 
						|
                  System.out.println(cmd);
 | 
						|
 | 
						|
                  ShellExec exec = new ShellExec(true, true);
 | 
						|
                  int rv = exec.execute("curl", null, true,
 | 
						|
                                        "-X", "POST",
 | 
						|
                                        "-H", slack_content_type,
 | 
						|
                                        "--data", slack_msg, slack);
 | 
						|
                  if (rv != 0) {
 | 
						|
                     System.out.println("curl slack post of msg: " + slack_msg + " to: " + slack + " failed.\n");
 | 
						|
                     System.out.println(exec.getOutput());
 | 
						|
                     System.out.println(exec.getError());
 | 
						|
                  }
 | 
						|
               }
 | 
						|
               catch (Exception eeee) {
 | 
						|
                  System.out.println("slack_msg: " + slack_msg + " slack_fname: " + slack_fname);
 | 
						|
                  eeee.printStackTrace();
 | 
						|
               }
 | 
						|
            }
 | 
						|
         }
 | 
						|
 | 
						|
         runs_rows.append(row_text);
 | 
						|
 | 
						|
         if (cp) {
 | 
						|
            try {
 | 
						|
               String fname;
 | 
						|
               copy("CandelaLogo2-90dpi-200x90-trans.png", dir + File.separator + run.getName(), dir);
 | 
						|
               copy("candela_swirl_small-72h.png", dir + File.separator + run.getName(), dir);
 | 
						|
               copy("canvil.ico", dir + File.separator + run.getName(), dir);
 | 
						|
               copy("custom.css", dir + File.separator + run.getName(), dir);
 | 
						|
               copy("report.css", dir + File.separator + run.getName(), dir);
 | 
						|
 | 
						|
               cp = false;
 | 
						|
            }
 | 
						|
            catch (Exception ee) {
 | 
						|
               ee.printStackTrace();
 | 
						|
            }
 | 
						|
         }
 | 
						|
      }
 | 
						|
 | 
						|
      try {
 | 
						|
         // Read in the testbed_template.html and update it with our info
 | 
						|
         BufferedReader br = new BufferedReader(new FileReader(new File(kpi.TESTBED_TEMPLATE)));
 | 
						|
         String ofile = dir + File.separator + "index.html";
 | 
						|
         BufferedWriter bw = new BufferedWriter(new FileWriter(ofile));
 | 
						|
         String line;
 | 
						|
         while ((line = br.readLine()) != null) {
 | 
						|
            line = line.replace("___TITLE___", test_bed + " Report History");
 | 
						|
            line = line.replace("___SCORE_RUNS___", scores.toString());
 | 
						|
            line = line.replace("___GROUP_GRAPHS___", groups.toString());
 | 
						|
            line = line.replace("___DATA_GRAPHS___", plots.toString());
 | 
						|
            line = line.replace("___TEST_RUNS___", runs_rows.toString());
 | 
						|
            line = line.replace("___LATEST_RUN___", last_run);
 | 
						|
            line = line.replace("___LATEST_RUN_PNGS___", pngs.toString());
 | 
						|
            bw.write(line);
 | 
						|
            bw.write(System.lineSeparator());
 | 
						|
         }
 | 
						|
         
 | 
						|
         br.close();
 | 
						|
         bw.close();
 | 
						|
 | 
						|
         System.out.println("See " + ofile);
 | 
						|
      }
 | 
						|
      catch (Exception eee) {
 | 
						|
         eee.printStackTrace();
 | 
						|
      }
 | 
						|
   } // ~work()
 | 
						|
 | 
						|
   public void copy(String fname, String from, String to) throws Exception {
 | 
						|
      Path fromp = Paths.get(from + File.separator + fname);
 | 
						|
      Path top = Paths.get(to + File.separator + fname);
 | 
						|
      Files.copy(fromp, top, StandardCopyOption.REPLACE_EXISTING);
 | 
						|
   }
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/** This holds a datapoint for a particular test result for each of the Runs.
 | 
						|
 * This is used to generate historical graphs and comparisons.
 | 
						|
 * If a test generates two KPI results, there will be two HistRows for that
 | 
						|
 * test case.
 | 
						|
 */
 | 
						|
class HistRow {
 | 
						|
   String fname;
 | 
						|
   String name;
 | 
						|
   String title = "";
 | 
						|
   String units = "";
 | 
						|
   String graph_group = "";
 | 
						|
   int prio = 0;
 | 
						|
   StringBuffer csv = new StringBuffer();
 | 
						|
   Vector<Double> scores = new Vector();
 | 
						|
 | 
						|
   public HistRow(Row r) {
 | 
						|
      name = r.getShortDescKey();
 | 
						|
      title = r.getShortDesc();
 | 
						|
      units = r.getUnits();
 | 
						|
      graph_group = r.getGraphGroup();
 | 
						|
      prio = r.getPriority();
 | 
						|
   }
 | 
						|
 | 
						|
   int getPriority() {
 | 
						|
      return prio;
 | 
						|
   }
 | 
						|
 | 
						|
   String getFname() {
 | 
						|
      return fname;
 | 
						|
   }
 | 
						|
 | 
						|
   void setFname(String s) {
 | 
						|
      fname = s;
 | 
						|
   }
 | 
						|
 | 
						|
   String getName() {
 | 
						|
      return name;
 | 
						|
   }
 | 
						|
 | 
						|
   String getTitle() {
 | 
						|
      return title;
 | 
						|
   }
 | 
						|
 | 
						|
   String getUnits() {
 | 
						|
      return units;
 | 
						|
   }
 | 
						|
 | 
						|
   String getGraphGroup() {
 | 
						|
      return graph_group;
 | 
						|
   }
 | 
						|
}
 | 
						|
 | 
						|
/** This holds all HistRow objects for each test name for each run.
 | 
						|
 * It is used for generating historical graphs and comparisons.
 | 
						|
 */
 | 
						|
class History {
 | 
						|
   String name;
 | 
						|
   Vector<HistRow> csv = new Vector();
 | 
						|
   Hashtable<String, HistRow> csvh = new Hashtable(); // lookup by name
 | 
						|
 | 
						|
   public History(String n) {
 | 
						|
      name = n;
 | 
						|
   }
 | 
						|
 | 
						|
   public String getName() {
 | 
						|
      return name;
 | 
						|
   }
 | 
						|
 | 
						|
   HistRow findRow(String n) {
 | 
						|
      //System.out.println("findCsv, n: " + n);
 | 
						|
      return csvh.get(n);
 | 
						|
   }
 | 
						|
 | 
						|
   void addRow(HistRow r) {
 | 
						|
      csv.add(r);
 | 
						|
      csvh.put(r.getName(), r);
 | 
						|
   }
 | 
						|
}
 | 
						|
 | 
						|
/** A Row represents a single KPI csv data point.  The csv is split into tokens
 | 
						|
 * for easier manipulation by other code.
 | 
						|
 */
 | 
						|
class Row {
 | 
						|
   Vector<String> rdata = new Vector();
 | 
						|
   String short_desc_key = null;
 | 
						|
 | 
						|
   String getNotes() {
 | 
						|
      return rdata.elementAt(kpi.NOTES_IDX);
 | 
						|
   }
 | 
						|
 | 
						|
   String getSubtestPassed() {
 | 
						|
      if (rdata.size() > kpi.SUBTEST_PASS_IDX)
 | 
						|
         return rdata.elementAt(kpi.SUBTEST_PASS_IDX);
 | 
						|
      return "";
 | 
						|
   }
 | 
						|
 | 
						|
   String getSubtestFailed() {
 | 
						|
      if (rdata.size() > kpi.SUBTEST_FAIL_IDX)
 | 
						|
         return rdata.elementAt(kpi.SUBTEST_FAIL_IDX);
 | 
						|
      return "";
 | 
						|
   }
 | 
						|
 | 
						|
   String getUnits() {
 | 
						|
      try {
 | 
						|
         return rdata.elementAt(kpi.UNITS_IDX);
 | 
						|
      }
 | 
						|
      catch (Exception e) {
 | 
						|
         return "";
 | 
						|
      }
 | 
						|
   }
 | 
						|
 | 
						|
   String getGraphGroup() {
 | 
						|
      try {
 | 
						|
         return rdata.elementAt(kpi.GRAPH_GROUP_IDX);
 | 
						|
      }
 | 
						|
      catch (Exception e) {
 | 
						|
         return "";
 | 
						|
      }
 | 
						|
   }
 | 
						|
 | 
						|
   int getPriority() {
 | 
						|
      try {
 | 
						|
         return Long.valueOf(rdata.elementAt(kpi.PRIORITY_IDX)).intValue();
 | 
						|
      }
 | 
						|
      catch (Exception e) {
 | 
						|
         return 0;
 | 
						|
      }
 | 
						|
   }
 | 
						|
 | 
						|
   String getPassFail() {
 | 
						|
      return rdata.elementAt(kpi.PASS_FAIL_IDX);
 | 
						|
   }
 | 
						|
 | 
						|
   String getScore() {
 | 
						|
      return rdata.elementAt(kpi.NUMERIC_SCORE_IDX);
 | 
						|
   }
 | 
						|
 | 
						|
   String getShortDesc() {
 | 
						|
      return rdata.elementAt(kpi.SHORT_DESC_IDX);
 | 
						|
   }
 | 
						|
 | 
						|
   String getTestId() {
 | 
						|
      return rdata.elementAt(kpi.TEST_ID_IDX);
 | 
						|
   }
 | 
						|
 | 
						|
   String getShortDescKey() {
 | 
						|
      return short_desc_key;
 | 
						|
   }
 | 
						|
 | 
						|
   void setShortDescKey(String s) {
 | 
						|
      short_desc_key = s;
 | 
						|
   }
 | 
						|
 | 
						|
   public String toString() {
 | 
						|
      StringBuffer sb = new StringBuffer();
 | 
						|
      sb.append("Row " + getShortDescKey() + "  ");
 | 
						|
      for (int i = 0; i<rdata.size(); i++) {
 | 
						|
         sb.append("[" + i + "] == " + rdata.elementAt(i) + "  ");
 | 
						|
      }
 | 
						|
      return sb.toString();
 | 
						|
   }
 | 
						|
}
 | 
						|
 | 
						|
/** A test contains information on one executation of one test.  For instance,
 | 
						|
 * a wifi-capacity-test would be a single test.  It may create a KPI.csv file with
 | 
						|
 * multiple rows.  It may also have logs for this test run, and KPI graph images.
 | 
						|
 * A Run consists of multiple tests.
 | 
						|
 */
 | 
						|
class Test {
 | 
						|
   String name;
 | 
						|
   Vector<String> titles = null;
 | 
						|
   Vector<Row> data = new Vector();
 | 
						|
   Hashtable<String, String> descs = new Hashtable();
 | 
						|
   Vector<String> kpi_images = new Vector();
 | 
						|
   Vector<String> log_csv = new Vector();
 | 
						|
   int pass = 0;
 | 
						|
   int fail = 0;
 | 
						|
 | 
						|
   int log_bugs = 0;
 | 
						|
   int log_warnings = 0;
 | 
						|
   int log_crashes = 0;
 | 
						|
   int log_restarting = 0;
 | 
						|
 | 
						|
   long date_ms = 0;
 | 
						|
   public String date = "NA";
 | 
						|
   public String test_rig = "NA";
 | 
						|
   public String dut_hw_version = "NA";
 | 
						|
   public String dut_sw_version = "NA";
 | 
						|
   public String dut_model_num = "NA";
 | 
						|
   public String dut_serial_num = "NA";
 | 
						|
 | 
						|
   public Test(String n) {
 | 
						|
      name = n;
 | 
						|
   }
 | 
						|
 | 
						|
   public int getLogBugs() {
 | 
						|
      return log_bugs;
 | 
						|
   }
 | 
						|
 | 
						|
   public int getLogWarnings() {
 | 
						|
      return log_warnings;
 | 
						|
   }
 | 
						|
 | 
						|
   public int getLogCrashes() {
 | 
						|
      return log_crashes;
 | 
						|
   }
 | 
						|
 | 
						|
   public int getLogRestarting() {
 | 
						|
      return log_restarting;
 | 
						|
   }
 | 
						|
 | 
						|
   public String toString() {
 | 
						|
      return "Name: " + name;
 | 
						|
   }
 | 
						|
 | 
						|
   public int getPass() {
 | 
						|
      return pass;
 | 
						|
   }
 | 
						|
 | 
						|
   public int getFail() {
 | 
						|
      return fail;
 | 
						|
   }
 | 
						|
 | 
						|
   void addKpiImage(String s) {
 | 
						|
      kpi_images.add(s);
 | 
						|
   }
 | 
						|
 | 
						|
   long getDateMs() {
 | 
						|
      return date_ms;
 | 
						|
   }
 | 
						|
 | 
						|
   String getTestRig() {
 | 
						|
      return test_rig;
 | 
						|
   }
 | 
						|
 | 
						|
   String getDutHwVer() {
 | 
						|
      return dut_hw_version;
 | 
						|
   }
 | 
						|
 | 
						|
   String getDutSwVer() {
 | 
						|
      return dut_sw_version;
 | 
						|
   }
 | 
						|
 | 
						|
   String getDutSerialNum() {
 | 
						|
      return dut_serial_num;
 | 
						|
   }
 | 
						|
 | 
						|
   String getDutModelNum() {
 | 
						|
      //System.out.println("Test: " + getName() + " model-num: " + dut_model_num);
 | 
						|
      return dut_model_num;
 | 
						|
   }
 | 
						|
 | 
						|
   String getDate() {
 | 
						|
      return date;
 | 
						|
   }
 | 
						|
 | 
						|
   String getName() {
 | 
						|
      return name;
 | 
						|
   }
 | 
						|
 | 
						|
   void addLogCsv(String l) {
 | 
						|
      log_csv.add(l);
 | 
						|
      try {
 | 
						|
         StringTokenizer st = new StringTokenizer(l, "\t");
 | 
						|
         String tok = st.nextToken();
 | 
						|
         if (tok.equals("FILE")) {
 | 
						|
            // title, ignore rest of this title
 | 
						|
            return;
 | 
						|
         }
 | 
						|
         log_bugs += Long.valueOf(st.nextToken());
 | 
						|
         log_warnings += Long.valueOf(st.nextToken());
 | 
						|
         log_crashes += Long.valueOf(st.nextToken());
 | 
						|
         log_restarting += Long.valueOf(st.nextToken());
 | 
						|
      }
 | 
						|
      catch (Exception e) {
 | 
						|
         e.printStackTrace();
 | 
						|
      }
 | 
						|
   }
 | 
						|
 | 
						|
   void addLine(String l) {
 | 
						|
      if (titles == null) {
 | 
						|
         titles = new Vector();
 | 
						|
         StringTokenizer st = new StringTokenizer(l, kpi.in_sep, true);
 | 
						|
         boolean last_was_sep = false;
 | 
						|
         while (st.hasMoreTokens()) {
 | 
						|
            String tok = st.nextToken();
 | 
						|
            if (tok.equals(kpi.in_sep)) {
 | 
						|
               if (last_was_sep) {
 | 
						|
                  titles.add(new String());
 | 
						|
               }
 | 
						|
               last_was_sep = true;
 | 
						|
            }
 | 
						|
            else {
 | 
						|
               titles.add(tok);
 | 
						|
               last_was_sep = false;
 | 
						|
            }
 | 
						|
         }
 | 
						|
      }
 | 
						|
      else {
 | 
						|
         Row row = new Row();
 | 
						|
         data.add(row);
 | 
						|
         StringTokenizer st = new StringTokenizer(l, kpi.in_sep, true);
 | 
						|
         int idx = 0;
 | 
						|
         System.out.println("new line: " + l);
 | 
						|
         boolean last_was_sep = false;
 | 
						|
         while (st.hasMoreTokens()) {
 | 
						|
            String rtok = st.nextToken();
 | 
						|
            if (rtok.equals(kpi.in_sep)) {
 | 
						|
               if (last_was_sep) {
 | 
						|
                  row.rdata.add(new String());
 | 
						|
                  idx++;
 | 
						|
               }
 | 
						|
               last_was_sep = true;
 | 
						|
            }
 | 
						|
            else {
 | 
						|
               row.rdata.add(rtok);
 | 
						|
               idx++;
 | 
						|
               last_was_sep = false;
 | 
						|
            }
 | 
						|
 | 
						|
            if ((data.size() >= 1) && (!last_was_sep) && dut_sw_version.equals("NA")) { // first row is being added
 | 
						|
               if (titles.elementAt(idx - 1).equalsIgnoreCase("test-rig")) {
 | 
						|
                  test_rig = rtok;
 | 
						|
               }
 | 
						|
               else if (titles.elementAt(idx - 1).equalsIgnoreCase("dut-hw-version")) {
 | 
						|
                  dut_hw_version = rtok;
 | 
						|
               }
 | 
						|
               else if (titles.elementAt(idx - 1).equalsIgnoreCase("dut-sw-version")) {
 | 
						|
                  dut_sw_version = rtok;
 | 
						|
               }
 | 
						|
               else if (titles.elementAt(idx - 1).equalsIgnoreCase("dut-model-num")) {
 | 
						|
                  dut_model_num = rtok;
 | 
						|
               }
 | 
						|
               else if (titles.elementAt(idx - 1).equalsIgnoreCase("Date")) {
 | 
						|
                  //System.out.println("idx: " + idx + " rtok: " + rtok);
 | 
						|
                  date_ms = Long.valueOf(rtok).longValue();
 | 
						|
                  date = new Date(date_ms).toString();
 | 
						|
               }
 | 
						|
            }
 | 
						|
            //System.out.println("idx: " + idx);
 | 
						|
         }
 | 
						|
         //System.out.println("done tok reading loop");
 | 
						|
 | 
						|
         String pf = row.getPassFail().toLowerCase();
 | 
						|
         if (pf.indexOf("pass") >= 0) {
 | 
						|
            pass++;
 | 
						|
         }
 | 
						|
         else if (pf.indexOf("fail") >= 0) {
 | 
						|
            fail++;
 | 
						|
         }
 | 
						|
 | 
						|
         String spass = row.getSubtestPassed();
 | 
						|
         try {
 | 
						|
            if (!spass.equals("")) {
 | 
						|
               pass += Long.valueOf(spass).longValue();
 | 
						|
            }
 | 
						|
         }
 | 
						|
         catch (Exception ee) {
 | 
						|
            ee.printStackTrace();
 | 
						|
         }
 | 
						|
 | 
						|
         String sfail = row.getSubtestFailed();
 | 
						|
         try {
 | 
						|
            if (!sfail.equals("")) {
 | 
						|
               fail += Long.valueOf(sfail).longValue();
 | 
						|
            }
 | 
						|
         }
 | 
						|
         catch (Exception ee) {
 | 
						|
            ee.printStackTrace();
 | 
						|
         }
 | 
						|
 | 
						|
         row.setShortDescKey(row.getShortDesc().replace(" ", "_"));
 | 
						|
         //System.out.println("Row: " + row);
 | 
						|
         descs.put(row.getShortDesc(), row.getShortDesc());
 | 
						|
      }
 | 
						|
   }//addline
 | 
						|
}//Test
 | 
						|
 | 
						|
 | 
						|
/** A Run is a collection of tests.  This class encompasses the entire regression test
 | 
						|
 * for a particular flavor of test (basic, fast, etc).
 | 
						|
 * The CI/CD process will create one or more Runs per build artifact per testbed.
 | 
						|
 */
 | 
						|
class Run {
 | 
						|
   String name;
 | 
						|
   Hashtable<String, Test> tests = new Hashtable();
 | 
						|
   Vector<Test> testsv = new Vector();
 | 
						|
 | 
						|
   public Run(String n) {
 | 
						|
      name = n;
 | 
						|
   }
 | 
						|
 | 
						|
   public String toString() {
 | 
						|
      return "Name: " + name;
 | 
						|
   }
 | 
						|
 | 
						|
   int getPass() {
 | 
						|
      int pass = 0;
 | 
						|
      for (Test t: testsv) {
 | 
						|
         pass += t.getPass();
 | 
						|
      }
 | 
						|
      return pass;
 | 
						|
   }
 | 
						|
 | 
						|
   int getFail() {
 | 
						|
      int fail = 0;
 | 
						|
      for (Test t: testsv) {
 | 
						|
         fail += t.getFail();
 | 
						|
      }
 | 
						|
      return fail;
 | 
						|
   }
 | 
						|
 | 
						|
   int getLogBugs() {
 | 
						|
      int fail = 0;
 | 
						|
      for (Test t: testsv) {
 | 
						|
         fail += t.getLogBugs();
 | 
						|
      }
 | 
						|
      return fail;
 | 
						|
   }
 | 
						|
 | 
						|
   int getLogWarnings() {
 | 
						|
      int fail = 0;
 | 
						|
      for (Test t: testsv) {
 | 
						|
         fail += t.getLogWarnings();
 | 
						|
      }
 | 
						|
      return fail;
 | 
						|
   }
 | 
						|
 | 
						|
   int getLogCrashes() {
 | 
						|
      int fail = 0;
 | 
						|
      for (Test t: testsv) {
 | 
						|
         fail += t.getLogCrashes();
 | 
						|
      }
 | 
						|
      return fail;
 | 
						|
   }
 | 
						|
 | 
						|
   int getLogRestarting() {
 | 
						|
      int fail = 0;
 | 
						|
      for (Test t: testsv) {
 | 
						|
         fail += t.getLogRestarting();
 | 
						|
      }
 | 
						|
      return fail;
 | 
						|
   }
 | 
						|
 | 
						|
   Test getFirstTest() {
 | 
						|
      return testsv.elementAt(0);
 | 
						|
   }
 | 
						|
 | 
						|
   String getDate() {
 | 
						|
      Test t = getFirstTest();
 | 
						|
      if (t != null)
 | 
						|
         return t.getDate();
 | 
						|
      return "";
 | 
						|
   }
 | 
						|
 | 
						|
   long getDateMs() {
 | 
						|
      Test t = getFirstTest();
 | 
						|
      if (t != null)
 | 
						|
         return t.getDateMs();
 | 
						|
      return 0;
 | 
						|
   }
 | 
						|
 | 
						|
   String getTestRig() {
 | 
						|
      Test t = getFirstTest();
 | 
						|
      if (t != null)
 | 
						|
         return t.getTestRig();
 | 
						|
      return "";
 | 
						|
   }
 | 
						|
 | 
						|
   String getDutHwVer() {
 | 
						|
      Test t = getFirstTest();
 | 
						|
      if (t != null)
 | 
						|
         return t.getDutHwVer();
 | 
						|
      return "";
 | 
						|
   }
 | 
						|
 | 
						|
   String getDutSwVer() {
 | 
						|
      Test t = getFirstTest();
 | 
						|
      if (t != null)
 | 
						|
         return t.getDutSwVer();
 | 
						|
      return "";
 | 
						|
   }
 | 
						|
 | 
						|
   String getDutSerNum() {
 | 
						|
      Test t = getFirstTest();
 | 
						|
      if (t != null)
 | 
						|
         return t.getDutSerialNum();
 | 
						|
      return "";
 | 
						|
   }
 | 
						|
 | 
						|
   String getDutModelNum() {
 | 
						|
      Test t = getFirstTest();
 | 
						|
      if (t != null)
 | 
						|
         return t.getDutModelNum();
 | 
						|
      return "";
 | 
						|
   }
 | 
						|
 | 
						|
   String getName() {
 | 
						|
      return name;
 | 
						|
   }
 | 
						|
 | 
						|
   void addTest(Test t) {
 | 
						|
      tests.put(t.getName(), t);
 | 
						|
      testsv.add(t);
 | 
						|
   }
 | 
						|
 | 
						|
   Test findTest(String n) {
 | 
						|
      return tests.get(n);
 | 
						|
   }
 | 
						|
}//Run
 | 
						|
 | 
						|
 | 
						|
// From: https://stackoverflow.com/questions/882772/capturing-stdout-when-calling-runtime-exec
 | 
						|
 | 
						|
/**
 | 
						|
 * Execute external process and optionally read output buffer.
 | 
						|
 */
 | 
						|
class ShellExec {
 | 
						|
   private int exitCode;
 | 
						|
   private boolean readOutput, readError;
 | 
						|
   private StreamGobbler errorGobbler, outputGobbler;
 | 
						|
 | 
						|
   public ShellExec() { 
 | 
						|
      this(false, false);
 | 
						|
   }
 | 
						|
 | 
						|
   public ShellExec(boolean readOutput, boolean readError) {
 | 
						|
      this.readOutput = readOutput;
 | 
						|
      this.readError = readError;
 | 
						|
   }
 | 
						|
 | 
						|
   /**
 | 
						|
    * Execute a command.
 | 
						|
    * @param command   command ("c:/some/folder/script.bat" or "some/folder/script.sh")
 | 
						|
    * @param workdir   working directory or NULL to use command folder
 | 
						|
    * @param wait  wait for process to end
 | 
						|
    * @param args  0..n command line arguments
 | 
						|
    * @return  process exit code
 | 
						|
    */
 | 
						|
   public int execute(String command, String workdir, boolean wait, String...args) throws IOException {
 | 
						|
      String[] cmdArr;
 | 
						|
      StringBuffer dbg = new StringBuffer();
 | 
						|
      dbg.append("Command: " + command);
 | 
						|
      if (args != null && args.length > 0) {
 | 
						|
         cmdArr = new String[1+args.length];
 | 
						|
         cmdArr[0] = command;
 | 
						|
         System.arraycopy(args, 0, cmdArr, 1, args.length);
 | 
						|
         for (int i = 0; i<args.length; i++) {
 | 
						|
            dbg.append(" " + args[i]);
 | 
						|
         }
 | 
						|
      } else {
 | 
						|
         cmdArr = new String[] { command };
 | 
						|
      }
 | 
						|
 | 
						|
      System.out.println("Running cmd: " + dbg.toString());
 | 
						|
 | 
						|
      ProcessBuilder pb =  new ProcessBuilder(cmdArr);
 | 
						|
      File workingDir = (workdir==null ? new File(command).getParentFile() : new File(workdir) );
 | 
						|
      pb.directory(workingDir);
 | 
						|
 | 
						|
      Process process = pb.start();
 | 
						|
 | 
						|
      // Consume streams, older jvm's had a memory leak if streams were not read,
 | 
						|
      // some other jvm+OS combinations may block unless streams are consumed.
 | 
						|
      errorGobbler  = new StreamGobbler(process.getErrorStream(), readError);
 | 
						|
      outputGobbler = new StreamGobbler(process.getInputStream(), readOutput);
 | 
						|
      errorGobbler.start();
 | 
						|
      outputGobbler.start();
 | 
						|
 | 
						|
      exitCode = 0;
 | 
						|
      if (wait) {
 | 
						|
         try { 
 | 
						|
            process.waitFor();
 | 
						|
            exitCode = process.exitValue();                 
 | 
						|
         } catch (InterruptedException ex) { }
 | 
						|
      }
 | 
						|
      return exitCode;
 | 
						|
   }   
 | 
						|
 | 
						|
   public int getExitCode() {
 | 
						|
      return exitCode;
 | 
						|
   }
 | 
						|
 | 
						|
   public boolean isOutputCompleted() {
 | 
						|
      return (outputGobbler != null ? outputGobbler.isCompleted() : false);
 | 
						|
   }
 | 
						|
 | 
						|
   public boolean isErrorCompleted() {
 | 
						|
      return (errorGobbler != null ? errorGobbler.isCompleted() : false);
 | 
						|
   }
 | 
						|
 | 
						|
   public String getOutput() {
 | 
						|
      return (outputGobbler != null ? outputGobbler.getOutput() : null);        
 | 
						|
   }
 | 
						|
 | 
						|
   public String getError() {
 | 
						|
      return (errorGobbler != null ? errorGobbler.getOutput() : null);        
 | 
						|
   }
 | 
						|
 | 
						|
   //********************************************
 | 
						|
   //********************************************    
 | 
						|
 | 
						|
    /**
 | 
						|
     * StreamGobbler reads inputstream to "gobble" it.
 | 
						|
     * This is used by Executor class when running 
 | 
						|
     * a commandline applications. Gobblers must read/purge
 | 
						|
     * INSTR and ERRSTR process streams.
 | 
						|
     * http://www.javaworld.com/javaworld/jw-12-2000/jw-1229-traps.html?page=4
 | 
						|
     */
 | 
						|
   private class StreamGobbler extends Thread {
 | 
						|
      private InputStream is;
 | 
						|
      private StringBuilder output;
 | 
						|
      private volatile boolean completed; // mark volatile to guarantee a thread safety
 | 
						|
 | 
						|
      public StreamGobbler(InputStream is, boolean readStream) {
 | 
						|
         this.is = is;
 | 
						|
         this.output = (readStream ? new StringBuilder(256) : null);
 | 
						|
      }
 | 
						|
 | 
						|
      public void run() {
 | 
						|
         completed = false;
 | 
						|
         try {
 | 
						|
            String NL = System.getProperty("line.separator", "\r\n");
 | 
						|
 | 
						|
            InputStreamReader isr = new InputStreamReader(is);
 | 
						|
            BufferedReader br = new BufferedReader(isr);
 | 
						|
            String line;
 | 
						|
            while ( (line = br.readLine()) != null) {
 | 
						|
               if (output != null)
 | 
						|
                  output.append(line + NL); 
 | 
						|
            }
 | 
						|
         } catch (IOException ex) {
 | 
						|
            // ex.printStackTrace();
 | 
						|
         }
 | 
						|
         completed = true;
 | 
						|
      }
 | 
						|
 | 
						|
      /**
 | 
						|
       * Get inputstream buffer or null if stream
 | 
						|
       * was not consumed.
 | 
						|
       * @return
 | 
						|
       */
 | 
						|
      public String getOutput() {
 | 
						|
         return (output != null ? output.toString() : null);
 | 
						|
      }
 | 
						|
 | 
						|
      /**
 | 
						|
       * Is input stream completed.
 | 
						|
       * @return
 | 
						|
       */
 | 
						|
      public boolean isCompleted() {
 | 
						|
         return completed;
 | 
						|
      }
 | 
						|
 | 
						|
   }
 | 
						|
 | 
						|
}
 |