Attachment 'passlog-1.0.py'

Download

   1 """
   2     MoinMoin - PassLog class Version 1.0, 4.01.2013
   3     
   4     This is used for accessing the wiki pass-log.
   5     
   6     To be installed wiki/data/plugin/logfile along with the 
   7     standard __init__.py for plugin modules.
   8 
   9     @copyright: 2013 Ian Riley <ian.riley@internode.on.net>
  10     
  11     Based on: editlog
  12     
  13     @copyright: 2006 MoinMoin:ThomasWaldmann
  14     @license: GNU GPL, see COPYING for details.
  15 """
  16 
  17 import time
  18 from MoinMoin import log
  19 logging = log.getLogger(__name__)
  20 
  21 from MoinMoin.logfile import LogFile
  22 from MoinMoin import wikiutil, config
  23 from MoinMoin.user import User
  24 from MoinMoin.Page import Page
  25 
  26 def makeCode(length=16):
  27     from string import digits, ascii_letters
  28     from random import choice
  29     return ''.join([choice(digits+ascii_letters) for i in range(length)])
  30 
  31 class PassLogLine:
  32     """
  33     Given the following attributes
  34 
  35     pass_time_usecs
  36     pagename
  37     action
  38     userid
  39     username
  40     addr
  41     hostname
  42     passcode
  43     lifespan (days)
  44     """
  45     def __init__(self, usercache):
  46         self._usercache = usercache
  47         
  48     def __repr__(self, user = None):
  49         return 'Dated: % s Page: %s Pass: %s Expires: %s' % (
  50                                                 self.dated(user),
  51                                                 self.pagename,
  52                                                 self.passcode,
  53                                                 self.expiry_date(user))
  54 
  55     def expiry_usecs(self):
  56         return (self.pass_time_usecs + 
  57                 wikiutil.timestamp2version(int(self.lifespan) * 86400))
  58                
  59     def expiry_days(self):
  60         if self.is_current():
  61             d = (self.expiry_usecs() - wikiutil.timestamp2version(time.time()))
  62             days = wikiutil.version2timestamp(d) / 86400
  63             if days <= 1.0:
  64                 result  = '<1 day'
  65             elif days < 1.5:
  66                 result = '1 day'
  67             else:
  68                 result = '%s days' % int(days + 0.5)
  69         else:
  70             result = ''
  71         return result
  72    
  73     def _dated(self, time_usecs, user = None):
  74         t = wikiutil.version2timestamp(time_usecs)
  75         if user:
  76             return user.getFormattedDateTime(t)
  77         else:
  78             return time.strftime("%Y-%m-%d %H:%M:%S GMT", time.gmtime(t))
  79             
  80     def expiry_date(self, user = None):
  81         return self._dated(self.expiry_usecs(), user)
  82 
  83     def dated(self, user = None):
  84         return self._dated(self.pass_time_usecs, user)
  85             
  86     def link_to(self, request):
  87         page = Page(request, self.pagename)
  88         querystr = 'action=GuestPass&pass=%s' % self.passcode
  89         return page.link_to(request, querystr=querystr)
  90         
  91     def is_current(self):
  92         return wikiutil.timestamp2version(time.time()) < self.expiry_usecs()
  93         
  94     def is_passable(self, request):
  95         page = Page(request, self.pagename)
  96         user = User(request, self.userid) # the user who created the pass
  97         # check pagename, even if used as filter 
  98         return (page.exists() and 
  99                 (self.pagename == request.page.page_name) and 
 100                 user.may.read(self.pagename))
 101     
 102     def is_read_allowed(self, request):
 103         return self.is_current() and self.is_passable(request)
 104 
 105 class PassLog(LogFile):
 106     """ Used for accessing wiki pass-log.
 107     """
 108     def __init__(self, request, filename=None, buffer_size=4096, **kw):
 109         if filename is None:
 110             rootpagename = kw.get('rootpagename', None)
 111             if rootpagename:
 112                 filename = Page(request, rootpagename).getPagePath('pass-log', isfile=1)
 113             else:
 114                 filename = request.rootpage.getPagePath('pass-log', isfile=1)
 115         if filename:
 116             LogFile.__init__(self, filename, buffer_size)
 117   	    
 118   	    self._NUM_FIELDS = 9
 119         self._usercache = {}
 120         
 121     def add(self, request, pass_time_usecs, action, pagename, username, 
 122             host=None, passcode = '', lifespan=30):
 123         """ Generate (and add) a line to the pass-log.
 124 
 125         If `host` is None, it's read from request vars.
 126         """
 127         if request.cfg.log_remote_addr or self.force_ip:
 128             if host is None:
 129                 host = request.remote_addr or ''
 130 
 131             if request.cfg.log_reverse_dns_lookups:
 132                 import socket
 133                 try:
 134                     hostname = socket.gethostbyaddr(host)[0]
 135                     hostname = unicode(hostname, config.charset)
 136                 except (socket.error, UnicodeError):
 137                     hostname = host
 138             else:
 139                 hostname = host
 140         else:
 141             host = ''
 142             hostname = ''
 143 
 144         userid = request.user.valid and request.user.id or ''
 145             
 146         if not passcode:
 147             passcode = makeCode()
 148 
 149         line = u"\t".join((str(long(pass_time_usecs)),
 150                            wikiutil.quoteWikinameFS(pagename),
 151                            action,
 152                            userid,
 153                            username,
 154                            host,
 155                            hostname,
 156                            passcode,
 157                            str(lifespan)
 158                            )) + "\n"
 159         self._add(line)
 160         return self.parser(line)
 161 
 162     def parser(self, line):
 163         """ Parse pass-log line into fields """
 164         fields = line.strip().split('\t')
 165         # Pad empty fields
 166         missing = self._NUM_FIELDS - len(fields)
 167         if missing:
 168             fields.extend([''] * missing)
 169         result = PassLogLine(self._usercache)
 170         (result.pass_time_usecs,
 171          result.pagename, result.action,
 172          result.userid, result.username,
 173          result.addr, result.hostname,
 174          result.passcode, result.lifespan ) = fields[:self._NUM_FIELDS]
 175         if not result.hostname:
 176             result.hostname = result.addr
 177         result.pagename = wikiutil.unquoteWikiname(result.pagename.encode('ascii'))
 178         result.pass_time_usecs = long(result.pass_time_usecs or '0')
 179         return result
 180 
 181     def set_filter(self, **kw):
 182         """ optionally filter for specific pagenames, userids, passcodes"""
 183         expr = "1"
 184         for field in ['pagename', 'userid', 'passcode']:
 185             if field in kw and kw[field]: # treat empty as non-existant
 186                 expr = "%s and x.%s == %s" % (expr, field, repr(kw[field]))
 187 
 188         if 'pass_time_usecs' in kw:
 189             expr = "%s and long(x.ed_time_usecs) == %s" % (expr, long(kw['pass_time_usecs']))
 190 
 191         self.filter = eval("lambda x: " + expr)

You are not allowed to attach a file to this page.