196 lines
6.1 KiB
Python
196 lines
6.1 KiB
Python
# -*- coding: utf-8 -*-
|
|
from time import time
|
|
import os,struct
|
|
try:
|
|
import cPickle as pickle
|
|
except ImportError: # pragma: no cover
|
|
import pickle
|
|
|
|
from cachelib.base import BaseCache
|
|
import builtins
|
|
|
|
safe_builtins = {
|
|
'range',
|
|
'complex',
|
|
'set',
|
|
'frozenset',
|
|
'slice',
|
|
}
|
|
|
|
class RestrictedUnpickler(pickle.Unpickler):
|
|
def find_class(self, module, name):
|
|
if module == "builtins" and name in safe_builtins:
|
|
# print(name)
|
|
return getattr(builtins, name)
|
|
return None
|
|
|
|
def restricted_loads(s):
|
|
# return RestrictedUnpickler(io.BytesIO(s)).load()
|
|
return True
|
|
|
|
|
|
|
|
class SimpleCacheSession(BaseCache):
|
|
|
|
"""Simple memory cache for single process environments. This class exists
|
|
mainly for the development server and is not 100% thread safe. It tries
|
|
to use as many atomic operations as possible and no locks for simplicity
|
|
but it could happen under heavy load that keys are added multiple times.
|
|
|
|
:param threshold: the maximum number of items the cache stores before
|
|
it starts deleting some.
|
|
:param default_timeout: the default timeout that is used if no timeout is
|
|
specified on :meth:`~BaseCache.set`. A timeout of
|
|
0 indicates that the cache never expires.
|
|
"""
|
|
__session_key = 'BT_:'
|
|
__session_basedir = '/www/server/panel/data/session'
|
|
|
|
def __init__(self, threshold=500, default_timeout=300):
|
|
BaseCache.__init__(self, default_timeout)
|
|
self._cache = {}
|
|
self.clear = self._cache.clear
|
|
self._threshold = threshold
|
|
|
|
def _prune(self):
|
|
if len(self._cache) > self._threshold:
|
|
now = time()
|
|
toremove = []
|
|
for idx, (key, (expires, _)) in enumerate(self._cache.items()):
|
|
if (expires != 0 and expires <= now) or not os.path.exists(os.path.join(self.__session_basedir, self.md5(key))):
|
|
toremove.append(key)
|
|
for key in toremove:
|
|
self._cache.pop(key, None)
|
|
self.del_session_by_file(key)
|
|
|
|
|
|
def _normalize_timeout(self, timeout):
|
|
timeout = BaseCache._normalize_timeout(self, timeout)
|
|
if timeout > 0:
|
|
timeout = time() + timeout
|
|
return timeout
|
|
|
|
def get_session_by_file(self,key):
|
|
try:
|
|
if key[:4] == self.__session_key:
|
|
filename = '/'.join((self.__session_basedir,self.md5(key)))
|
|
if not os.path.exists(filename): return None
|
|
|
|
with open(filename, 'rb') as fp:
|
|
_val = fp.read()
|
|
fp.close()
|
|
expires = struct.unpack('f',_val[:4])[0]
|
|
if expires == 0 or expires > time():
|
|
value = _val[4:]
|
|
|
|
self._cache[key] = (expires,value)
|
|
return pickle.loads(value)
|
|
except :pass
|
|
|
|
def set_session_by_file(self,key,_val,expires):
|
|
try:
|
|
if key[:4] == self.__session_key:
|
|
if not os.path.exists(self.__session_basedir): os.makedirs(self.__session_basedir,384)
|
|
expires = struct.pack('f',expires)
|
|
filename = '/'.join((self.__session_basedir,self.md5(key)))
|
|
fp = open(filename, 'wb+')
|
|
fp.write(expires + _val)
|
|
fp.close()
|
|
os.chmod(filename,384)
|
|
except :pass
|
|
|
|
def del_session_by_file(self,key):
|
|
try:
|
|
if key[:4] == self.__session_key:
|
|
filename = '/'.join((self.__session_basedir,self.md5(key)))
|
|
if os.path.exists(filename): os.remove(filename)
|
|
except : pass
|
|
|
|
def get(self, key):
|
|
if not isinstance(key,str): return None
|
|
try:
|
|
expires, value = self._cache[key]
|
|
if expires == 0 or expires > time():
|
|
return pickle.loads(value)
|
|
except (KeyError, pickle.PickleError):
|
|
return self.get_session_by_file(key)
|
|
|
|
def set(self, key, value, timeout=None):
|
|
|
|
# 类型判断
|
|
if not isinstance(key,str): return False
|
|
type_list=(int,float,bool,str,list,dict,tuple,set,bytes)
|
|
value_type=type(value)
|
|
if value_type not in type_list:
|
|
return False
|
|
|
|
# 过期清理
|
|
expires = self._normalize_timeout(timeout)
|
|
self._prune()
|
|
try:
|
|
restricted_loads(pickle.dumps(value))
|
|
except:
|
|
return False
|
|
|
|
# 转换
|
|
_val = pickle.dumps(value, pickle.HIGHEST_PROTOCOL)
|
|
self._cache[key] = (expires,_val)
|
|
self.set_session_by_file(key,_val,expires)
|
|
return True
|
|
|
|
def add(self, key, value, timeout=None):
|
|
|
|
# 类型判断
|
|
if not isinstance(key,str): return False
|
|
type_list=(int,float,bool,str,list,dict,tuple,set,bytes)
|
|
value_type=type(value)
|
|
if value_type not in type_list:
|
|
return False
|
|
|
|
expires = self._normalize_timeout(timeout)
|
|
self._prune()
|
|
try:
|
|
restricted_loads(pickle.dumps(value))
|
|
except:
|
|
return False
|
|
item = (expires, pickle.dumps(value,pickle.HIGHEST_PROTOCOL))
|
|
if key in self._cache:
|
|
return False
|
|
self._cache.setdefault(key, item)
|
|
self.set_session_by_file(key,item[1],expires)
|
|
return True
|
|
|
|
def delete(self, key):
|
|
result = self._cache.pop(key, None) is not None
|
|
self.del_session_by_file(key)
|
|
return result
|
|
|
|
def has(self, key):
|
|
try:
|
|
expires, value = self._cache[key]
|
|
return expires == 0 or expires > time()
|
|
except KeyError:
|
|
if self.get_session_by_file(key): return True
|
|
return False
|
|
|
|
|
|
def get_expire_time(self, key):
|
|
try:
|
|
expires, value = self._cache[key]
|
|
return expires
|
|
except KeyError:
|
|
return 0
|
|
|
|
def md5(self,strings):
|
|
"""
|
|
生成MD5
|
|
@strings 要被处理的字符串
|
|
return string(32)
|
|
"""
|
|
import hashlib
|
|
m = hashlib.md5()
|
|
|
|
m.update(strings.encode('utf-8'))
|
|
return m.hexdigest()
|
|
|