2017-09-26 05:07:18 -04:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
import hashlib
|
|
|
|
import logging
|
|
|
|
import os
|
|
|
|
import pickle
|
|
|
|
from collections import MutableMapping, namedtuple
|
|
|
|
|
|
|
|
logger = logging.getLogger('tsstats')
|
|
|
|
CachedLog = namedtuple('CachedLog', ['path', 'hash', 'events'])
|
|
|
|
|
|
|
|
|
|
|
|
def _calculate_hash(path):
|
|
|
|
with open(path, 'rb') as f:
|
|
|
|
return hashlib.sha256(f.read()).hexdigest()
|
|
|
|
|
|
|
|
|
|
|
|
class Cache(MutableMapping):
|
|
|
|
def __init__(self, path, data={}):
|
|
|
|
self.path = path
|
|
|
|
self.store = data
|
2018-01-11 12:04:54 -05:00
|
|
|
self.initial_store_version = self.store.setdefault('version', 0)
|
2017-09-26 05:07:18 -04:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def read(cls, path):
|
|
|
|
data = {}
|
|
|
|
if os.path.exists(path):
|
|
|
|
logger.debug('Reading cache from %s', path)
|
|
|
|
with open(path, 'rb') as f:
|
|
|
|
try:
|
|
|
|
data = pickle.load(f)
|
|
|
|
except EOFError:
|
|
|
|
logger.debug('Couldn\'t read cache')
|
|
|
|
return cls(path, data)
|
|
|
|
|
|
|
|
def write(self, path=None):
|
|
|
|
if not path:
|
|
|
|
path = self.path
|
2018-01-11 12:04:54 -05:00
|
|
|
if self.initial_store_version == self.store['version']:
|
|
|
|
logger.debug('Cached content did not change, skipping write')
|
|
|
|
return
|
2017-09-26 05:07:18 -04:00
|
|
|
logger.debug('Writing cache to %s', path)
|
|
|
|
with open(path, 'wb') as f:
|
|
|
|
pickle.dump(self.store, f)
|
|
|
|
|
|
|
|
def needs_parsing(self, path):
|
|
|
|
if path not in self.store:
|
|
|
|
return True
|
|
|
|
return _calculate_hash(path) != self.store[path].hash
|
|
|
|
|
|
|
|
def __setitem__(self, path, events):
|
|
|
|
self.store[path] = CachedLog(path, _calculate_hash(path), list(events))
|
2018-01-11 12:04:54 -05:00
|
|
|
self.store['version'] += 1
|
2017-09-26 05:07:18 -04:00
|
|
|
|
|
|
|
def __getitem__(self, path):
|
|
|
|
return self.store[path]
|
|
|
|
|
|
|
|
def __delitem__(self, path):
|
|
|
|
del self.store[path]
|
|
|
|
|
|
|
|
def __iter__(self):
|
|
|
|
return iter(self.store.keys())
|
|
|
|
|
|
|
|
def __len__(self):
|
|
|
|
return len(self.store)
|