[Zope-Checkins] CVS: Zope/lib/python/zdaemon - zdctl.py:1.37

Chris McDonough chrism@zope.com
Sun, 26 Jan 2003 00:44:22 -0500


Update of /cvs-repository/Zope/lib/python/zdaemon
In directory cvs.zope.org:/tmp/cvs-serv2444

Modified Files:
	zdctl.py 
Log Message:
Use tail reimplementation instead of os.system(tail) to deal with systems that do not have tail.


=== Zope/lib/python/zdaemon/zdctl.py 1.36 => 1.37 ===
--- Zope/lib/python/zdaemon/zdctl.py:1.36	Sat Jan 25 22:36:38 2003
+++ Zope/lib/python/zdaemon/zdctl.py	Sun Jan 26 00:44:19 2003
@@ -54,6 +54,7 @@
 import time
 import signal
 import socket
+import stat
 
 if __name__ == "__main__":
     # Add the parent of the script directory to the module search path
@@ -399,9 +400,12 @@
                 print "No default log file specified; use logtail <logfile>"
                 return
         try:
-            os.system("tail -f %s" % arg)
+            helper = TailHelper(arg)
+            helper.tailf()
         except KeyboardInterrupt:
             print
+        except IOError, msg:
+            print msg
 
     def help_logtail(self):
         print "logtail [logfile] -- Run tail -f on the given logfile."
@@ -490,6 +494,62 @@
         print "        stop the daemon manager."
 
 
+class TailHelper:
+
+    MAX_BUFFSIZE = 1024
+
+    def __init__(self, fname):
+        self.fname = fname
+
+    def tailf(self):
+        f = open(self.fname, 'r')
+        sz, lines = self.tail(10)
+        for line in lines:
+            sys.stdout.write(line)
+            sys.stdout.flush()
+        while 1:
+            newsz = self.fsize()
+            bytes_added = newsz - sz
+            if bytes_added > 0:
+                f.seek(-bytes_added, 2)
+                bytes = f.read(bytes_added)
+                sys.stdout.write(bytes)
+                sys.stdout.flush()
+                sz = newsz
+            time.sleep(1)
+
+    def tail(self, max=10):
+        f = open(self.fname, 'r')
+        f.seek(0, 2)
+        pos = sz = f.tell()
+
+        lines = []
+        bytes = []
+        num_bytes = 0
+
+        while 1:
+            if pos == 0:
+                break
+            f.seek(pos)
+            byte = f.read(1)
+            if byte == '\n':
+                if len(lines) == max:
+                    break
+                bytes.reverse()
+                line = ''.join(bytes)
+                line and lines.append(line)
+                bytes = []
+            bytes.append(byte)
+            num_bytes = num_bytes + 1
+            if num_bytes > self.MAX_BUFFSIZE:
+                break
+            pos = pos - 1
+        lines.reverse()
+        return sz, lines
+
+    def fsize(self):
+        return os.stat(self.fname)[stat.ST_SIZE]
+    
 def main(args=None):
     options = ZDCtlOptions()
     options.realize(args)