Files

348 lines
10 KiB
Python
Raw Permalink Normal View History

2026-04-07 02:04:22 +05:30
# -*- 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 io
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 SimpleCache(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'
__SHM_PREFIX = 'SHM_:'
__SHM_BASEDIR = '/dev/shm/aap-shm'
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 idx % 3 == 0:
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:
# 优先从shm中查找
_shm_val = self.__get_shm(key)
if _shm_val is not None:
return _shm_val
except: pass
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
try:
# 优先写入shm
if self.__set_shm(key, value, timeout):
return True
except: pass
# 过期清理
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
try:
# 优先写入shm
if self.__add_shm(key, value, timeout):
return True
except: pass
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):
try:
# 优先删除shm
if self.__del_shm(key):
return True
except: pass
result = self._cache.pop(key, None) is not None
self.del_session_by_file(key)
return result
def has(self, key):
try:
# 优先shm
if self.__has_shm(key):
return True
except: pass
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()
def __set_shm(self, key, value, timeout=None):
'''
@name 尝试将缓存写入shm目录
@author Zhj<2022-10-08>
@param key<string> 键名
@param value<mixed>
@param timeout<int> 存活时间/
@return bool
'''
if key[:5] != self.__SHM_PREFIX:
return False
self.__makesure_shm_basedir()
expires = struct.pack('f', self._normalize_timeout(timeout))
filename = '/'.join((self.__SHM_BASEDIR, self.md5(key)))
with open(filename, 'wb') as fp:
fp.write(expires + pickle.dumps(value, pickle.HIGHEST_PROTOCOL))
os.chmod(filename, 384)
return True
def __get_shm(self, key):
'''
@name 尝试从shm目录下读取缓存
@author Zhj<2022-10-08>
@param key<string> 键名
@return mixed|None
'''
if key[:5] != self.__SHM_PREFIX:
return None
self.__makesure_shm_basedir()
filename = '/'.join((self.__SHM_BASEDIR, self.md5(key)))
if not os.path.exists(filename): return None
with open(filename, 'rb') as fp:
_val = fp.read()
expires = struct.unpack('f', _val[:4])[0]
# 过期 删除缓存文件
if expires > 0 and expires <= time():
os.remove(filename)
return None
return pickle.loads(_val[4:])
def __del_shm(self, key):
'''
@name 删除shm目录下的缓存
@author Zhj<2022-10-08>
@param key<string> 键名
@return bool
'''
if key[:5] != self.__SHM_PREFIX:
return False
self.__makesure_shm_basedir()
filename = '/'.join((self.__SHM_BASEDIR, self.md5(key)))
if os.path.exists(filename):
os.remove(filename)
return True
def __has_shm(self, key):
'''
@name 检查shm目录下的缓存是否存在
@author Zhj<2022-10-08>
@param key<string> 键名
@return bool
'''
if key[:5] != self.__SHM_PREFIX:
return False
self.__makesure_shm_basedir()
filename = '/'.join((self.__SHM_BASEDIR, self.md5(key)))
if not os.path.exists(filename): return False
# 获取缓存过期时间
with open(filename, 'rb') as fp:
expires = struct.unpack('f', fp.read(4))[0]
# 过期 删除缓存文件
if expires > 0 and expires <= time():
os.remove(filename)
return False
return True
def __add_shm(self, key, value, timeout=None):
'''
@name 尝试添加缓存到shm目录下
@author Zhj<2022-10-08>
@param key<string> 键名
@param value<mixed>
@param timeout<int> 缓存存活时间/
@return bool
'''
if self.__has_shm(key):
return False
return self.__set_shm(key, value, timeout)
def __makesure_shm_basedir(self):
'''
@name 确保shm下的缓存目录存在
@author Zhj<2022-10-08>
@return void
'''
if not os.path.exists(self.__SHM_BASEDIR):
os.makedirs(self.__SHM_BASEDIR, 384)