429 lines
13 KiB
Python
429 lines
13 KiB
Python
|
|
#coding: utf-8
|
|||
|
|
#-------------------------------------------------------------------
|
|||
|
|
# YakPanel
|
|||
|
|
#-------------------------------------------------------------------
|
|||
|
|
# Copyright (c) 2015-2099 YakPanel(www.yakpanel.com) All rights reserved.
|
|||
|
|
#-------------------------------------------------------------------
|
|||
|
|
# Author: hwliang <hwl@yakpanel.com>
|
|||
|
|
#-------------------------------------------------------------------
|
|||
|
|
|
|||
|
|
# sqlite模型
|
|||
|
|
#------------------------------
|
|||
|
|
import os,re,json,shutil,time
|
|||
|
|
from databaseModel.base import databaseBase
|
|||
|
|
import public
|
|||
|
|
try:
|
|||
|
|
import redis
|
|||
|
|
except:
|
|||
|
|
public.ExecShell("btpip install redis")
|
|||
|
|
import redis
|
|||
|
|
try:
|
|||
|
|
from YakPanel import session
|
|||
|
|
except :pass
|
|||
|
|
|
|||
|
|
|
|||
|
|
class panelRedisDB():
|
|||
|
|
|
|||
|
|
__DB_PASS = None
|
|||
|
|
__DB_USER = None
|
|||
|
|
__DB_PORT = 6379
|
|||
|
|
__DB_HOST = '127.0.0.1'
|
|||
|
|
__DB_CONN = None
|
|||
|
|
__DB_ERR = None
|
|||
|
|
|
|||
|
|
__DB_CLOUD = None
|
|||
|
|
def __init__(self):
|
|||
|
|
self.__config = self.get_options(None)
|
|||
|
|
|
|||
|
|
def redis_conn(self,db_idx = 0):
|
|||
|
|
|
|||
|
|
if self.__DB_HOST in ['127.0.0.1','localhost']:
|
|||
|
|
if not os.path.exists('/www/server/redis'): return False
|
|||
|
|
|
|||
|
|
if not self.__DB_CLOUD:
|
|||
|
|
self.__DB_PASS = self.__config['requirepass']
|
|||
|
|
self.__DB_PORT = int(self.__config['port'])
|
|||
|
|
|
|||
|
|
try:
|
|||
|
|
redis_pool = redis.ConnectionPool(host=self.__DB_HOST, port= self.__DB_PORT, password= self.__DB_PASS, db= db_idx)
|
|||
|
|
self.__DB_CONN = redis.Redis(connection_pool= redis_pool)
|
|||
|
|
return self.__DB_CONN
|
|||
|
|
except :
|
|||
|
|
self.__DB_ERR = public.get_error_info()
|
|||
|
|
return False
|
|||
|
|
|
|||
|
|
|
|||
|
|
def set_host(self,host,port,name,username,password,prefix = ''):
|
|||
|
|
self.__DB_HOST = host
|
|||
|
|
self.__DB_PORT = int(port)
|
|||
|
|
self.__DB_NAME = name
|
|||
|
|
if self.__DB_NAME: self.__DB_NAME = str(self.__DB_NAME)
|
|||
|
|
self.__DB_USER = str(username)
|
|||
|
|
self._USER = str(username)
|
|||
|
|
self.__DB_PASS = str(password)
|
|||
|
|
self.__DB_PREFIX = prefix
|
|||
|
|
self.__DB_CLOUD = 1
|
|||
|
|
return self
|
|||
|
|
|
|||
|
|
|
|||
|
|
#获取配置项
|
|||
|
|
def get_options(self,get = None):
|
|||
|
|
|
|||
|
|
result = {}
|
|||
|
|
redis_conf = public.readFile("{}/redis/redis.conf".format(public.get_setup_path()))
|
|||
|
|
if not redis_conf: return False
|
|||
|
|
|
|||
|
|
keys = ["bind","port","timeout","maxclients","databases","requirepass","maxmemory"]
|
|||
|
|
for k in keys:
|
|||
|
|
v = ""
|
|||
|
|
rep = "\n%s\\s+(.+)" % k
|
|||
|
|
group = re.search(rep,redis_conf)
|
|||
|
|
if not group:
|
|||
|
|
if k == "maxmemory":
|
|||
|
|
v = "0"
|
|||
|
|
if k == "maxclients":
|
|||
|
|
v = "10000"
|
|||
|
|
if k == "requirepass":
|
|||
|
|
v = ""
|
|||
|
|
else:
|
|||
|
|
if k == "maxmemory":
|
|||
|
|
v = int(group.group(1)) / 1024 / 1024
|
|||
|
|
else:
|
|||
|
|
v = group.group(1)
|
|||
|
|
result[k] = v
|
|||
|
|
return result
|
|||
|
|
|
|||
|
|
|
|||
|
|
|
|||
|
|
class main(databaseBase):
|
|||
|
|
|
|||
|
|
_db_max = 16 #最大redis数据库
|
|||
|
|
def __init__(self):
|
|||
|
|
pass
|
|||
|
|
|
|||
|
|
|
|||
|
|
|
|||
|
|
def GetCloudServer(self,args):
|
|||
|
|
'''
|
|||
|
|
@name 获取远程服务器列表
|
|||
|
|
@author hwliang<2021-01-10>
|
|||
|
|
@return list
|
|||
|
|
'''
|
|||
|
|
return self.GetBaseCloudServer(args)
|
|||
|
|
# return public.return_message(0, 0, self.GetBaseCloudServer(args))
|
|||
|
|
|
|||
|
|
|
|||
|
|
def AddCloudServer(self,args):
|
|||
|
|
'''
|
|||
|
|
@添加远程数据库
|
|||
|
|
'''
|
|||
|
|
return self.AddBaseCloudServer(args)
|
|||
|
|
|
|||
|
|
def RemoveCloudServer(self,args):
|
|||
|
|
'''
|
|||
|
|
@删除远程数据库
|
|||
|
|
'''
|
|||
|
|
return self.RemoveBaseCloudServer(args)
|
|||
|
|
|
|||
|
|
def ModifyCloudServer(self,args):
|
|||
|
|
'''
|
|||
|
|
@修改远程数据库
|
|||
|
|
'''
|
|||
|
|
return self.ModifyBaseCloudServer(args)
|
|||
|
|
|
|||
|
|
def get_obj_by_sid(self,sid = 0,conn_config = None):
|
|||
|
|
"""
|
|||
|
|
@取mssql数据库对像 By sid
|
|||
|
|
@sid 数据库分类,0:本地
|
|||
|
|
"""
|
|||
|
|
if type(sid) == str:
|
|||
|
|
try:
|
|||
|
|
sid = int(sid)
|
|||
|
|
except :sid = 0
|
|||
|
|
|
|||
|
|
if sid:
|
|||
|
|
if not conn_config: conn_config = public.M('database_servers').where("id=?" ,sid).find()
|
|||
|
|
db_obj = panelRedisDB()
|
|||
|
|
|
|||
|
|
try:
|
|||
|
|
db_obj = db_obj.set_host(conn_config['db_host'],conn_config['db_port'],None,conn_config['db_user'],conn_config['db_password'])
|
|||
|
|
except Exception as e:
|
|||
|
|
raise public.PanelError(e)
|
|||
|
|
else:
|
|||
|
|
db_obj = panelRedisDB()
|
|||
|
|
return db_obj
|
|||
|
|
|
|||
|
|
|
|||
|
|
|
|||
|
|
def get_list(self,args):
|
|||
|
|
"""
|
|||
|
|
@获取数据库列表
|
|||
|
|
@sql_type = redis
|
|||
|
|
"""
|
|||
|
|
result = []
|
|||
|
|
self.sid = args.get('sid/d',0)
|
|||
|
|
for x in range(0,self._db_max):
|
|||
|
|
|
|||
|
|
data = {}
|
|||
|
|
data['id'] = x
|
|||
|
|
data['name'] = 'DB{}'.format(x)
|
|||
|
|
|
|||
|
|
|
|||
|
|
try:
|
|||
|
|
redis_obj = self.get_obj_by_sid(self.sid).redis_conn(x)
|
|||
|
|
|
|||
|
|
data['keynum'] = redis_obj.dbsize()
|
|||
|
|
if data['keynum'] > 0:
|
|||
|
|
result.append(data)
|
|||
|
|
except :pass
|
|||
|
|
|
|||
|
|
#result = sorted(result,key= lambda x:x['keynum'],reverse=True)
|
|||
|
|
return result
|
|||
|
|
|
|||
|
|
|
|||
|
|
def set_redis_val(self,args):
|
|||
|
|
"""
|
|||
|
|
@设置或修改指定值
|
|||
|
|
"""
|
|||
|
|
|
|||
|
|
self.sid = args.get('sid/d',0)
|
|||
|
|
if not 'name' in args or not 'val' in args:
|
|||
|
|
return public.returnMsg(False, public.lang("Parameter passing error."));
|
|||
|
|
|
|||
|
|
endtime = 0
|
|||
|
|
if 'endtime' in args : endtime = int(args.endtime)
|
|||
|
|
|
|||
|
|
redis_obj = self.get_obj_by_sid(self.sid).redis_conn(args.db_idx)
|
|||
|
|
if endtime:
|
|||
|
|
redis_obj.set(args.name, args.val, endtime)
|
|||
|
|
else:
|
|||
|
|
redis_obj.set(args.name, args.val)
|
|||
|
|
public.set_module_logs('linux_redis','set_redis_val',1)
|
|||
|
|
return public.returnMsg(True, public.lang("Operation is successful."));
|
|||
|
|
|
|||
|
|
def del_redis_val(self,args):
|
|||
|
|
"""
|
|||
|
|
@删除key值
|
|||
|
|
"""
|
|||
|
|
self.sid = args.get('sid/d',0)
|
|||
|
|
if not 'key' in args:
|
|||
|
|
return public.returnMsg(False, public.lang("Parameter passing error."));
|
|||
|
|
|
|||
|
|
redis_obj = self.get_obj_by_sid(self.sid).redis_conn(args.db_idx)
|
|||
|
|
redis_obj.delete(args.key)
|
|||
|
|
|
|||
|
|
return public.returnMsg(True, public.lang("Operation is successful."));
|
|||
|
|
|
|||
|
|
|
|||
|
|
def clear_flushdb(self,args):
|
|||
|
|
"""
|
|||
|
|
清空数据库
|
|||
|
|
@ids 清空数据库列表,不传则清空所有
|
|||
|
|
"""
|
|||
|
|
self.sid = args.get('sid/d',0)
|
|||
|
|
ids = json.loads(args.ids)
|
|||
|
|
#ids = []
|
|||
|
|
if len(ids) == 0:
|
|||
|
|
for x in range(0,self._db_max):
|
|||
|
|
ids.append(x)
|
|||
|
|
|
|||
|
|
for x in ids:
|
|||
|
|
redis_obj = self.get_obj_by_sid(self.sid).redis_conn(x)
|
|||
|
|
redis_obj.flushdb()
|
|||
|
|
|
|||
|
|
return public.returnMsg(True, public.lang("Operation is successful."));
|
|||
|
|
|
|||
|
|
def get_db_keylist(self,args):
|
|||
|
|
"""
|
|||
|
|
@获取指定数据库key集合
|
|||
|
|
"""
|
|||
|
|
|
|||
|
|
search = '*'
|
|||
|
|
if 'search' in args: search = "*" + args.search+"*"
|
|||
|
|
db_idx = args.db_idx
|
|||
|
|
self.sid = args.get('sid/d',0)
|
|||
|
|
|
|||
|
|
redis_obj = self.get_obj_by_sid(self.sid).redis_conn(db_idx)
|
|||
|
|
try:
|
|||
|
|
keylist = sorted(redis_obj.keys(search))
|
|||
|
|
except :
|
|||
|
|
keylist = []
|
|||
|
|
|
|||
|
|
|
|||
|
|
info = {'p':1,'row':10,'count':len(keylist)}
|
|||
|
|
|
|||
|
|
if hasattr(args,'limit'): info['row'] = int(args.limit)
|
|||
|
|
if hasattr(args,'p'): info['p'] = int(args['p'])
|
|||
|
|
|
|||
|
|
import page
|
|||
|
|
#实例化分页类
|
|||
|
|
page = page.Page();
|
|||
|
|
|
|||
|
|
info['uri'] = args
|
|||
|
|
info['return_js'] = ''
|
|||
|
|
if hasattr(args,'tojs'): info['return_js'] = args.tojs
|
|||
|
|
|
|||
|
|
slist = keylist[(info['p']-1) * info['row']:info['p'] * info['row']]
|
|||
|
|
|
|||
|
|
rdata = {}
|
|||
|
|
rdata['page'] = page.GetPage(info,'1,2,3,4,5,8')
|
|||
|
|
rdata['where'] = ''
|
|||
|
|
rdata['data'] = []
|
|||
|
|
|
|||
|
|
idx = 0
|
|||
|
|
for key in slist:
|
|||
|
|
item = {}
|
|||
|
|
try:
|
|||
|
|
item['name'] = key.decode()
|
|||
|
|
except:
|
|||
|
|
item['name'] = str(key)
|
|||
|
|
|
|||
|
|
item['endtime'] = redis_obj.ttl(key)
|
|||
|
|
if item['endtime'] == -1: item['endtime'] = 0
|
|||
|
|
|
|||
|
|
item['type'] = redis_obj.type(key).decode()
|
|||
|
|
|
|||
|
|
if item['type'] == 'string':
|
|||
|
|
try:
|
|||
|
|
item['val'] = redis_obj.get(key).decode()
|
|||
|
|
except:
|
|||
|
|
item['val'] = str(redis_obj.get(key))
|
|||
|
|
elif item['type'] == 'hash':
|
|||
|
|
item['val'] = str(redis_obj.hgetall(key))
|
|||
|
|
elif item['type'] == 'list':
|
|||
|
|
item['val'] = str(redis_obj.lrange(key, 0, -1))
|
|||
|
|
elif item['type'] == 'set':
|
|||
|
|
item['val'] = str(redis_obj.smembers(key))
|
|||
|
|
elif item['type'] == 'zset':
|
|||
|
|
item['val'] = str(redis_obj.zrange(key, 0, 1, withscores=True))
|
|||
|
|
else:
|
|||
|
|
item['val'] = ''
|
|||
|
|
try:
|
|||
|
|
item['len'] = redis_obj.strlen(key)
|
|||
|
|
except:
|
|||
|
|
item['len'] = len(item['val'])
|
|||
|
|
item['val'] = public.xsssec(item['val'])
|
|||
|
|
item['name'] = public.xsssec(item['name'])
|
|||
|
|
rdata['data'].append(item)
|
|||
|
|
idx += 1
|
|||
|
|
return rdata
|
|||
|
|
|
|||
|
|
|
|||
|
|
def ToBackup(self,args):
|
|||
|
|
"""
|
|||
|
|
@备份数据库
|
|||
|
|
"""
|
|||
|
|
|
|||
|
|
self.sid = args.get('sid/d',0)
|
|||
|
|
|
|||
|
|
redis_obj = self.get_obj_by_sid(self.sid).redis_conn(0)
|
|||
|
|
redis_obj.save()
|
|||
|
|
|
|||
|
|
src_path = '{}/dump.rdb'.format(redis_obj.config_get()['dir'])
|
|||
|
|
if not os.path.exists(src_path):
|
|||
|
|
return public.returnMsg(False, public.lang("Backup error"));
|
|||
|
|
|
|||
|
|
backup_path = session['config']['backup_path'] + '/database/redis/'
|
|||
|
|
if not os.path.exists(backup_path): os.makedirs(backup_path)
|
|||
|
|
|
|||
|
|
fileName = backup_path + str(self.sid) + '_db_' + time.strftime('%Y%m%d_%H%M%S',time.localtime()) +'.rdb'
|
|||
|
|
|
|||
|
|
shutil.copyfile(src_path,fileName)
|
|||
|
|
if not os.path.exists(fileName):
|
|||
|
|
return public.returnMsg(False, public.lang("Backup error"));
|
|||
|
|
|
|||
|
|
return public.returnMsg(True, public.lang("Backup Succeeded!"))
|
|||
|
|
|
|||
|
|
def DelBackup(self,args):
|
|||
|
|
"""
|
|||
|
|
@删除备份文件
|
|||
|
|
"""
|
|||
|
|
file = args.file
|
|||
|
|
if os.path.exists(file): os.remove(file)
|
|||
|
|
|
|||
|
|
return public.returnMsg(True, public.lang("Delete successfully!"));
|
|||
|
|
|
|||
|
|
def InputSql(self,get):
|
|||
|
|
"""
|
|||
|
|
@导入数据库
|
|||
|
|
"""
|
|||
|
|
file = get.file
|
|||
|
|
self.sid = get.get('sid/d',0)
|
|||
|
|
|
|||
|
|
redis_obj = self.get_obj_by_sid(self.sid).redis_conn(0)
|
|||
|
|
|
|||
|
|
rpath = redis_obj.config_get()['dir']
|
|||
|
|
dst_path = '{}/dump.rdb'.format(rpath)
|
|||
|
|
public.ExecShell("/etc/init.d/redis stop")
|
|||
|
|
if os.path.exists(dst_path): os.remove(dst_path)
|
|||
|
|
shutil.copy2(file, dst_path)
|
|||
|
|
public.ExecShell("chown redis.redis {dump} && chmod 644 {dump}".format(dump=dst_path))
|
|||
|
|
# self.restart_services()
|
|||
|
|
public.ExecShell("/etc/init.d/redis start")
|
|||
|
|
if os.path.exists(dst_path):
|
|||
|
|
return public.returnMsg(True, public.lang("Restore Successful."))
|
|||
|
|
return public.returnMsg(False, public.lang("Restore failure."))
|
|||
|
|
|
|||
|
|
|
|||
|
|
def get_backup_list(self,get):
|
|||
|
|
"""
|
|||
|
|
@获取备份文件列表
|
|||
|
|
"""
|
|||
|
|
search = ''
|
|||
|
|
if hasattr(get,'search'): search = get['search'].strip().lower();
|
|||
|
|
|
|||
|
|
|
|||
|
|
nlist = []
|
|||
|
|
cloud_list = {}
|
|||
|
|
for x in self.GetCloudServer({'type':'redis'}):
|
|||
|
|
cloud_list['id-' + str(x['id'])] = x
|
|||
|
|
|
|||
|
|
path = session['config']['backup_path'] + '/database/redis/'
|
|||
|
|
if not os.path.exists(path): os.makedirs(path)
|
|||
|
|
for name in os.listdir(path):
|
|||
|
|
if search:
|
|||
|
|
if name.lower().find(search) == -1: continue;
|
|||
|
|
|
|||
|
|
arrs = name.split('_')
|
|||
|
|
|
|||
|
|
filepath = '{}/{}'.format(path,name).replace('//','/')
|
|||
|
|
stat = os.stat(filepath)
|
|||
|
|
|
|||
|
|
item = {}
|
|||
|
|
item['name'] = name
|
|||
|
|
item['filepath'] = filepath
|
|||
|
|
item['size'] = stat.st_size
|
|||
|
|
item['mtime'] = int(stat.st_mtime)
|
|||
|
|
item['sid'] = arrs[0]
|
|||
|
|
item['conn_config'] = cloud_list['id-' + str(arrs[0])]
|
|||
|
|
|
|||
|
|
nlist.append(item)
|
|||
|
|
if hasattr(get, 'sort'):
|
|||
|
|
nlist = sorted(nlist, key=lambda data: data['mtime'], reverse=get["sort"] == "desc")
|
|||
|
|
return nlist
|
|||
|
|
|
|||
|
|
|
|||
|
|
|
|||
|
|
def restart_services(self):
|
|||
|
|
"""
|
|||
|
|
@重启服务
|
|||
|
|
"""
|
|||
|
|
public.ExecShell('net stop redis')
|
|||
|
|
public.ExecShell('net start redis')
|
|||
|
|
return True
|
|||
|
|
|
|||
|
|
|
|||
|
|
def check_cloud_database_status(self,conn_config):
|
|||
|
|
"""
|
|||
|
|
@检测远程数据库是否连接
|
|||
|
|
@conn_config 远程数据库配置,包含host port pwd等信息
|
|||
|
|
"""
|
|||
|
|
try:
|
|||
|
|
|
|||
|
|
sql_obj = panelRedisDB().set_host(conn_config['db_host'],conn_config['db_port'],conn_config['db_name'],conn_config['db_user'],conn_config['db_password'])
|
|||
|
|
keynum = sql_obj.redis_conn(0).dbsize()
|
|||
|
|
return True
|
|||
|
|
except Exception as ex:
|
|||
|
|
|
|||
|
|
return public.returnMsg(False,ex)
|