# -*- Mode: Python; test-case-name: morituri.test.test_common_common -*- # vi:si:et:sw=4:sts=4:ts=4 # Morituri - for those about to RIP # Copyright (C) 2009 Thomas Vander Stichele # This file is part of morituri. # # morituri is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # morituri is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with morituri. If not, see . import os import tempfile import shutil def msfToFrames(msf): """ Converts a string value in MM:SS:FF to frames @param msf: @type msf: str @rtype int @returns number of frames """ if not ':' in msf: return int(msf) m, s, f = msf.split(':') return 60 * 75 * int(m) + 75 * int(s) + int(f) def framesToMSF(frames): f = frames % 75 frames -= f s = (frames / 75) % 60 frames -= s * 60 m = frames / 75 / 60 return "%02d:%02d:%02d" % (m, s, f) def framesToHMSF(frames): # cdparanoia style f = frames % 75 frames -= f s = (frames / 75) % 60 frames -= s * 75 m = (frames / 75 / 60) % 60 frames -= m * 75 * 60 h = frames / 75 / 60 / 60 return "%02d:%02d:%02d.%02d" % (h, m, s, f) class Persister(object): """ I wrap an optional pickle to persist an object to disk. Instantiate me with a path to automatically unpickle the object. Call persist to store the object to disk; it will get stored if it changed from the on-disk object. @ivar object: the persistent object """ def __init__(self, path=None, default=None): """ If path is not given, the object will not be persisted. This allows code to transparently deal with both persisted and non-persisted objects, since the persist method will just end up doing nothing. """ self._path = path self.object = None self._unpickle(default) def persist(self, object=None): """ Persist the given object, if we have a persistence path and the object changed. If object is not given, re-persist our object, always. If object is given, only persist if it was changed. """ # don't pickle if it's already ok if object and object == self.object: return # don't pickle if there is no path if not self._path: return # default to pickling our object again if object is None: object = self.object # pickle self.object = object (fd, path) = tempfile.mkstemp(suffix='.morituri.pickle') handle = os.fdopen(fd, 'wb') import pickle pickle.dump(object, handle, 2) handle.close() # do an atomic move shutil.move(path, self._path) def _unpickle(self, default=None): self.object = default if not self._path: return None if not os.path.exists(self._path): return None handle = open(self._path) import pickle self.object = pickle.load(handle)