2014-04-23 06:24:08 +00:00
|
|
|
import hashlib
|
2014-03-27 21:06:03 +00:00
|
|
|
import os
|
|
|
|
|
|
|
|
from lockfile import FileLock
|
|
|
|
|
|
|
|
|
2014-04-23 06:24:08 +00:00
|
|
|
def _secure_open_write(filename, fmode):
|
|
|
|
# We only want to write to this file, so open it in write only mode
|
|
|
|
flags = os.O_WRONLY
|
|
|
|
|
|
|
|
# os.O_CREAT | os.O_EXCL will fail if the file already exists, so we only
|
|
|
|
# will open *new* files.
|
|
|
|
# We specify this because we want to ensure that the mode we pass is the
|
|
|
|
# mode of the file.
|
|
|
|
flags |= os.O_CREAT | os.O_EXCL
|
|
|
|
|
|
|
|
# Do not follow symlinks to prevent someone from making a symlink that
|
|
|
|
# we follow and insecurely open a cache file.
|
|
|
|
if hasattr(os, "O_NOFOLLOW"):
|
|
|
|
flags |= os.O_NOFOLLOW
|
|
|
|
|
|
|
|
# On Windows we'll mark this file as binary
|
|
|
|
if hasattr(os, "O_BINARY"):
|
|
|
|
flags |= os.O_BINARY
|
|
|
|
|
|
|
|
# Before we open our file, we want to delete any existing file that is
|
|
|
|
# there
|
|
|
|
try:
|
|
|
|
os.remove(filename)
|
|
|
|
except (IOError, OSError):
|
|
|
|
# The file must not exist already, so we can just skip ahead to opening
|
|
|
|
pass
|
|
|
|
|
|
|
|
# Open our file, the use of os.O_CREAT | os.O_EXCL will ensure that if a
|
|
|
|
# race condition happens between the os.remove and this line, that an
|
|
|
|
# error will be raised. Because we utilize a lockfile this should only
|
|
|
|
# happen if someone is attempting to attack us.
|
|
|
|
fd = os.open(filename, flags, fmode)
|
|
|
|
try:
|
|
|
|
return os.fdopen(fd, "wb")
|
|
|
|
except:
|
|
|
|
# An error occurred wrapping our FD in a file object
|
|
|
|
os.close(fd)
|
|
|
|
raise
|
|
|
|
|
|
|
|
|
2014-03-27 21:06:03 +00:00
|
|
|
class FileCache(object):
|
2014-04-23 06:24:08 +00:00
|
|
|
def __init__(self, directory, forever=False, filemode=0o0600,
|
|
|
|
dirmode=0o0700):
|
2014-03-27 21:06:03 +00:00
|
|
|
self.directory = directory
|
|
|
|
self.forever = forever
|
2014-04-23 06:24:08 +00:00
|
|
|
self.filemode = filemode
|
2014-03-27 21:06:03 +00:00
|
|
|
|
|
|
|
if not os.path.isdir(self.directory):
|
2014-04-23 06:24:08 +00:00
|
|
|
os.makedirs(self.directory, dirmode)
|
2014-03-27 21:06:03 +00:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def encode(x):
|
2014-04-23 06:24:08 +00:00
|
|
|
return hashlib.sha224(x.encode()).hexdigest()
|
2014-03-27 21:06:03 +00:00
|
|
|
|
|
|
|
def _fn(self, name):
|
|
|
|
return os.path.join(self.directory, self.encode(name))
|
|
|
|
|
|
|
|
def get(self, key):
|
|
|
|
name = self._fn(key)
|
|
|
|
if not os.path.exists(name):
|
|
|
|
return None
|
|
|
|
|
|
|
|
with open(name, 'rb') as fh:
|
2014-04-23 06:24:08 +00:00
|
|
|
return fh.read()
|
2014-03-27 21:06:03 +00:00
|
|
|
|
|
|
|
def set(self, key, value):
|
|
|
|
name = self._fn(key)
|
|
|
|
with FileLock(name) as lock:
|
2014-04-23 06:24:08 +00:00
|
|
|
with _secure_open_write(lock.path, self.filemode) as fh:
|
|
|
|
fh.write(value)
|
2014-03-27 21:06:03 +00:00
|
|
|
|
|
|
|
def delete(self, key):
|
|
|
|
name = self._fn(key)
|
|
|
|
if not self.forever:
|
2014-04-23 06:24:08 +00:00
|
|
|
os.remove(name)
|