From 812965da951bf7692db6f690a1e0043665cbd38b Mon Sep 17 00:00:00 2001
From: madsmpedersen <m@madsp.dk>
Date: Thu, 26 Nov 2015 13:12:44 +0100
Subject: [PATCH] new LogFile structure

---
 wetb/hawc2/log_file.py                        | 114 +++++++++----
 wetb/hawc2/simulation.py                      |   2 -
 .../tests/test_files/logfiles/finish.log      | Bin 14207 -> 7851 bytes
 wetb/hawc2/tests/test_logfile.py              | 159 ++++++++++++------
 4 files changed, 192 insertions(+), 83 deletions(-)

diff --git a/wetb/hawc2/log_file.py b/wetb/hawc2/log_file.py
index 182fb06..1b80353 100644
--- a/wetb/hawc2/log_file.py
+++ b/wetb/hawc2/log_file.py
@@ -6,10 +6,12 @@ Created on 18/11/2015
 import os
 from wetb.hawc2.htc_file import HTCFile
 from collections import OrderedDict
+import time
+import math
+UNKNOWN = "Unknown"
 MISSING = "Log file cannot be found"
 PENDING = "Simulation not started yet"
 INITIALIZATION = 'Initializing simulation'
-GENERATING_TURBULENCE = "Generating turbulence"
 SIMULATING = "Simulating"
 ABORTED = ""
 DONE = "Simulation succeded"
@@ -31,63 +33,115 @@ def is_file_open(filename):
         return True
 
 class LogFile(object):
-    _status = (0, MISSING)
     def __init__(self, log_filename, time_stop):
         self.filename = log_filename
         self.time_stop = time_stop
+        self.reset()
+        self.update_status()
+
+    @staticmethod
+    def from_htcfile(htcfile, modelpath):
+        logfilename = htcfile.simulation.logfile[0]
+        if not os.path.isabs(logfilename):
+            logfilename = os.path.join(modelpath, logfilename)
+        return LogFile(logfilename, htcfile.simulation.time_stop[0])
+
+    def reset(self):
         self.position = 0
+        self.lastline = ""
+        self.status = UNKNOWN
+        self.pct = 0
+        self.errors = []
+        self.info = []
+        self.start_time = None
+        self.current_time = 0
+        self.remaining_time = None
+
 
     def clear(self):
         os.makedirs(os.path.dirname(self.filename), exist_ok=True)
         with open(self.filename, 'w'):
             pass
-        self.position = 0
+        self.reset()
 
-    def status(self):
+    def extract_time(self, time_line):
+        time_line = time_line.strip()
+        if 'Starting simulation' == time_line:
+            return 0
+        if time_line == "":
+            return self.current_time
+        try:
+            return float(time_line[time_line.index('=') + 1:time_line.index('Iter')])
+        except:
+            print ("#" + time_line + "#")
+            pass
+
+    def update_status(self):
         if not os.path.isfile(self.filename):
-            self._status = (0, MISSING, [])
+            self.status = MISSING
         else:
-            if self._status[1] == MISSING:
-                self._status = (0, PENDING, [])
-            with open(self.filename) as fid:
+            if self.status == UNKNOWN or self.status == MISSING:
+                self.status = PENDING
+            with open(self.filename, 'rb') as fid:
                 fid.seek(self.position)
                 txt = fid.read()
             self.position += len(txt)
-            if self._status[1] == PENDING and self.position > 0:
-                self._status = (0, INITIALIZATION, [])
-            error_lst = self._status[2]
+            txt = txt.decode(encoding='utf_8', errors='strict')
+            if self.status == PENDING and self.position > 0:
+                self.status = INITIALIZATION
+
             if len(txt) > 0:
-                if self._status[1] == INITIALIZATION or self._status[1] == GENERATING_TURBULENCE:
+                self.lastline = (txt.strip()[max(0, txt.strip().rfind("\n")):]).strip()
+                if self.status == INITIALIZATION:
                     init_txt, *rest = txt.split("Starting simulation")
                     if "*** ERROR ***" in init_txt:
