build: add a new script helper
authorMarcel Hollerbach <mail@marcel-hollerbach.de>
Fri, 8 Mar 2019 13:30:52 +0000 (08:30 -0500)
committerJunsuChoi <jsuya.choi@samsung.com>
Thu, 14 Mar 2019 05:46:29 +0000 (14:46 +0900)
Summary:
This script can be used to run all examples that have been build with
meson, this is usefull in order to check if everything works, and checks
if (due to some patches) errors are happening in the output.

This is not meant as function checking of the whole example, just
automatically checking the first-look-appearance.

For now a few examples are setted up correctly, there might be more that
need setups / custom arguments, they can be added over time.

Reviewers: stefan_schmidt, zmike, cedric, q66, devilhorns, segfaultxavi

Reviewed By: zmike

Subscribers: cedric, #reviewers, #committers

Tags: #efl

Differential Revision: https://phab.enlightenment.org/D7979

examples_checks.py [new file with mode: 0755]

diff --git a/examples_checks.py b/examples_checks.py
new file mode 100755 (executable)
index 0000000..fa9badc
--- /dev/null
@@ -0,0 +1,142 @@
+#!/usr/bin/python3
+import os
+import sys
+import subprocess
+import json
+import time
+import concurrent.futures
+import argparse
+import tempfile
+
+#
+# preparation calls for the examples
+#
+
+def prep_eina_file_02():
+  f = tempfile.NamedTemporaryFile(delete=False)
+  f.write(b"Simulation")
+  return [f.name, "/tmp/copy_file"]
+
+def prep_eina_xattr_01():
+  f = tempfile.NamedTemporaryFile(delete=False)
+  f.write(b"Simulation")
+  return ["list", f.name]
+
+def prep_eina_xattr_02():
+  f1 = tempfile.NamedTemporaryFile(delete=False)
+  f1.write(b"Simulation")
+  f2 = tempfile.NamedTemporaryFile(delete=False)
+  f2.write(b"Simulation2")
+  return [f1.name, f2.name]
+
+def prep_eet_data_simple():
+  f1 = tempfile.NamedTemporaryFile(delete=False)
+  f1.write(b"Simulation")
+  f2 = tempfile.NamedTemporaryFile(delete=False)
+  f2.write(b"Simulation2")
+  return [f1.name, f2.name]
+
+def prep_eet_data_nested():
+  f1 = tempfile.NamedTemporaryFile(delete=False)
+  f1.write(b"Simulation")
+  f2 = tempfile.NamedTemporaryFile(delete=False)
+  f2.write(b"Simulation2")
+  return [f1.name, f2.name]
+
+def prep_eet_data_file_descriptor_01():
+  f1 = tempfile.NamedTemporaryFile(delete=False)
+  f1.write(b"Simulation")
+  f2 = tempfile.NamedTemporaryFile(delete=False)
+  f2.write(b"Simulation2")
+  return [f1.name, f2.name, "acc", "Example-Simulation"]
+
+def prep_eet_data_file_descriptor_02():
+  f1 = tempfile.NamedTemporaryFile(delete=False)
+  f1.write(b"Simulation")
+  f2 = tempfile.NamedTemporaryFile(delete=False)
+  f2.write(b"Simulation2")
+  return [f1.name, f2.name, "union", "5", "Example-Simulation"]
+
+example_preparation = {
+  "eina_file_02" : prep_eina_file_02,
+  "eina_xattr_01" : prep_eina_xattr_01,
+  "eina_xattr_02" : prep_eina_xattr_02,
+  "eet-data-simple" : prep_eet_data_simple,
+  "eet-data-nested" : prep_eet_data_nested,
+  "eet-data-simple" : prep_eet_data_simple,
+  "eet-data-file_descriptor_01" : prep_eet_data_file_descriptor_01,
+  "eet-data-file_descriptor_02" : prep_eet_data_file_descriptor_02,
+}
+
+#
+# Holds up the state of the ran examples
+#
+
+class State:
+  def __init__(self, examples):
+    self.max_n = examples
+    self.n = 1
+    self.count_fail = 0
+    self.count_success = 0
+    self.count_err_output = 0
+    print("Found "+str(self.max_n)+" Examples")
+
+  def add_run(self, command, error_in_output, exitcode):
+    print("{}/{} {} {} {} ".format(self.n, self.max_n, ("SUCCESS" if exitcode == 0 else "FAIL"), ("CLEAN" if error_in_output == False else "ERR"), command))
+    self.n = self.n + 1
+    if exitcode != 0:
+      self.count_fail += 1
+    if error_in_output == True:
+      self.count_err_output += 1
+    if exitcode == 0 and error_in_output == False:
+      self.count_success += 1
+
+  def print_summary(self):
+    print("Summary")
+    print("  Failed: "+str(self.count_fail)+"/"+str(self.max_n))
+    print("  Errored: "+str(self.count_err_output)+"/"+str(self.max_n))
+    print("  Success: "+str(self.count_success)+"/"+str(self.max_n))
+
+#
+# this simulates the startup of the example, and the closing after 1s
+#
+
+def simulate_example(example):
+  args = []
+  if os.path.basename(example) in example_preparation:
+    args = example_preparation[os.path.basename(example)]()
+  run = subprocess.Popen([G.builddir + "/" + example] + args,
+      stdout = subprocess.PIPE,
+      stderr = subprocess.PIPE,
+  )
+  time.sleep(1)
+  run.terminate()
+  try:
+    outs, errs = run.communicate(timeout=2)
+  except Exception as e:
+    run.kill()
+    return (example, True, -1)
+  else:
+    return (example, True if b'ERR' in outs or b'ERR' in errs else False, run.poll())
+
+
+parser = argparse.ArgumentParser(description='Run the examples of efl')
+parser.add_argument('builddir', metavar='build', help='the path where to find the meson build directory')
+
+G = parser.parse_args()
+#Run meson to fetch all examples
+meson_introspect = subprocess.Popen(["meson", "introspect", G.builddir, "--targets"],
+      stdout = subprocess.PIPE,
+      stderr = subprocess.PIPE,
+)
+meson_introspect.poll()
+build_targets = json.loads(meson_introspect.stdout.read())
+examples = [b["filename"] for b in build_targets if "examples" in b["filename"] and b["type"] == "executable"]
+state = State(len(examples))
+#simulate all examples in parallel with up to 5 runners
+with concurrent.futures.ThreadPoolExecutor(max_workers=5) as executor:
+  futures = [executor.submit(simulate_example, example) for example in examples]
+  for future in concurrent.futures.as_completed(futures):
+    example_run = future.result()
+    state.add_run(example_run[0], example_run[1], example_run[2])
+state.print_summary()