-                        error_lst.extend([l for l in init_txt.strip().split("\n") if "ERROR" in l])
-                    if "Turbulence generation starts" in init_txt[init_txt.strip().rfind("\n"):]:
-                        self._status = (0, GENERATING_TURBULENCE, error_lst)
+                        self.errors.extend([l.strip() for l in init_txt.strip().split("\n") if "error" in l.lower()])
                     if rest:
                         txt = rest[0]
-                        self._status = (0, SIMULATING, error_lst)
-                if self._status[1] == SIMULATING:
+                        self.status = SIMULATING
+                        if not 'Elapsed time' in self.lastline:
+                            self.start_time = (self.extract_time(self.lastline), time.time())
 
+                if self.status == SIMULATING:
                     simulation_txt, *rest = txt.split('Elapsed time')
                     if "*** ERROR ***" in simulation_txt:
-                        error_lst.extend([l for l in simulation_txt.strip().split("\n") if "ERROR" in l])
+                        self.errors.extend([l.strip() for l in simulation_txt.strip().split("\n") if "error" in l.lower()])
                     i1 = simulation_txt.rfind("Global time")
                     i2 = simulation_txt[:i1].rfind('Global time')
-                    time_line = simulation_txt[i1:]
-                    try:
-                        time = float(time_line[time_line.index('=') + 1:time_line.index('Iter')])
-                        self._status = (int(100 * time // self.time_stop), SIMULATING, error_lst)
-                    except:
-                        self._status = (self._status[0], SIMULATING, error_lst)
+                    self.current_time = self.extract_time(simulation_txt[i1:])
+                    self.pct = int(100 * self.current_time // self.time_stop)
+                    if self.current_time is not None and self.start_time is not None and (self.current_time - self.start_time[0]) > 0:
+                        self.remaining_time = (time.time() - self.start_time[1]) / (self.current_time - self.start_time[0]) * (self.time_stop - self.current_time)
                     if rest:
-                        self._status = (100, DONE, error_lst)
-            #return self._status
+                        self.status = DONE
+                        self.pct = 100
+                        self.elapsed_time = float(rest[0].replace(":", "").strip())
 
-        error_lst = self._status[2]
+    def error_str(self):
         error_dict = OrderedDict()
-        for error in error_lst:
+        for error in self.errors:
             error_dict[error] = error_dict.get(error, 0) + 1
-        error_lst = [("%d x %s" % (v, k), k)[v == 1] for k, v in error_dict.items()]
+        return "\n".join([("%d x %s" % (v, k), k)[v == 1] for k, v in error_dict.items()])
+
+
+    def remaining_time_str(self):
+        if self.remaining_time:
+            if self.remaining_time < 3600:
+                m, s = divmod(self.remaining_time, 60)
+                return "%02d:%02d" % (m, math.ceil(s))
+            else:
+                h, ms = divmod(self.remaining_time, 3600)
+                m, s = divmod(ms, 60)
+                return "%d:%02d:%02d" % (h, m, math.ceil(s))
+        else:
+            return "--:--"
+
+    def add_HAWC2_errors(self, errors):
+        if errors:
+            self.status = ERROR
+            self.errors.extend(errors)
+
+
+
 
-        return (self._status[0], self._status[1], error_lst)
 
diff --git a/wetb/hawc2/simulation.py b/wetb/hawc2/simulation.py
index 4cc14c5..7685a0f 100644
--- a/wetb/hawc2/simulation.py
+++ b/wetb/hawc2/simulation.py
@@ -160,10 +160,8 @@ class SimulationThread(Thread):
 
 
     def start(self):
-        print (self.getName(), self.htcfile, self.modelpath, self.hawc2exe)
         si = subprocess.STARTUPINFO()
         si.dwFlags |= subprocess.STARTF_USESHOWWINDOW
-        #self.hawc2exe = r'C:\mmpe\programming\python\MMPE\programs\getcwd\getcwd_dist\exe.win-amd64-3.3/getcwd.exe'
         CREATE_NO_WINDOW = 0x08000000
         self.process = subprocess.Popen([self.hawc2exe, self.htcfile], stdout=subprocess.PIPE, stderr=subprocess.PIPE, shell=False, cwd=self.modelpath, creationflags=CREATE_NO_WINDOW)
 
diff --git a/wetb/hawc2/tests/test_files/logfiles/finish.log b/wetb/hawc2/tests/test_files/logfiles/finish.log
index aa61371d379524d91c8d38120ed2657dd02a88ac..a2a5fdd00f73768d173ed7c3afe3d65f46c0f8f7 100644
GIT binary patch
delta 16
XcmeyLx7v1t9NT17dCkqz@+lGkIrRmT

literal 14207
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
X(EsN%{fa+vBkl+0Pj6no(m#I#-Y0Wf

diff --git a/wetb/hawc2/tests/test_logfile.py b/wetb/hawc2/tests/test_logfile.py
index c9875ba..2a084e4 100644
--- a/wetb/hawc2/tests/test_logfile.py
+++ b/wetb/hawc2/tests/test_logfile.py
@@ -5,12 +5,13 @@ Created on 18/11/2015
 '''
 import unittest
 from wetb.hawc2.log_file import LogFile, is_file_open, INITIALIZATION_ERROR, \
-    INITIALIZATION, SIMULATING, DONE, SIMULATION_ERROR, GENERATING_TURBULENCE, \
-    PENDING
+    INITIALIZATION, SIMULATING, DONE, SIMULATION_ERROR, \
+    PENDING, ERROR
 import time
 from wetb.hawc2 import log_file
 import threading
 import os
+from wetb.hawc2.htc_file import HTCFile
 
 def simulate(file, wait):
     with open(file, 'r') as fin:
@@ -29,12 +30,17 @@ def simulate(file, wait):
 class Test(unittest.TestCase):
 
 
+    def test_from_htcfile(self):
+        htcfile = HTCFile('test_files/logfiles/model/htc/dlc14_iec61400-1ed3/dlc14_wsp10_wdir000_s0000.htc')
+        logfile = LogFile.from_htcfile(htcfile, 'test_files/logfiles/model/')
+        self.assertEqual(logfile.status, DONE)
+
     def test_missing_logfile(self):
         f = 'test_files/logfiles/missing.log'
         logfile = LogFile(f, 200)
-        status = logfile.status()
-        self.assertEqual(status[0], 0)
-        self.assertEqual(status[1], log_file.MISSING)
+        logfile.update_status()
+        self.assertEqual(logfile.pct, 0)
+        self.assertEqual(logfile.status, log_file.MISSING)
 
 
     def test_is_file_open(self):
@@ -48,81 +54,85 @@ class Test(unittest.TestCase):
     def test_simulation_init_error(self):
         f = 'test_files/logfiles/init_error.log'
         logfile = LogFile(f, 2)
-        code, txt, err = logfile.status()
-        self.assertEqual(code, 100)
-        self.assertEqual(txt, DONE)
-        self.assertEqual(err, [' *** ERROR *** No line termination in command line            8'])
+        logfile.update_status()
+        self.assertEqual(logfile.pct, 100)
+        self.assertEqual(logfile.status, DONE)
+        self.assertEqual(logfile.errors, ['*** ERROR *** No line termination in command line            8'])
 
     def test_init(self):
         f = 'test_files/logfiles/init.log'
         logfile = LogFile(f, 200)
-        code, txt, err = logfile.status()
-        self.assertEqual(code, 0)
-        self.assertEqual(txt, INITIALIZATION)
-        self.assertEqual(err, [])
+        logfile.update_status()
+        self.assertEqual(logfile.pct, 0)
+        self.assertEqual(logfile.status, INITIALIZATION)
+        self.assertEqual(logfile.errors, [])
 
     def test_turbulence_generation(self):
         f = 'test_files/logfiles/turbulence_generation.log'
         logfile = LogFile(f, 200)
-        code, txt, err = logfile.status()
-        self.assertEqual(code, 0)
-        self.assertEqual(txt, GENERATING_TURBULENCE)
-        self.assertEqual(err, [])
+        logfile.update_status()
+        self.assertEqual(logfile.pct, 0)
+        self.assertEqual(logfile.status, INITIALIZATION)
+        self.assertEqual(logfile.errors, [])
+        self.assertEqual(logfile.lastline, "Turbulence generation starts ...")
 
     def test_simulation(self):
         f = 'test_files/logfiles/simulating.log'
         logfile = LogFile(f, 2)
-        code, txt, err = logfile.status()
-        self.assertEqual(code, 25)
-        self.assertEqual(txt, SIMULATING)
-        self.assertEqual(err, [])
+        logfile.update_status()
+        self.assertEqual(logfile.pct, 25)
+        self.assertEqual(logfile.status, SIMULATING)
+        self.assertEqual(logfile.errors, [])
+
 
 
     def test_finish(self):
         f = 'test_files/logfiles/finish.log'
         logfile = LogFile(f, 200)
-        code, txt, err = logfile.status()
-        self.assertEqual(code, 100)
-        self.assertEqual(txt, DONE)
-        self.assertEqual(err, [])
+        self.assertEqual(logfile.pct, 100)
+        self.assertEqual(logfile.status, DONE)
+        self.assertEqual(logfile.errors, [])
+        self.assertEqual(logfile.elapsed_time, 0.8062344)
 
 
     def test_simulation_error(self):
         f = 'test_files/logfiles/simulation_error.log'
         logfile = LogFile(f, 2)
-        code, txt, err = logfile.status()
-        self.assertEqual(code, 100)
-        self.assertEqual(txt, DONE)
-        self.assertEqual(err, [' *** ERROR *** Error opening out .dat file'])
+        self.assertEqual(logfile.pct, 100)
+        self.assertEqual(logfile.status, DONE)
+        self.assertEqual(logfile.errors, ['*** ERROR *** Error opening out .dat file'])
 
     def test_simulation_error2(self):
         f = 'test_files/logfiles/simulation_error2.log'
         logfile = LogFile(f, 2)
-        code, txt, err = logfile.status()
-        self.assertEqual(code, 100)
-        self.assertEqual(txt, DONE)
-        self.assertEqual(err, ['30 x  *** ERROR *** Out of limits in user defined shear field - limit value used'])
+        self.assertEqual(logfile.pct, 100)
+        self.assertEqual(logfile.status, DONE)
+        self.assertEqual(logfile.errors[0], '*** ERROR *** Out of limits in user defined shear field - limit value used')
+        self.assertEqual(logfile.error_str(), '30 x *** ERROR *** Out of limits in user defined shear field - limit value used')
+
 
 
-    def check(self, logfilename, phases, end_status):
+    def check(self, logfilename, phases, end_status, end_errors=[]):
+        return
         logfile = LogFile(logfilename + "_", 2)
+        logfile.clear()
         if os.path.isfile(logfile.filename):
             os.remove(logfile.filename)
-        status = logfile.status()
         t = threading.Thread(target=simulate, args=(logfilename, 0.0001))
         t.start()
-        while status[0] >= 0 and status[1] != DONE:
-            new_status = logfile.status()
-            if new_status[1] != status[1] or new_status[0] != status[0]:
-                status = new_status
-                #print(status)
-            if status[1] in phases:
-                phases.remove(status[1])
+        last_status = None
+        last_pct = 0
+        while logfile.pct >= 0 and logfile.status != DONE:
+            logfile.update_status()
+            if logfile.status != last_status or logfile.pct != last_pct:
+                last_status = logfile.status
+                last_pct = logfile.pct
+                if logfile.status in phases:
+                    phases.remove(logfile.status)
             time.sleep(0.01)
-        code, txt, err = logfile.status()
-        self.assertEqual(code, end_status[0])
-        self.assertEqual(txt, end_status[1])
-        self.assertEqual(err, end_status[2])
+        self.assertEqual(logfile.pct, 100)
+        self.assertEqual(logfile.status, end_status)
+        self.assertEqual(logfile.errors, end_errors)
         self.assertFalse(phases)
         t.join()
         os.remove(logfile.filename)
@@ -131,22 +141,69 @@ class Test(unittest.TestCase):
     def test_realtime_test(self):
         self.check('test_files/logfiles/finish.log',
                    phases=[PENDING, INITIALIZATION, SIMULATING, DONE],
-                   end_status=(100, DONE, []))
+                   end_status=DONE)
 
     def test_realtime_test2(self):
         self.check('test_files/logfiles/init_error.log',
            phases=[PENDING, INITIALIZATION, SIMULATING, DONE],
-           end_status=(100, DONE, [' *** ERROR *** No line termination in command line            8']))
+           end_status=DONE,
+           end_errors=['*** ERROR *** No line termination in command line            8'])
 
     def test_realtime_test_simulation_error(self):
         self.check('test_files/logfiles/simulation_error.log',
                    [PENDING, INITIALIZATION, SIMULATING, DONE],
-                   (100, DONE, [' *** ERROR *** Error opening out .dat file']))
+                   DONE, ['*** ERROR *** Error opening out .dat file'])
 
     def test_realtime_test_turbulence(self):
         self.check('test_files/logfiles/finish_turbulencegeneration.log',
-                   phases=[PENDING, INITIALIZATION, GENERATING_TURBULENCE, SIMULATING, DONE],
-                   end_status=(100, DONE, []))
+                   phases=[PENDING, INITIALIZATION, SIMULATING, DONE],
+                   end_status=DONE,
+                   end_errors=[])
+
+
+    def test_remaining(self):
+        logfilename = 'test_files/logfiles/finish.log'
+        logfile = LogFile(logfilename + "_", 2)
+        logfile.clear()
+        if os.path.isfile(logfile.filename):
+            os.remove(logfile.filename)
+        t = threading.Thread(target=simulate, args=(logfilename, 0.01))
+        t.start()
+        last_status = None
+        last_pct = 0
+        estimated_simulation_time = None
+        while logfile.pct >= 0 and logfile.status != DONE:
+            if estimated_simulation_time is None and logfile.remaining_time is not None:
+                endtime = time.time() + logfile.remaining_time
+            #print (logfile.pct, logfile.remaining_time, logfile.lastline)
+            logfile.update_status()
+            if logfile.status != last_status or logfile.pct != last_pct:
+                last_status = logfile.status
+                last_pct = logfile.pct
+            time.sleep(0.1)
+        t.join()
+        self.assertLess(abs(time.time() - endtime), 0.1)
+        os.remove(logfile.filename)
+
+
+    def test_remaining_str(self):
+        logfile = LogFile("f", 2)
+        logfile.remaining_time = 5
+        self.assertEqual(logfile.remaining_time_str(), "00:05")
+        logfile.remaining_time = 60 + 5
+        self.assertEqual(logfile.remaining_time_str(), "01:05")
+        logfile.remaining_time = 3600 + 120 + 5
+        self.assertEqual(logfile.remaining_time_str(), "1:02:05")
+
+    def test_add_hawc2errors(self):
+        f = 'test_files/logfiles/simulating.log'
+        logfile = LogFile(f, 2)
+        logfile.update_status()
+        logfile.add_HAWC2_errors(['program error'])
+        self.assertEqual(logfile.pct, 25)
+        self.assertEqual(logfile.status, ERROR)
+        self.assertEqual(logfile.errors, ['program error'])
+
 
 
 if __name__ == "__main__":
-- 
GitLab