mirror of https://github.com/midoks/mdserver-web
parent
7b94a61b65
commit
8a640981d2
Before Width: | Height: | Size: 2.3 KiB After Width: | Height: | Size: 2.3 KiB |
@ -0,0 +1,25 @@ |
||||
<style> |
||||
.overflow_hide { |
||||
overflow: hidden; |
||||
text-overflow: ellipsis; |
||||
white-space: nowrap; |
||||
vertical-align: middle; |
||||
} |
||||
</style> |
||||
|
||||
<div class="bt-form"> |
||||
<div class='plugin_version'></div> |
||||
<div class="bt-w-main"> |
||||
<div class="bt-w-menu"> |
||||
<p class="bgw" onclick="pluginService('data_query');">服务</p> |
||||
</div> |
||||
<div class="bt-w-con pd15"> |
||||
<div class="soft-man-con"></div> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
<script type="text/javascript"> |
||||
resetPluginWinHeight(300); |
||||
resetPluginWinWidth(380); |
||||
pluginService('data_query', $('.plugin_version').attr('version')); |
||||
</script> |
@ -0,0 +1,97 @@ |
||||
# coding:utf-8 |
||||
|
||||
import sys |
||||
import io |
||||
import os |
||||
import time |
||||
import re |
||||
|
||||
sys.path.append(os.getcwd() + "/class/core") |
||||
import mw |
||||
|
||||
app_debug = False |
||||
if mw.isAppleSystem(): |
||||
app_debug = True |
||||
|
||||
|
||||
def getPluginName(): |
||||
return 'data_query' |
||||
|
||||
|
||||
def getPluginDir(): |
||||
return mw.getPluginDir() + '/' + getPluginName() |
||||
|
||||
|
||||
def getServerDir(): |
||||
return mw.getServerDir() + '/' + getPluginName() |
||||
|
||||
|
||||
def getInitDTpl(): |
||||
path = getPluginDir() + "/init.d/" + getPluginName() + ".tpl" |
||||
return path |
||||
|
||||
|
||||
def getArgs(): |
||||
args = sys.argv[3:] |
||||
tmp = {} |
||||
args_len = len(args) |
||||
|
||||
if args_len == 1: |
||||
t = args[0].strip('{').strip('}') |
||||
if t.strip() == '': |
||||
tmp = [] |
||||
else: |
||||
t = t.split(':') |
||||
tmp[t[0]] = t[1] |
||||
tmp[t[0]] = t[1] |
||||
elif args_len > 1: |
||||
for i in range(len(args)): |
||||
t = args[i].split(':') |
||||
tmp[t[0]] = t[1] |
||||
return tmp |
||||
|
||||
def checkArgs(data, ck=[]): |
||||
for i in range(len(ck)): |
||||
if not ck[i] in data: |
||||
return (False, mw.returnJson(False, '参数:(' + ck[i] + ')没有!')) |
||||
return (True, mw.returnJson(True, 'ok')) |
||||
|
||||
|
||||
def status(): |
||||
return 'start' |
||||
|
||||
def appOp(method): |
||||
return 'ok' |
||||
|
||||
|
||||
|
||||
def start(): |
||||
return appOp('start') |
||||
|
||||
|
||||
def stop(): |
||||
return appOp('stop') |
||||
|
||||
|
||||
def restart(): |
||||
status = appOp('restart') |
||||
return status |
||||
|
||||
def reload(): |
||||
return appOp('reload') |
||||
|
||||
|
||||
if __name__ == "__main__": |
||||
func = sys.argv[1] |
||||
if func == 'status': |
||||
print(status()) |
||||
elif func == 'start': |
||||
print(start()) |
||||
elif func == 'stop': |
||||
print(stop()) |
||||
elif func == 'restart': |
||||
print(restart()) |
||||
elif func == 'reload': |
||||
print(reload()) |
||||
else: |
||||
print('error') |
@ -0,0 +1,36 @@ |
||||
{ |
||||
"hook":[ |
||||
{ |
||||
"tag":"menu", |
||||
"menu": { |
||||
"title":"数据管理", |
||||
"name":"data_query", |
||||
"path":"static/html/index.html", |
||||
"css_path":"static/css/data.css", |
||||
"js_path":"static/js/data_query.js" |
||||
} |
||||
}, |
||||
{ |
||||
"tag":"global_static", |
||||
"global_static": { |
||||
"title":"数据管理", |
||||
"name":"data_query", |
||||
"css_path":"static/css/ico.css" |
||||
} |
||||
} |
||||
], |
||||
"sort": 7, |
||||
"ps": "数据简单管理(memcached,redis,mongodb)", |
||||
"name": "data_query", |
||||
"title": "数据管理", |
||||
"shell": "install.sh", |
||||
"versions":"1.0", |
||||
"tip": "soft", |
||||
"checks": "server/data_query", |
||||
"path": "server/data_query", |
||||
"display": 1, |
||||
"author": "midoks", |
||||
"date": "2024-01-22", |
||||
"type": 0, |
||||
"pid": "2" |
||||
} |
@ -0,0 +1,34 @@ |
||||
#!/bin/bash |
||||
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin:/opt/homebrew/bin |
||||
export PATH |
||||
|
||||
curPath=`pwd` |
||||
rootPath=$(dirname "$curPath") |
||||
rootPath=$(dirname "$rootPath") |
||||
serverPath=$(dirname "$rootPath") |
||||
|
||||
install_tmp=${rootPath}/tmp/mw_install.pl |
||||
VERSION=$2 |
||||
|
||||
Install_App() |
||||
{ |
||||
echo '正在安装脚本文件...' > $install_tmp |
||||
mkdir -p $serverPath/source |
||||
mkdir -p $serverPath/data_query |
||||
echo "${VERSION}" > $serverPath/data_query/version.pl |
||||
|
||||
echo '安装完成' |
||||
} |
||||
|
||||
Uninstall_App() |
||||
{ |
||||
rm -rf $serverPath/data_query |
||||
echo "卸载成功" |
||||
} |
||||
|
||||
action=$1 |
||||
if [ "${1}" == 'install' ];then |
||||
Install_App |
||||
else |
||||
Uninstall_App |
||||
fi |
@ -0,0 +1,563 @@ |
||||
# coding:utf-8 |
||||
|
||||
import sys |
||||
import io |
||||
import os |
||||
import time |
||||
import re |
||||
|
||||
sys.path.append(os.getcwd() + "/class/core") |
||||
import mw |
||||
|
||||
app_debug = False |
||||
if mw.isAppleSystem(): |
||||
app_debug = True |
||||
|
||||
|
||||
def getPluginName(): |
||||
return 'nosql_query' |
||||
|
||||
|
||||
def getPluginDir(): |
||||
return mw.getPluginDir() + '/' + getPluginName() |
||||
|
||||
|
||||
def getServerDir(): |
||||
return mw.getServerDir() + '/' + getPluginName() |
||||
|
||||
|
||||
def getInitDFile(): |
||||
current_os = mw.getOs() |
||||
if current_os == 'darwin': |
||||
return '/tmp/' + getPluginName() |
||||
|
||||
if current_os.startswith('freebsd'): |
||||
return '/etc/rc.d/' + getPluginName() |
||||
|
||||
return '/etc/init.d/' + getPluginName() |
||||
|
||||
|
||||
def getConf(): |
||||
path = getServerDir() + "/redis.conf" |
||||
return path |
||||
|
||||
|
||||
def getConfTpl(): |
||||
path = getPluginDir() + "/config/redis.conf" |
||||
return path |
||||
|
||||
|
||||
def getInitDTpl(): |
||||
path = getPluginDir() + "/init.d/" + getPluginName() + ".tpl" |
||||
return path |
||||
|
||||
|
||||
def getArgs(): |
||||
args = sys.argv[3:] |
||||
tmp = {} |
||||
args_len = len(args) |
||||
|
||||
if args_len == 1: |
||||
t = args[0].strip('{').strip('}') |
||||
if t.strip() == '': |
||||
tmp = [] |
||||
else: |
||||
t = t.split(':') |
||||
tmp[t[0]] = t[1] |
||||
tmp[t[0]] = t[1] |
||||
elif args_len > 1: |
||||
for i in range(len(args)): |
||||
t = args[i].split(':') |
||||
tmp[t[0]] = t[1] |
||||
return tmp |
||||
|
||||
def checkArgs(data, ck=[]): |
||||
for i in range(len(ck)): |
||||
if not ck[i] in data: |
||||
return (False, mw.returnJson(False, '参数:(' + ck[i] + ')没有!')) |
||||
return (True, mw.returnJson(True, 'ok')) |
||||
|
||||
def configTpl(): |
||||
path = getPluginDir() + '/tpl' |
||||
pathFile = os.listdir(path) |
||||
tmp = [] |
||||
for one in pathFile: |
||||
file = path + '/' + one |
||||
tmp.append(file) |
||||
return mw.getJson(tmp) |
||||
|
||||
|
||||
def readConfigTpl(): |
||||
args = getArgs() |
||||
data = checkArgs(args, ['file']) |
||||
if not data[0]: |
||||
return data[1] |
||||
|
||||
content = mw.readFile(args['file']) |
||||
content = contentReplace(content) |
||||
return mw.returnJson(True, 'ok', content) |
||||
|
||||
def getPidFile(): |
||||
file = getConf() |
||||
content = mw.readFile(file) |
||||
rep = 'pidfile\s*(.*)' |
||||
tmp = re.search(rep, content) |
||||
return tmp.groups()[0].strip() |
||||
|
||||
def status(): |
||||
return 'start' |
||||
|
||||
def contentReplace(content): |
||||
service_path = mw.getServerDir() |
||||
content = content.replace('{$ROOT_PATH}', mw.getRootDir()) |
||||
content = content.replace('{$SERVER_PATH}', service_path) |
||||
content = content.replace('{$SERVER_APP}', service_path + '/redis') |
||||
content = content.replace('{$REDIS_PASS}', mw.getRandomString(10)) |
||||
return content |
||||
|
||||
|
||||
|
||||
def initDreplace(): |
||||
|
||||
file_tpl = getInitDTpl() |
||||
service_path = os.path.dirname(os.getcwd()) |
||||
|
||||
initD_path = getServerDir() + '/init.d' |
||||
if not os.path.exists(initD_path): |
||||
os.mkdir(initD_path) |
||||
file_bin = initD_path + '/' + getPluginName() |
||||
|
||||
# initd replace |
||||
if not os.path.exists(file_bin): |
||||
content = mw.readFile(file_tpl) |
||||
content = content.replace('{$SERVER_PATH}', service_path) |
||||
mw.writeFile(file_bin, content) |
||||
mw.execShell('chmod +x ' + file_bin) |
||||
|
||||
# log |
||||
dataLog = getServerDir() + '/data' |
||||
if not os.path.exists(dataLog): |
||||
mw.execShell('chmod +x ' + file_bin) |
||||
|
||||
# config replace |
||||
dst_conf = getServerDir() + '/redis.conf' |
||||
dst_conf_init = getServerDir() + '/init.pl' |
||||
if not os.path.exists(dst_conf_init): |
||||
conf_content = mw.readFile(getConfTpl()) |
||||
conf_content = conf_content.replace('{$SERVER_PATH}', service_path) |
||||
conf_content = conf_content.replace( |
||||
'{$REDIS_PASS}', mw.getRandomString(10)) |
||||
|
||||
mw.writeFile(dst_conf, conf_content) |
||||
mw.writeFile(dst_conf_init, 'ok') |
||||
|
||||
# systemd |
||||
systemDir = mw.systemdCfgDir() |
||||
systemService = systemDir + '/' + getPluginName() + '.service' |
||||
if os.path.exists(systemDir) and not os.path.exists(systemService): |
||||
systemServiceTpl = getPluginDir() + '/init.d/' + getPluginName() + '.service.tpl' |
||||
service_path = mw.getServerDir() |
||||
se_content = mw.readFile(systemServiceTpl) |
||||
se_content = se_content.replace('{$SERVER_PATH}', service_path) |
||||
mw.writeFile(systemService, se_content) |
||||
mw.execShell('systemctl daemon-reload') |
||||
|
||||
return file_bin |
||||
|
||||
|
||||
def redisOp(method): |
||||
file = initDreplace() |
||||
|
||||
current_os = mw.getOs() |
||||
if current_os == "darwin": |
||||
data = mw.execShell(file + ' ' + method) |
||||
if data[1] == '': |
||||
return 'ok' |
||||
return data[1] |
||||
|
||||
if current_os.startswith("freebsd"): |
||||
data = mw.execShell('service ' + getPluginName() + ' ' + method) |
||||
if data[1] == '': |
||||
return 'ok' |
||||
return data[1] |
||||
|
||||
data = mw.execShell('systemctl ' + method + ' ' + getPluginName()) |
||||
if data[1] == '': |
||||
return 'ok' |
||||
return data[1] |
||||
|
||||
|
||||
def start(): |
||||
return redisOp('start') |
||||
|
||||
|
||||
def stop(): |
||||
return redisOp('stop') |
||||
|
||||
|
||||
def restart(): |
||||
status = redisOp('restart') |
||||
|
||||
log_file = runLog() |
||||
mw.execShell("echo '' > " + log_file) |
||||
return status |
||||
|
||||
|
||||
def reload(): |
||||
return redisOp('reload') |
||||
|
||||
|
||||
def getPort(): |
||||
conf = getServerDir() + '/redis.conf' |
||||
content = mw.readFile(conf) |
||||
|
||||
rep = "^(" + 'port' + ')\s*([.0-9A-Za-z_& ~]+)' |
||||
tmp = re.search(rep, content, re.M) |
||||
if tmp: |
||||
return tmp.groups()[1] |
||||
|
||||
return '6379' |
||||
|
||||
|
||||
def getRedisCmd(): |
||||
requirepass = "" |
||||
conf = getServerDir() + '/redis.conf' |
||||
content = mw.readFile(conf) |
||||
rep = "^(requirepass" + ')\s*([.0-9A-Za-z_& ~]+)' |
||||
tmp = re.search(rep, content, re.M) |
||||
if tmp: |
||||
requirepass = tmp.groups()[1] |
||||
|
||||
default_ip = '127.0.0.1' |
||||
port = getPort() |
||||
# findDebian = mw.execShell('cat /etc/issue |grep Debian') |
||||
# if findDebian[0] != '': |
||||
# default_ip = mw.getLocalIp() |
||||
cmd = getServerDir() + "/bin/redis-cli -h " + \ |
||||
default_ip + ' -p ' + port + " " |
||||
|
||||
if requirepass != "": |
||||
cmd = getServerDir() + '/bin/redis-cli -h ' + default_ip + \ |
||||
' -p ' + port + ' -a "' + requirepass + '" ' |
||||
|
||||
return cmd |
||||
|
||||
def runInfo(): |
||||
s = status() |
||||
if s == 'stop': |
||||
return mw.returnJson(False, '未启动') |
||||
|
||||
|
||||
cmd = getRedisCmd() |
||||
cmd = cmd + 'info' |
||||
|
||||
# print(cmd) |
||||
data = mw.execShell(cmd)[0] |
||||
# print(data) |
||||
res = [ |
||||
'tcp_port', |
||||
'uptime_in_days', # 已运行天数 |
||||
'connected_clients', # 连接的客户端数量 |
||||
'used_memory', # Redis已分配的内存总量 |
||||
'used_memory_rss', # Redis占用的系统内存总量 |
||||
'used_memory_peak', # Redis所用内存的高峰值 |
||||
'mem_fragmentation_ratio', # 内存碎片比率 |
||||
'total_connections_received', # 运行以来连接过的客户端的总数量 |
||||
'total_commands_processed', # 运行以来执行过的命令的总数量 |
||||
'instantaneous_ops_per_sec', # 服务器每秒钟执行的命令数量 |
||||
'keyspace_hits', # 查找数据库键成功的次数 |
||||
'keyspace_misses', # 查找数据库键失败的次数 |
||||
'latest_fork_usec' # 最近一次 fork() 操作耗费的毫秒数 |
||||
] |
||||
data = data.split("\n") |
||||
result = {} |
||||
for d in data: |
||||
if len(d) < 3: |
||||
continue |
||||
t = d.strip().split(':') |
||||
if not t[0] in res: |
||||
continue |
||||
result[t[0]] = t[1] |
||||
return mw.getJson(result) |
||||
|
||||
def infoReplication(): |
||||
# 复制信息 |
||||
s = status() |
||||
if s == 'stop': |
||||
return mw.returnJson(False, '未启动') |
||||
|
||||
cmd = getRedisCmd() |
||||
cmd = cmd + 'info replication' |
||||
|
||||
# print(cmd) |
||||
data = mw.execShell(cmd)[0] |
||||
# print(data) |
||||
res = [ |
||||
#slave |
||||
'role',#角色 |
||||
'master_host', # 连接主库HOST |
||||
'master_port', # 连接主库PORT |
||||
'master_link_status', # 连接主库状态 |
||||
'master_last_io_seconds_ago', # 上次同步时间 |
||||
'master_sync_in_progress', # 正在同步中 |
||||
'slave_read_repl_offset', # 从库读取复制位置 |
||||
'slave_repl_offset', # 从库复制位置 |
||||
'slave_priority', # 从库同步优先级 |
||||
'slave_read_only', # 从库是否仅读 |
||||
'replica_announced', # 已复制副本 |
||||
'connected_slaves', # 连接从库数量 |
||||
'master_failover_state', # 主库故障状态 |
||||
'master_replid', # 主库复制ID |
||||
'master_repl_offset', # 主库复制位置 |
||||
'second_repl_offset', # 主库复制位置时间 |
||||
'repl_backlog_active', # 复制状态 |
||||
'repl_backlog_size', # 复制大小 |
||||
'repl_backlog_first_byte_offset', # 第一个字节偏移量 |
||||
'repl_backlog_histlen', # backlog中数据的长度 |
||||
] |
||||
|
||||
data = data.split("\n") |
||||
result = {} |
||||
for d in data: |
||||
if len(d) < 3: |
||||
continue |
||||
t = d.strip().split(':') |
||||
if not t[0] in res: |
||||
continue |
||||
result[t[0]] = t[1] |
||||
|
||||
if 'role' in result and result['role'] == 'master': |
||||
connected_slaves = int(result['connected_slaves']) |
||||
slave_l = [] |
||||
for x in range(connected_slaves): |
||||
slave_l.append('slave'+str(x)) |
||||
|
||||
for d in data: |
||||
if len(d) < 3: |
||||
continue |
||||
t = d.strip().split(':') |
||||
if not t[0] in slave_l: |
||||
continue |
||||
result[t[0]] = t[1] |
||||
|
||||
return mw.getJson(result) |
||||
|
||||
|
||||
def clusterInfo(): |
||||
#集群信息 |
||||
# https://redis.io/commands/cluster-info/ |
||||
s = status() |
||||
if s == 'stop': |
||||
return mw.returnJson(False, '未启动') |
||||
|
||||
cmd = getRedisCmd() |
||||
cmd = cmd + 'cluster info' |
||||
|
||||
# print(cmd) |
||||
data = mw.execShell(cmd)[0] |
||||
# print(data) |
||||
|
||||
res = [ |
||||
'cluster_state',#状态 |
||||
'cluster_slots_assigned', # 被分配的槽 |
||||
'cluster_slots_ok', # 被分配的槽状态 |
||||
'cluster_slots_pfail', # 连接主库状态 |
||||
'cluster_slots_fail', # 失败的槽 |
||||
'cluster_known_nodes', # 知道的节点 |
||||
'cluster_size', # 大小 |
||||
'cluster_current_epoch', # |
||||
'cluster_my_epoch', # |
||||
'cluster_stats_messages_sent', # 发送 |
||||
'cluster_stats_messages_received', # 接受 |
||||
'total_cluster_links_buffer_limit_exceeded', # |
||||
] |
||||
|
||||
data = data.split("\n") |
||||
result = {} |
||||
for d in data: |
||||
if len(d) < 3: |
||||
continue |
||||
t = d.strip().split(':') |
||||
if not t[0] in res: |
||||
continue |
||||
result[t[0]] = t[1] |
||||
|
||||
return mw.getJson(result) |
||||
|
||||
def clusterNodes(): |
||||
s = status() |
||||
if s == 'stop': |
||||
return mw.returnJson(False, '未启动') |
||||
|
||||
cmd = getRedisCmd() |
||||
cmd = cmd + 'cluster nodes' |
||||
|
||||
# print(cmd) |
||||
data = mw.execShell(cmd)[0] |
||||
# print(data) |
||||
|
||||
data = data.strip().split("\n") |
||||
return mw.getJson(data) |
||||
|
||||
def initdStatus(): |
||||
current_os = mw.getOs() |
||||
if current_os == 'darwin': |
||||
return "Apple Computer does not support" |
||||
|
||||
if current_os.startswith('freebsd'): |
||||
initd_bin = getInitDFile() |
||||
if os.path.exists(initd_bin): |
||||
return 'ok' |
||||
|
||||
shell_cmd = 'systemctl status ' + \ |
||||
getPluginName() + ' | grep loaded | grep "enabled;"' |
||||
data = mw.execShell(shell_cmd) |
||||
if data[0] == '': |
||||
return 'fail' |
||||
return 'ok' |
||||
|
||||
|
||||
def initdInstall(): |
||||
current_os = mw.getOs() |
||||
if current_os == 'darwin': |
||||
return "Apple Computer does not support" |
||||
|
||||
# freebsd initd install |
||||
if current_os.startswith('freebsd'): |
||||
import shutil |
||||
source_bin = initDreplace() |
||||
initd_bin = getInitDFile() |
||||
shutil.copyfile(source_bin, initd_bin) |
||||
mw.execShell('chmod +x ' + initd_bin) |
||||
mw.execShell('sysrc ' + getPluginName() + '_enable="YES"') |
||||
return 'ok' |
||||
|
||||
mw.execShell('systemctl enable ' + getPluginName()) |
||||
return 'ok' |
||||
|
||||
|
||||
def initdUinstall(): |
||||
current_os = mw.getOs() |
||||
if current_os == 'darwin': |
||||
return "Apple Computer does not support" |
||||
|
||||
if current_os.startswith('freebsd'): |
||||
initd_bin = getInitDFile() |
||||
os.remove(initd_bin) |
||||
mw.execShell('sysrc ' + getPluginName() + '_enable="NO"') |
||||
return 'ok' |
||||
|
||||
mw.execShell('systemctl disable ' + getPluginName()) |
||||
return 'ok' |
||||
|
||||
|
||||
def runLog(): |
||||
return getServerDir() + '/data/redis.log' |
||||
|
||||
|
||||
def getRedisConfInfo(): |
||||
conf = getServerDir() + '/redis.conf' |
||||
|
||||
gets = [ |
||||
{'name': 'bind', 'type': 2, 'ps': '绑定IP(修改绑定IP可能会存在安全隐患)','must_show':1}, |
||||
{'name': 'port', 'type': 2, 'ps': '绑定端口','must_show':1}, |
||||
{'name': 'timeout', 'type': 2, 'ps': '空闲链接超时时间,0表示不断开','must_show':1}, |
||||
{'name': 'maxclients', 'type': 2, 'ps': '最大连接数','must_show':1}, |
||||
{'name': 'databases', 'type': 2, 'ps': '数据库数量','must_show':1}, |
||||
{'name': 'requirepass', 'type': 2, 'ps': 'redis密码,留空代表没有设置密码','must_show':1}, |
||||
{'name': 'maxmemory', 'type': 2, 'ps': 'MB,最大使用内存,0表示不限制','must_show':1}, |
||||
{'name': 'slaveof', 'type': 2, 'ps': '同步主库地址','must_show':0}, |
||||
{'name': 'masterauth', 'type': 2, 'ps': '同步主库密码', 'must_show':0} |
||||
] |
||||
content = mw.readFile(conf) |
||||
|
||||
result = [] |
||||
for g in gets: |
||||
rep = "^(" + g['name'] + ')\s*([.0-9A-Za-z_& ~]+)' |
||||
tmp = re.search(rep, content, re.M) |
||||
if not tmp: |
||||
if g['must_show'] == 0: |
||||
continue |
||||
|
||||
g['value'] = '' |
||||
result.append(g) |
||||
continue |
||||
g['value'] = tmp.groups()[1] |
||||
if g['name'] == 'maxmemory': |
||||
g['value'] = g['value'].strip("mb") |
||||
result.append(g) |
||||
|
||||
return result |
||||
|
||||
|
||||
def getRedisConf(): |
||||
data = getRedisConfInfo() |
||||
return mw.getJson(data) |
||||
|
||||
|
||||
def submitRedisConf(): |
||||
gets = ['bind', 'port', 'timeout', 'maxclients', |
||||
'databases', 'requirepass', 'maxmemory','slaveof','masterauth'] |
||||
args = getArgs() |
||||
conf = getServerDir() + '/redis.conf' |
||||
content = mw.readFile(conf) |
||||
for g in gets: |
||||
if g in args: |
||||
rep = g + '\s*([.0-9A-Za-z_& ~]+)' |
||||
val = g + ' ' + args[g] |
||||
|
||||
if g == 'maxmemory': |
||||
val = g + ' ' + args[g] + "mb" |
||||
|
||||
if g == 'requirepass' and args[g] == '': |
||||
content = re.sub('requirepass', '#requirepass', content) |
||||
if g == 'requirepass' and args[g] != '': |
||||
content = re.sub('#requirepass', 'requirepass', content) |
||||
content = re.sub(rep, val, content) |
||||
|
||||
if g != 'requirepass': |
||||
content = re.sub(rep, val, content) |
||||
mw.writeFile(conf, content) |
||||
reload() |
||||
return mw.returnJson(True, '设置成功') |
||||
|
||||
if __name__ == "__main__": |
||||
func = sys.argv[1] |
||||
if func == 'status': |
||||
print(status()) |
||||
elif func == 'start': |
||||
print(start()) |
||||
elif func == 'stop': |
||||
print(stop()) |
||||
elif func == 'restart': |
||||
print(restart()) |
||||
elif func == 'reload': |
||||
print(reload()) |
||||
elif func == 'initd_status': |
||||
print(initdStatus()) |
||||
elif func == 'initd_install': |
||||
print(initdInstall()) |
||||
elif func == 'initd_uninstall': |
||||
print(initdUinstall()) |
||||
elif func == 'run_info': |
||||
print(runInfo()) |
||||
elif func == 'info_replication': |
||||
print(infoReplication()) |
||||
elif func == 'cluster_info': |
||||
print(clusterInfo()) |
||||
elif func == 'cluster_nodes': |
||||
print(clusterNodes()) |
||||
elif func == 'conf': |
||||
print(getConf()) |
||||
elif func == 'run_log': |
||||
print(runLog()) |
||||
elif func == 'get_redis_conf': |
||||
print(getRedisConf()) |
||||
elif func == 'submit_redis_conf': |
||||
print(submitRedisConf()) |
||||
elif func == 'config_tpl': |
||||
print(configTpl()) |
||||
elif func == 'read_config_tpl': |
||||
print(readConfigTpl()) |
||||
else: |
||||
print('error') |
@ -0,0 +1,563 @@ |
||||
# coding:utf-8 |
||||
|
||||
import sys |
||||
import io |
||||
import os |
||||
import time |
||||
import re |
||||
|
||||
sys.path.append(os.getcwd() + "/class/core") |
||||
import mw |
||||
|
||||
app_debug = False |
||||
if mw.isAppleSystem(): |
||||
app_debug = True |
||||
|
||||
|
||||
def getPluginName(): |
||||
return 'nosql_query' |
||||
|
||||
|
||||
def getPluginDir(): |
||||
return mw.getPluginDir() + '/' + getPluginName() |
||||
|
||||
|
||||
def getServerDir(): |
||||
return mw.getServerDir() + '/' + getPluginName() |
||||
|
||||
|
||||
def getInitDFile(): |
||||
current_os = mw.getOs() |
||||
if current_os == 'darwin': |
||||
return '/tmp/' + getPluginName() |
||||
|
||||
if current_os.startswith('freebsd'): |
||||
return '/etc/rc.d/' + getPluginName() |
||||
|
||||
return '/etc/init.d/' + getPluginName() |
||||
|
||||
|
||||
def getConf(): |
||||
path = getServerDir() + "/redis.conf" |
||||
return path |
||||
|
||||
|
||||
def getConfTpl(): |
||||
path = getPluginDir() + "/config/redis.conf" |
||||
return path |
||||
|
||||
|
||||
def getInitDTpl(): |
||||
path = getPluginDir() + "/init.d/" + getPluginName() + ".tpl" |
||||
return path |
||||
|
||||
|
||||
def getArgs(): |
||||
args = sys.argv[3:] |
||||
tmp = {} |
||||
args_len = len(args) |
||||
|
||||
if args_len == 1: |
||||
t = args[0].strip('{').strip('}') |
||||
if t.strip() == '': |
||||
tmp = [] |
||||
else: |
||||
t = t.split(':') |
||||
tmp[t[0]] = t[1] |
||||
tmp[t[0]] = t[1] |
||||
elif args_len > 1: |
||||
for i in range(len(args)): |
||||
t = args[i].split(':') |
||||
tmp[t[0]] = t[1] |
||||
return tmp |
||||
|
||||
def checkArgs(data, ck=[]): |
||||
for i in range(len(ck)): |
||||
if not ck[i] in data: |
||||
return (False, mw.returnJson(False, '参数:(' + ck[i] + ')没有!')) |
||||
return (True, mw.returnJson(True, 'ok')) |
||||
|
||||
def configTpl(): |
||||
path = getPluginDir() + '/tpl' |
||||
pathFile = os.listdir(path) |
||||
tmp = [] |
||||
for one in pathFile: |
||||
file = path + '/' + one |
||||
tmp.append(file) |
||||
return mw.getJson(tmp) |
||||
|
||||
|
||||
def readConfigTpl(): |
||||
args = getArgs() |
||||
data = checkArgs(args, ['file']) |
||||
if not data[0]: |
||||
return data[1] |
||||
|
||||
content = mw.readFile(args['file']) |
||||
content = contentReplace(content) |
||||
return mw.returnJson(True, 'ok', content) |
||||
|
||||
def getPidFile(): |
||||
file = getConf() |
||||
content = mw.readFile(file) |
||||
rep = 'pidfile\s*(.*)' |
||||
tmp = re.search(rep, content) |
||||
return tmp.groups()[0].strip() |
||||
|
||||
def status(): |
||||
return 'start' |
||||
|
||||
def contentReplace(content): |
||||
service_path = mw.getServerDir() |
||||
content = content.replace('{$ROOT_PATH}', mw.getRootDir()) |
||||
content = content.replace('{$SERVER_PATH}', service_path) |
||||
content = content.replace('{$SERVER_APP}', service_path + '/redis') |
||||
content = content.replace('{$REDIS_PASS}', mw.getRandomString(10)) |
||||
return content |
||||
|
||||
|
||||
|
||||
def initDreplace(): |
||||
|
||||
file_tpl = getInitDTpl() |
||||
service_path = os.path.dirname(os.getcwd()) |
||||
|
||||
initD_path = getServerDir() + '/init.d' |
||||
if not os.path.exists(initD_path): |
||||
os.mkdir(initD_path) |
||||
file_bin = initD_path + '/' + getPluginName() |
||||
|
||||
# initd replace |
||||
if not os.path.exists(file_bin): |
||||
content = mw.readFile(file_tpl) |
||||
content = content.replace('{$SERVER_PATH}', service_path) |
||||
mw.writeFile(file_bin, content) |
||||
mw.execShell('chmod +x ' + file_bin) |
||||
|
||||
# log |
||||
dataLog = getServerDir() + '/data' |
||||
if not os.path.exists(dataLog): |
||||
mw.execShell('chmod +x ' + file_bin) |
||||
|
||||
# config replace |
||||
dst_conf = getServerDir() + '/redis.conf' |
||||
dst_conf_init = getServerDir() + '/init.pl' |
||||
if not os.path.exists(dst_conf_init): |
||||
conf_content = mw.readFile(getConfTpl()) |
||||
conf_content = conf_content.replace('{$SERVER_PATH}', service_path) |
||||
conf_content = conf_content.replace( |
||||
'{$REDIS_PASS}', mw.getRandomString(10)) |
||||
|
||||
mw.writeFile(dst_conf, conf_content) |
||||
mw.writeFile(dst_conf_init, 'ok') |
||||
|
||||
# systemd |
||||
systemDir = mw.systemdCfgDir() |
||||
systemService = systemDir + '/' + getPluginName() + '.service' |
||||
if os.path.exists(systemDir) and not os.path.exists(systemService): |
||||
systemServiceTpl = getPluginDir() + '/init.d/' + getPluginName() + '.service.tpl' |
||||
service_path = mw.getServerDir() |
||||
se_content = mw.readFile(systemServiceTpl) |
||||
se_content = se_content.replace('{$SERVER_PATH}', service_path) |
||||
mw.writeFile(systemService, se_content) |
||||
mw.execShell('systemctl daemon-reload') |
||||
|
||||
return file_bin |
||||
|
||||
|
||||
def redisOp(method): |
||||
file = initDreplace() |
||||
|
||||
current_os = mw.getOs() |
||||
if current_os == "darwin": |
||||
data = mw.execShell(file + ' ' + method) |
||||
if data[1] == '': |
||||
return 'ok' |
||||
return data[1] |
||||
|
||||
if current_os.startswith("freebsd"): |
||||
data = mw.execShell('service ' + getPluginName() + ' ' + method) |
||||
if data[1] == '': |
||||
return 'ok' |
||||
return data[1] |
||||
|
||||
data = mw.execShell('systemctl ' + method + ' ' + getPluginName()) |
||||
if data[1] == '': |
||||
return 'ok' |
||||
return data[1] |
||||
|
||||
|
||||
def start(): |
||||
return redisOp('start') |
||||
|
||||
|
||||
def stop(): |
||||
return redisOp('stop') |
||||
|
||||
|
||||
def restart(): |
||||
status = redisOp('restart') |
||||
|
||||
log_file = runLog() |
||||
mw.execShell("echo '' > " + log_file) |
||||
return status |
||||
|
||||
|
||||
def reload(): |
||||
return redisOp('reload') |
||||
|
||||
|
||||
def getPort(): |
||||
conf = getServerDir() + '/redis.conf' |
||||
content = mw.readFile(conf) |
||||
|
||||
rep = "^(" + 'port' + ')\s*([.0-9A-Za-z_& ~]+)' |
||||
tmp = re.search(rep, content, re.M) |
||||
if tmp: |
||||
return tmp.groups()[1] |
||||
|
||||
return '6379' |
||||
|
||||
|
||||
def getRedisCmd(): |
||||
requirepass = "" |
||||
conf = getServerDir() + '/redis.conf' |
||||
content = mw.readFile(conf) |
||||
rep = "^(requirepass" + ')\s*([.0-9A-Za-z_& ~]+)' |
||||
tmp = re.search(rep, content, re.M) |
||||
if tmp: |
||||
requirepass = tmp.groups()[1] |
||||
|
||||
default_ip = '127.0.0.1' |
||||
port = getPort() |
||||
# findDebian = mw.execShell('cat /etc/issue |grep Debian') |
||||
# if findDebian[0] != '': |
||||
# default_ip = mw.getLocalIp() |
||||
cmd = getServerDir() + "/bin/redis-cli -h " + \ |
||||
default_ip + ' -p ' + port + " " |
||||
|
||||
if requirepass != "": |
||||
cmd = getServerDir() + '/bin/redis-cli -h ' + default_ip + \ |
||||
' -p ' + port + ' -a "' + requirepass + '" ' |
||||
|
||||
return cmd |
||||
|
||||
def runInfo(): |
||||
s = status() |
||||
if s == 'stop': |
||||
return mw.returnJson(False, '未启动') |
||||
|
||||
|
||||
cmd = getRedisCmd() |
||||
cmd = cmd + 'info' |
||||
|
||||
# print(cmd) |
||||
data = mw.execShell(cmd)[0] |
||||
# print(data) |
||||
res = [ |
||||
'tcp_port', |
||||
'uptime_in_days', # 已运行天数 |
||||
'connected_clients', # 连接的客户端数量 |
||||
'used_memory', # Redis已分配的内存总量 |
||||
'used_memory_rss', # Redis占用的系统内存总量 |
||||
'used_memory_peak', # Redis所用内存的高峰值 |
||||
'mem_fragmentation_ratio', # 内存碎片比率 |
||||
'total_connections_received', # 运行以来连接过的客户端的总数量 |
||||
'total_commands_processed', # 运行以来执行过的命令的总数量 |
||||
'instantaneous_ops_per_sec', # 服务器每秒钟执行的命令数量 |
||||
'keyspace_hits', # 查找数据库键成功的次数 |
||||
'keyspace_misses', # 查找数据库键失败的次数 |
||||
'latest_fork_usec' # 最近一次 fork() 操作耗费的毫秒数 |
||||
] |
||||
data = data.split("\n") |
||||
result = {} |
||||
for d in data: |
||||
if len(d) < 3: |
||||
continue |
||||
t = d.strip().split(':') |
||||
if not t[0] in res: |
||||
continue |
||||
result[t[0]] = t[1] |
||||
return mw.getJson(result) |
||||
|
||||
def infoReplication(): |
||||
# 复制信息 |
||||
s = status() |
||||
if s == 'stop': |
||||
return mw.returnJson(False, '未启动') |
||||
|
||||
cmd = getRedisCmd() |
||||
cmd = cmd + 'info replication' |
||||
|
||||
# print(cmd) |
||||
data = mw.execShell(cmd)[0] |
||||
# print(data) |
||||
res = [ |
||||
#slave |
||||
'role',#角色 |
||||
'master_host', # 连接主库HOST |
||||
'master_port', # 连接主库PORT |
||||
'master_link_status', # 连接主库状态 |
||||
'master_last_io_seconds_ago', # 上次同步时间 |
||||
'master_sync_in_progress', # 正在同步中 |
||||
'slave_read_repl_offset', # 从库读取复制位置 |
||||
'slave_repl_offset', # 从库复制位置 |
||||
'slave_priority', # 从库同步优先级 |
||||
'slave_read_only', # 从库是否仅读 |
||||
'replica_announced', # 已复制副本 |
||||
'connected_slaves', # 连接从库数量 |
||||
'master_failover_state', # 主库故障状态 |
||||
'master_replid', # 主库复制ID |
||||
'master_repl_offset', # 主库复制位置 |
||||
'second_repl_offset', # 主库复制位置时间 |
||||
'repl_backlog_active', # 复制状态 |
||||
'repl_backlog_size', # 复制大小 |
||||
'repl_backlog_first_byte_offset', # 第一个字节偏移量 |
||||
'repl_backlog_histlen', # backlog中数据的长度 |
||||
] |
||||
|
||||
data = data.split("\n") |
||||
result = {} |
||||
for d in data: |
||||
if len(d) < 3: |
||||
continue |
||||
t = d.strip().split(':') |
||||
if not t[0] in res: |
||||
continue |
||||
result[t[0]] = t[1] |
||||
|
||||
if 'role' in result and result['role'] == 'master': |
||||
connected_slaves = int(result['connected_slaves']) |
||||
slave_l = [] |
||||
for x in range(connected_slaves): |
||||
slave_l.append('slave'+str(x)) |
||||
|
||||
for d in data: |
||||
if len(d) < 3: |
||||
continue |
||||
t = d.strip().split(':') |
||||
if not t[0] in slave_l: |
||||
continue |
||||
result[t[0]] = t[1] |
||||
|
||||
return mw.getJson(result) |
||||
|
||||
|
||||
def clusterInfo(): |
||||
#集群信息 |
||||
# https://redis.io/commands/cluster-info/ |
||||
s = status() |
||||
if s == 'stop': |
||||
return mw.returnJson(False, '未启动') |
||||
|
||||
cmd = getRedisCmd() |
||||
cmd = cmd + 'cluster info' |
||||
|
||||
# print(cmd) |
||||
data = mw.execShell(cmd)[0] |
||||
# print(data) |
||||
|
||||
res = [ |
||||
'cluster_state',#状态 |
||||
'cluster_slots_assigned', # 被分配的槽 |
||||
'cluster_slots_ok', # 被分配的槽状态 |
||||
'cluster_slots_pfail', # 连接主库状态 |
||||
'cluster_slots_fail', # 失败的槽 |
||||
'cluster_known_nodes', # 知道的节点 |
||||
'cluster_size', # 大小 |
||||
'cluster_current_epoch', # |
||||
'cluster_my_epoch', # |
||||
'cluster_stats_messages_sent', # 发送 |
||||
'cluster_stats_messages_received', # 接受 |
||||
'total_cluster_links_buffer_limit_exceeded', # |
||||
] |
||||
|
||||
data = data.split("\n") |
||||
result = {} |
||||
for d in data: |
||||
if len(d) < 3: |
||||
continue |
||||
t = d.strip().split(':') |
||||
if not t[0] in res: |
||||
continue |
||||
result[t[0]] = t[1] |
||||
|
||||
return mw.getJson(result) |
||||
|
||||
def clusterNodes(): |
||||
s = status() |
||||
if s == 'stop': |
||||
return mw.returnJson(False, '未启动') |
||||
|
||||
cmd = getRedisCmd() |
||||
cmd = cmd + 'cluster nodes' |
||||
|
||||
# print(cmd) |
||||
data = mw.execShell(cmd)[0] |
||||
# print(data) |
||||
|
||||
data = data.strip().split("\n") |
||||
return mw.getJson(data) |
||||
|
||||
def initdStatus(): |
||||
current_os = mw.getOs() |
||||
if current_os == 'darwin': |
||||
return "Apple Computer does not support" |
||||
|
||||
if current_os.startswith('freebsd'): |
||||
initd_bin = getInitDFile() |
||||
if os.path.exists(initd_bin): |
||||
return 'ok' |
||||
|
||||
shell_cmd = 'systemctl status ' + \ |
||||
getPluginName() + ' | grep loaded | grep "enabled;"' |
||||
data = mw.execShell(shell_cmd) |
||||
if data[0] == '': |
||||
return 'fail' |
||||
return 'ok' |
||||
|
||||
|
||||
def initdInstall(): |
||||
current_os = mw.getOs() |
||||
if current_os == 'darwin': |
||||
return "Apple Computer does not support" |
||||
|
||||
# freebsd initd install |
||||
if current_os.startswith('freebsd'): |
||||
import shutil |
||||
source_bin = initDreplace() |
||||
initd_bin = getInitDFile() |
||||
shutil.copyfile(source_bin, initd_bin) |
||||
mw.execShell('chmod +x ' + initd_bin) |
||||
mw.execShell('sysrc ' + getPluginName() + '_enable="YES"') |
||||
return 'ok' |
||||
|
||||
mw.execShell('systemctl enable ' + getPluginName()) |
||||
return 'ok' |
||||
|
||||
|
||||
def initdUinstall(): |
||||
current_os = mw.getOs() |
||||
if current_os == 'darwin': |
||||
return "Apple Computer does not support" |
||||
|
||||
if current_os.startswith('freebsd'): |
||||
initd_bin = getInitDFile() |
||||
os.remove(initd_bin) |
||||
mw.execShell('sysrc ' + getPluginName() + '_enable="NO"') |
||||
return 'ok' |
||||
|
||||
mw.execShell('systemctl disable ' + getPluginName()) |
||||
return 'ok' |
||||
|
||||
|
||||
def runLog(): |
||||
return getServerDir() + '/data/redis.log' |
||||
|
||||
|
||||
def getRedisConfInfo(): |
||||
conf = getServerDir() + '/redis.conf' |
||||
|
||||
gets = [ |
||||
{'name': 'bind', 'type': 2, 'ps': '绑定IP(修改绑定IP可能会存在安全隐患)','must_show':1}, |
||||
{'name': 'port', 'type': 2, 'ps': '绑定端口','must_show':1}, |
||||
{'name': 'timeout', 'type': 2, 'ps': '空闲链接超时时间,0表示不断开','must_show':1}, |
||||
{'name': 'maxclients', 'type': 2, 'ps': '最大连接数','must_show':1}, |
||||
{'name': 'databases', 'type': 2, 'ps': '数据库数量','must_show':1}, |
||||
{'name': 'requirepass', 'type': 2, 'ps': 'redis密码,留空代表没有设置密码','must_show':1}, |
||||
{'name': 'maxmemory', 'type': 2, 'ps': 'MB,最大使用内存,0表示不限制','must_show':1}, |
||||
{'name': 'slaveof', 'type': 2, 'ps': '同步主库地址','must_show':0}, |
||||
{'name': 'masterauth', 'type': 2, 'ps': '同步主库密码', 'must_show':0} |
||||
] |
||||
content = mw.readFile(conf) |
||||
|
||||
result = [] |
||||
for g in gets: |
||||
rep = "^(" + g['name'] + ')\s*([.0-9A-Za-z_& ~]+)' |
||||
tmp = re.search(rep, content, re.M) |
||||
if not tmp: |
||||
if g['must_show'] == 0: |
||||
continue |
||||
|
||||
g['value'] = '' |
||||
result.append(g) |
||||
continue |
||||
g['value'] = tmp.groups()[1] |
||||
if g['name'] == 'maxmemory': |
||||
g['value'] = g['value'].strip("mb") |
||||
result.append(g) |
||||
|
||||
return result |
||||
|
||||
|
||||
def getRedisConf(): |
||||
data = getRedisConfInfo() |
||||
return mw.getJson(data) |
||||
|
||||
|
||||
def submitRedisConf(): |
||||
gets = ['bind', 'port', 'timeout', 'maxclients', |
||||
'databases', 'requirepass', 'maxmemory','slaveof','masterauth'] |
||||
args = getArgs() |
||||
conf = getServerDir() + '/redis.conf' |
||||
content = mw.readFile(conf) |
||||
for g in gets: |
||||
if g in args: |
||||
rep = g + '\s*([.0-9A-Za-z_& ~]+)' |
||||
val = g + ' ' + args[g] |
||||
|
||||
if g == 'maxmemory': |
||||
val = g + ' ' + args[g] + "mb" |
||||
|
||||
if g == 'requirepass' and args[g] == '': |
||||
content = re.sub('requirepass', '#requirepass', content) |
||||
if g == 'requirepass' and args[g] != '': |
||||
content = re.sub('#requirepass', 'requirepass', content) |
||||
content = re.sub(rep, val, content) |
||||
|
||||
if g != 'requirepass': |
||||
content = re.sub(rep, val, content) |
||||
mw.writeFile(conf, content) |
||||
reload() |
||||
return mw.returnJson(True, '设置成功') |
||||
|
||||
if __name__ == "__main__": |
||||
func = sys.argv[1] |
||||
if func == 'status': |
||||
print(status()) |
||||
elif func == 'start': |
||||
print(start()) |
||||
elif func == 'stop': |
||||
print(stop()) |
||||
elif func == 'restart': |
||||
print(restart()) |
||||
elif func == 'reload': |
||||
print(reload()) |
||||
elif func == 'initd_status': |
||||
print(initdStatus()) |
||||
elif func == 'initd_install': |
||||
print(initdInstall()) |
||||
elif func == 'initd_uninstall': |
||||
print(initdUinstall()) |
||||
elif func == 'run_info': |
||||
print(runInfo()) |
||||
elif func == 'info_replication': |
||||
print(infoReplication()) |
||||
elif func == 'cluster_info': |
||||
print(clusterInfo()) |
||||
elif func == 'cluster_nodes': |
||||
print(clusterNodes()) |
||||
elif func == 'conf': |
||||
print(getConf()) |
||||
elif func == 'run_log': |
||||
print(runLog()) |
||||
elif func == 'get_redis_conf': |
||||
print(getRedisConf()) |
||||
elif func == 'submit_redis_conf': |
||||
print(submitRedisConf()) |
||||
elif func == 'config_tpl': |
||||
print(configTpl()) |
||||
elif func == 'read_config_tpl': |
||||
print(readConfigTpl()) |
||||
else: |
||||
print('error') |
@ -0,0 +1,347 @@ |
||||
# coding:utf-8 |
||||
|
||||
import sys |
||||
import io |
||||
import os |
||||
import time |
||||
import re |
||||
import redis |
||||
|
||||
sys.path.append(os.getcwd() + "/class/core") |
||||
import mw |
||||
|
||||
# def getPluginName(): |
||||
# return 'data_query' |
||||
|
||||
def singleton(cls): |
||||
_instance = {} |
||||
|
||||
def inner(): |
||||
if cls not in _instance: |
||||
_instance[cls] = cls() |
||||
return _instance[cls] |
||||
return inner |
||||
|
||||
@singleton |
||||
class nosqlRedis(): |
||||
|
||||
__DB_PASS = None |
||||
__DB_USER = None |
||||
__DB_PORT = 6379 |
||||
__DB_HOST = '127.0.0.1' |
||||
__DB_CONN = None |
||||
__DB_ERR = None |
||||
|
||||
__DB_LOCAL = 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']: |
||||
redis_path = "{}/redis".format(mw.getServerDir()) |
||||
if not os.path.exists(redis_path): return False |
||||
|
||||
if not self.__DB_LOCAL: |
||||
self.__DB_PASS = self.__config['requirepass'] |
||||
self.__DB_PORT = int(self.__config['port']) |
||||
|
||||
# print(self.__DB_HOST,self.__DB_PORT, self.__DB_PASS) |
||||
try: |
||||
redis_pool = redis.ConnectionPool(host=self.__DB_HOST, port=self.__DB_PORT, password=self.__DB_PASS, db=db_idx, socket_timeout=3) |
||||
self.__DB_CONN = redis.Redis(connection_pool=redis_pool) |
||||
self.__DB_CONN.ping() |
||||
return self.__DB_CONN |
||||
except redis.exceptions.ConnectionError: |
||||
return False |
||||
except Exception: |
||||
self.__DB_ERR = public.get_error_info() |
||||
return False |
||||
|
||||
# 获取配置项 |
||||
def get_options(self, get=None): |
||||
|
||||
result = {} |
||||
redis_conf = mw.readFile("{}/redis/redis.conf".format(mw.getServerDir())) |
||||
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).strip("mb")) |
||||
else: |
||||
v = group.group(1) |
||||
result[k] = v |
||||
return result |
||||
|
||||
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_LOCAL = 1 |
||||
return self |
||||
|
||||
|
||||
@singleton |
||||
class nosqlRedisCtr(): |
||||
|
||||
def __init__(self): |
||||
pass |
||||
|
||||
def getInstanceBySid(self, sid = 0): |
||||
instance = nosqlRedis() |
||||
return instance |
||||
|
||||
def getList(self, args): |
||||
|
||||
sid = args['sid'] |
||||
redis_instance = self.getInstanceBySid(sid).redis_conn(0) |
||||
if redis_instance is False: |
||||
return mw.returnData(False,'无法链接') |
||||
|
||||
|
||||
redis_info = redis_instance.info() |
||||
is_cluster = redis_info.get("cluster_enabled", 0) |
||||
if is_cluster != 0: |
||||
return mw.returnData(False, "当前不支持连接redis集群!") |
||||
|
||||
db_num = 16 |
||||
if sid != 0: |
||||
db_num = 1000 |
||||
|
||||
result = [] |
||||
for x in range(0, db_num): |
||||
data = {} |
||||
data['id'] = x |
||||
data['name'] = 'DB{}'.format(x) |
||||
try: |
||||
redis_instance = self.getInstanceBySid(sid).redis_conn(x) |
||||
data['keynum'] = redis_instance.dbsize() |
||||
result.append(data) |
||||
except: |
||||
break |
||||
|
||||
return mw.returnData(True,'ok', result) |
||||
|
||||
def getDbKeyList(self, args): |
||||
p = 1 |
||||
size = 10 |
||||
|
||||
if not 'sid' in args: |
||||
return mw.returnData(False, "缺少参数!sid") |
||||
|
||||
if 'p' in args: |
||||
p = args['p'] |
||||
if p < 1: |
||||
p = 1 |
||||
if p > 10: |
||||
p = 10 |
||||
|
||||
if 'size' in args: |
||||
size = args['size'] |
||||
|
||||
sid = args['sid'] |
||||
idx = args['idx'] |
||||
search = '*' |
||||
if 'search' in args and args['search'] != '': |
||||
search = args['search'] |
||||
|
||||
redis_instance = self.getInstanceBySid(sid).redis_conn(idx) |
||||
|
||||
total = redis_instance.dbsize() |
||||
|
||||
if search != '*': |
||||
keylist = redis_instance.keys(search) |
||||
total = len(keylist) |
||||
else: |
||||
keys = redis_instance.scan(cursor=0, match="{}".format(search), count=p*size) |
||||
keylist = keys[1] |
||||
|
||||
slist = keylist[(p - 1) * size:p*size] |
||||
|
||||
items = [] |
||||
for key in slist: |
||||
item = {} |
||||
try: |
||||
item['name'] = key.decode() |
||||
except Exception as e: |
||||
item['name'] = str(key) |
||||
|
||||
item['endtime'] = redis_instance.ttl(key) |
||||
item['type'] = redis_instance.type(key).decode() |
||||
|
||||
if item['type'] == 'string': |
||||
try: |
||||
item['val'] = redis_instance.get(key).decode() |
||||
except Exception as e: |
||||
item['val'] = str(redis_instance.get(key)) |
||||
elif item['type'] == 'hash': |
||||
if redis_instance.hlen(key) > 500: |
||||
item['val'] = "数据量过大无法显示!共 {} 条".format(redis_instance.hlen(key)) |
||||
else: |
||||
item['val'] = str(redis_instance.hgetall(key)) |
||||
elif item['type'] == 'list': |
||||
if redis_instance.llen(key) > 500: |
||||
item['val'] = "数据量过大无法显示!共 {} 条".format(redis_instance.llen(key)) |
||||
else: |
||||
item['val'] = str(redis_instance.lrange(key, 0, -1)) |
||||
elif item['type'] == 'set': |
||||
if redis_instance.scard(key) > 500: |
||||
item['val'] = "数据量过大无法显示!共 {} 条".format(redis_instance.scard(key)) |
||||
else: |
||||
item['val'] = str(redis_instance.smembers(key)) |
||||
elif item['type'] == 'zset': |
||||
if redis_instance.zcard(key) > 500: |
||||
item['val'] = "数据量过大无法显示!共 {} 条".format(redis_instance.zcard(key)) |
||||
else: |
||||
item['val'] = str(redis_instance.zrange(key, 0, -1, withscores=True)) |
||||
else: |
||||
item['val'] = '' |
||||
|
||||
try: |
||||
item['len'] = redis_instance.strlen(key) |
||||
except: |
||||
item['len'] = len(item['val']) |
||||
items.append(item) |
||||
|
||||
|
||||
page_args = {} |
||||
page_args['count'] = total |
||||
page_args['tojs'] = 'redisGetKeyList' |
||||
page_args['p'] = p |
||||
page_args['row'] = size |
||||
|
||||
rdata = {} |
||||
rdata['page'] = mw.getPage(page_args) |
||||
rdata['data'] = items |
||||
return mw.returnData(True,'ok',rdata) |
||||
|
||||
def setKv(self,args): |
||||
if not 'name' in args: |
||||
return mw.returnData(False, "缺少参数!name") |
||||
if not 'val' in args: |
||||
return mw.returnData(False, "缺少参数!val") |
||||
if not 'idx' in args: |
||||
return mw.returnData(False, "缺少参数!idx") |
||||
|
||||
sid = args['sid'] |
||||
idx = args['idx'] |
||||
|
||||
name = args["name"] |
||||
val = args["val"] |
||||
endtime = args["endtime"] |
||||
|
||||
redis_instance = self.getInstanceBySid(sid).redis_conn(idx) |
||||
|
||||
redis_info = redis_instance.info() |
||||
if redis_info['role'] == 'slave': |
||||
return mw.returnData(False,'从库不能写操作!') |
||||
|
||||
if endtime != '0': |
||||
redis_instance.set(name, val, int(endtime)) |
||||
else: |
||||
redis_instance.set(name, val) |
||||
|
||||
return mw.returnData(True,'操作成功') |
||||
|
||||
def delVal(self, args): |
||||
sid = args['sid'] |
||||
idx = args['idx'] |
||||
name = args["name"] |
||||
redis_instance = self.getInstanceBySid(sid).redis_conn(idx) |
||||
|
||||
redis_info = redis_instance.info() |
||||
if redis_info['role'] == 'slave': |
||||
return mw.returnData(False,'从库不能删除操作!') |
||||
|
||||
redis_instance.delete(name) |
||||
return mw.returnData(True,'操作成功') |
||||
|
||||
def batchDelVal(self, args): |
||||
sid = args['sid'] |
||||
idx = args['idx'] |
||||
keys = args["keys"] |
||||
redis_instance = self.getInstanceBySid(sid).redis_conn(idx) |
||||
|
||||
redis_info = redis_instance.info() |
||||
if redis_info['role'] == 'slave': |
||||
return mw.returnData(False,'从库不能删除操作!') |
||||
|
||||
for k in keys: |
||||
redis_instance.delete(k) |
||||
return mw.returnData(True,'操作成功') |
||||
|
||||
def clearFlushDB(self, args): |
||||
|
||||
sid = args['sid'] |
||||
idxs = args['idxs'] |
||||
|
||||
for idx in idxs: |
||||
redis_instance = self.getInstanceBySid(sid).redis_conn(idx) |
||||
|
||||
redis_info = redis_instance.info() |
||||
if redis_info['role'] == 'slave': |
||||
return mw.returnData(False,'从库不能清空操作!') |
||||
|
||||
redis_instance.flushdb() |
||||
|
||||
return mw.returnData(True,'操作成功') |
||||
|
||||
|
||||
# ---------------------------------- run ---------------------------------- |
||||
# 获取 redis databases 列表 |
||||
def get_list(args): |
||||
t = nosqlRedisCtr() |
||||
return t.getList(args) |
||||
|
||||
# 获取 redis key 列表 |
||||
def get_dbkey_list(args): |
||||
t = nosqlRedisCtr() |
||||
return t.getDbKeyList(args) |
||||
|
||||
|
||||
def set_kv(args): |
||||
t = nosqlRedisCtr() |
||||
return t.setKv(args) |
||||
|
||||
|
||||
def del_val(args): |
||||
t = nosqlRedisCtr() |
||||
return t.delVal(args) |
||||
|
||||
def batch_del_val(args): |
||||
t = nosqlRedisCtr() |
||||
return t.batchDelVal(args) |
||||
|
||||
def clear_flushdb(args): |
||||
t = nosqlRedisCtr() |
||||
return t.clearFlushDB(args) |
||||
|
||||
# 测试 |
||||
def test(args): |
||||
sid = args['sid'] |
||||
t = nosqlRedis() |
||||
print(t.get_options()) |
||||
print("test") |
||||
return 'ok' |
||||
|
||||
# ---------------------------------- run ---------------------------------- |
||||
|
@ -0,0 +1,30 @@ |
||||
.main-content .safe { |
||||
position: relative; |
||||
} |
||||
|
||||
.mask_layer { |
||||
position: absolute; |
||||
top: 0; |
||||
left: 0; |
||||
right: 0; |
||||
bottom: 0; |
||||
background: rgb(255 255 255 / 60%); |
||||
z-index: 9998; |
||||
} |
||||
|
||||
.mask_layer .prompt_description { |
||||
width: 350px; |
||||
height: 60px; |
||||
line-height: 60px; |
||||
text-align: center; |
||||
position: absolute; |
||||
left: 50%; |
||||
top: 50%; |
||||
margin-top: -25px; |
||||
margin-left: -175px; |
||||
background: #fff; |
||||
box-shadow: 1px 1px 50px rgb(0 0 0 / 30%); |
||||
border: 1px solid #eee; |
||||
border-radius: 2px; |
||||
font-size: 13px; |
||||
} |
@ -0,0 +1,11 @@ |
||||
|
||||
/* menu start */ |
||||
.menu .current .menu_plugin_data_query:hover { |
||||
background-image: url("/plugins/file?name=data_query&f=ico.png"); |
||||
} |
||||
|
||||
.menu .menu_plugin_data_query { |
||||
background-image: url("/plugins/file?name=data_query&f=ico.png"); |
||||
} |
||||
|
||||
/* menu end */ |
@ -0,0 +1,169 @@ |
||||
<div class="main-content" style="min-height: 389px;"> |
||||
<div class="container-fluid" style="padding-bottom: 50px;"> |
||||
<div id="cutTab" class="pos-box bgw mtb15" style="height:45px"> |
||||
<div class="tab-list"> |
||||
<!-- <div class="tabs-item active" data-name="mongodb">Mongodb</div> --> |
||||
<div class="tabs-item active" data-name="redis">Redis</div> |
||||
<!-- <div class="tabs-item" data-name="memcached">Memcached</div> --> |
||||
</div> |
||||
</div> |
||||
|
||||
|
||||
<div class="bgw mtb15 pd15 tab-view-box firewall-tab-view safe" style="overflow: hidden;"> |
||||
|
||||
<div class="mask_layer" style="display:none;"> |
||||
<div class="prompt_description">当前未安装本地服务器,<a class="btlink install_server" href="/soft">点击安装</a></div> |
||||
</div> |
||||
|
||||
<!-- mongodb start --> |
||||
<div class="tab-con hide" id="mongodb" style="padding: 0px;"> |
||||
<div id="operationLog" style="position: relative;"> |
||||
<div class="tootls_group tootls_top"> |
||||
<div class="pull-left"> |
||||
<button type="button" title="刷新日志" class="refresh btn btn-success btn-sm mr5"> |
||||
<span>刷新日志</span> |
||||
</button> |
||||
<button type="button" title="清空日志" class="clear btn btn-default btn-sm mr5"> |
||||
<span>清空日志</span> |
||||
</button> |
||||
</div> |
||||
</div> |
||||
<div class="divtable mtb10"> |
||||
<table class="table table-hover"> |
||||
<thead style="position: relative;z-index: 1;"><tr> |
||||
<th><span>编号</span></th> |
||||
<th><span>操作类型</span></th> |
||||
<th><span>详情</span></th> |
||||
<th><span>操作时间</span></th> |
||||
</tr></thead> |
||||
<tbody></tbody> |
||||
</table> |
||||
</div> |
||||
</div> |
||||
<div class="dataTables_paginate paging_bootstrap page operationLog"> |
||||
<div><span class="Pcurrent">1</span><span class="Pcount">共1条</span></div> |
||||
</div> |
||||
</div> |
||||
<!-- mongodb end --> |
||||
|
||||
<!-- reids start --> |
||||
<div class="tab-con show w-full" id="redis" style="padding: 0px;"> |
||||
<div id="redis_ctr" style="position: relative;"> |
||||
<div class="tootls_group tootls_top"> |
||||
<div class="pull-left"> |
||||
<button id="redis_add_key" type="button" class="refresh btn btn-success btn-sm mr5"> |
||||
<span>添加key</span> |
||||
</button> |
||||
<button id="redis_clear_all" type="button" class="clear btn btn-default btn-sm mr5" style="border-right: 1px solid #ccc;padding-right: 20px;"> |
||||
<span>清空数据库</span> |
||||
</button> |
||||
</div> |
||||
|
||||
</div> |
||||
|
||||
<div id="redis_list_tab" style="padding-top:10px;"> |
||||
<div class="tab-nav"> |
||||
<span data-id="0" class="on">DB0(0)</span> |
||||
</div> |
||||
</div> |
||||
|
||||
<div class="tootls_group tootls_top"> |
||||
<div class="pull-left"></div> |
||||
<div class="pull-right" style="padding-top: 5px;"> |
||||
<div class="search"> |
||||
<input id="redis_ksearch" type="text" class="search_input" placeholder="请输入键名称"> |
||||
<span id="redis_ksearch_span" class="glyphicon glyphicon-search" aria-hidden="true"></span> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
|
||||
<div class="divtable mtb10 redis_table_content"> |
||||
<table class="table table-hover"> |
||||
<thead style="position: relative;z-index: 1;"> |
||||
<tr> |
||||
<th width="30"><input type="checkbox" onclick="checkSelect();" id="setBox"></th> |
||||
<th><span>键</span></th> |
||||
<th><span>值</span></th> |
||||
<th><span>数据类型</span></th> |
||||
<th><span>数据长度</span></th> |
||||
<th><span>有效期</span></th> |
||||
<th style="text-align:right;"><span>操作</span></th> |
||||
</tr> |
||||
</thead> |
||||
<tbody></tbody> |
||||
</table> |
||||
</div> |
||||
</div> |
||||
|
||||
<div class="tootls_group tootls_bottom"> |
||||
<div class="pull-left"> |
||||
<button id="redis_batch_del" type="button" class="btn btn-sm set_batch_option bt-disabled btn-default">批量删除</button> |
||||
</div> |
||||
<div class="pull-right"> |
||||
<div class="page redis_list_page"> |
||||
<div><span class="Pcurrent">1</span><span class="Pcount">共1条</span></div> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
|
||||
</div> |
||||
<!-- reids end --> |
||||
|
||||
<!-- memcached start --> |
||||
<div class="tab-con hide" id="memcached" style="height: auto;"> |
||||
<div class="logAuditTabContent"> |
||||
<div class="logAuditTab" style="width: 300px;overflow: hidden;overflow-y: auto;"> |
||||
<!-- |
||||
<div class="logAuditItem active" title="wtmp - 登录和重启记录" data-file="/wtmp"> |
||||
wtmp - 登录和重启记录(4.88 KB) |
||||
</div> |
||||
--> |
||||
</div> |
||||
<div class="logAuditContent" style="overflow: hidden;overflow-y: auto;"> |
||||
<div id="logAuditTable" class="bt_table" style="display: block;"> |
||||
<div class="tootls_group tootls_top"> |
||||
<div class="pull-left"> |
||||
<button type="button" title="刷新列表" class="btn btn-success btn-sm mr5"><span>刷新列表</span></button> |
||||
</div> |
||||
<div class="pull-right"> |
||||
<div class="logs_search" style="position: relative;"> |
||||
<input type="text" class="search_input" style="" placeholder="请输入来源/端口/角色/事件"> |
||||
<span class="glyphicon glyphicon-search" aria-hidden="true"></span> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
<div class="divtable mtb10" style="max-height: 83px;"> |
||||
<table class="table table-hover"> |
||||
<thead style="position: relative;z-index: 1;"> |
||||
<tr> |
||||
<th><span data-index="0"><span>用户</span></span></th> |
||||
<th><span data-index="1"><span>来源</span></span></th> |
||||
<th><span data-index="2"><span>端口</span></span></th> |
||||
<th><span data-index="3"><span>时间</span></span></th> |
||||
</tr> |
||||
</thead> |
||||
<tbody> |
||||
<!-- <tr><td><span>root</span></td> |
||||
<td><span>117.139.193.29</span></td> |
||||
<td><span>pts/0</span></td> |
||||
<td><span>2023-08-25 13:27 still logged in</span></td> |
||||
</tr> --> |
||||
</tbody> |
||||
</table> |
||||
</div> |
||||
<div class="tootls_group tootls_bottom"> |
||||
<div class="pull-left"></div> |
||||
<div class="pull-right"> |
||||
<div class="page"><span class="Pcount">第 1 页</span></div> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
<div id="logAuditPages" class="page" style="display: flex; justify-content: flex-end;"></div> |
||||
<div id="logAuditPre" style="display: none;"><pre style="height: 188px; background-color: rgb(51, 51, 51); color: rgb(255, 255, 255); overflow-x: hidden; overflow-wrap: break-word; white-space: pre-wrap;"><code></code></pre></div> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
<!-- memcached end --> |
||||
</div> |
||||
</div> |
||||
</div> |
@ -0,0 +1,465 @@ |
||||
$(document).ready(function(){ |
||||
var tag = $.getUrlParam('tag'); |
||||
if(tag == 'data_query'){ |
||||
initDataQuery(); |
||||
} |
||||
}); |
||||
|
||||
function redisPostCB(method, args, callback){ |
||||
var loadT = layer.msg('正在获取...', { icon: 16, time: 0, shade: 0.3 }); |
||||
|
||||
var req_data = {}; |
||||
req_data['name'] = 'data_query'; |
||||
req_data['func'] = method; |
||||
req_data['script']='nosql_redis'; |
||||
args['version'] = ''; |
||||
|
||||
if (typeof(args) == 'string' && args == ''){ |
||||
req_data['args'] = JSON.stringify(toArrayObject(args)); |
||||
} else { |
||||
req_data['args'] = JSON.stringify(args); |
||||
} |
||||
|
||||
$.post('/plugins/callback', req_data, function(data) { |
||||
layer.close(loadT); |
||||
if (!data.status){ |
||||
layer.msg(data.msg,{icon:0,time:2000,shade: [0.3, '#000']}); |
||||
return; |
||||
} |
||||
|
||||
if(typeof(callback) == 'function'){ |
||||
callback(data); |
||||
} |
||||
},'json');
|
||||
} |
||||
|
||||
|
||||
function selectTab(tab = 'redis'){ |
||||
$('.tab-view-box .tab-con').addClass('hide').removeClass('show').removeClass('w-full'); |
||||
$('#'+tab).removeClass('hide').addClass('w-full'); |
||||
} |
||||
|
||||
function showInstallLayer(){ |
||||
$('.mask_layer').css('display','block'); |
||||
} |
||||
|
||||
function closeInstallLayer(){ |
||||
$('.mask_layer').css('display','none'); |
||||
} |
||||
|
||||
function initTabFunc(tab){ |
||||
switch(tab){ |
||||
case 'redis':initTabRedis();break; |
||||
} |
||||
} |
||||
|
||||
|
||||
|
||||
function initDataQuery(){ |
||||
var tab = $('#cutTab .tabs-item.active').data('name'); |
||||
initTabFunc(tab); |
||||
$('#cutTab .tabs-item').click(function(){ |
||||
var tab = $(this).data('name'); |
||||
$('#cutTab .tabs-item').removeClass('active'); |
||||
$(this).addClass('active'); |
||||
selectTab(tab); |
||||
initTabFunc(tab); |
||||
});
|
||||
} |
||||
|
||||
function initTabRedis(){ |
||||
//渲染数据
|
||||
redisGetList(); |
||||
|
||||
$('#redis_add_key').click(function(){ |
||||
redisAdd(); |
||||
}); |
||||
|
||||
//搜索
|
||||
$('#redis_ksearch').keyup(function(e){ |
||||
if (e.keyCode == 13){ |
||||
var val = $(this).val(); |
||||
if (val == ''){ |
||||
layer.msg('搜索不能为空!',{icon:7}); |
||||
return; |
||||
} |
||||
redisGetKeyList(1, val); |
||||
} |
||||
}); |
||||
|
||||
$('#redis_ksearch_span').click(function(){ |
||||
var val = $('#redis_ksearch').val(); |
||||
if (val == ''){ |
||||
layer.msg('搜索不能为空!',{icon:7}); |
||||
return; |
||||
} |
||||
redisGetKeyList(1, val); |
||||
}); |
||||
|
||||
//批量删除
|
||||
$('#redis_batch_del').click(function(){ |
||||
redisBatchDel(); |
||||
}); |
||||
|
||||
//清空所有
|
||||
$('#redis_clear_all').click(function(){ |
||||
redisBatchClear(); |
||||
}); |
||||
|
||||
readerTableChecked(); |
||||
} |
||||
|
||||
function redisGetSid(){ |
||||
return 0; |
||||
} |
||||
|
||||
function redisGetIdx(){ |
||||
return $('#redis_list_tab .tab-nav span.on').data('id'); |
||||
} |
||||
|
||||
function redisGetList(){ |
||||
var sid = redisGetSid(); |
||||
redisPostCB('get_list',{'sid':sid} ,function(rdata){ |
||||
if (rdata.data.status){ |
||||
var list = rdata.data.data; |
||||
var content = ''; |
||||
for (var i = 0; i < list.length; i++) { |
||||
if (i == 0){ |
||||
content += '<span data-id="'+i+'" class="on">'+list[i]['name'] + '('+ list[i]['keynum'] +')</span>';
|
||||
} else { |
||||
content += '<span data-id="'+i+'">'+list[i]['name'] + '('+ list[i]['keynum'] +')</span>';
|
||||
} |
||||
} |
||||
$('#redis_list_tab .tab-nav').html(content); |
||||
|
||||
$('#redis_list_tab .tab-nav span').click(function(){ |
||||
$('#redis_list_tab .tab-nav span').removeClass('on'); |
||||
$(this).addClass('on'); |
||||
redisGetKeyList(1); |
||||
}); |
||||
redisGetKeyList(1); |
||||
} else { |
||||
showInstallLayer(); |
||||
} |
||||
}); |
||||
} |
||||
|
||||
function redisGetKeyList(page,search = ''){ |
||||
|
||||
var args = {}; |
||||
args['sid'] = redisGetSid(); |
||||
args['idx'] = redisGetIdx(); |
||||
args['p'] = page; |
||||
args['search'] = search; |
||||
|
||||
var input_search_val = $('#redis_ksearch').val(); |
||||
if (input_search_val!=''){ |
||||
args['search'] = input_search_val; |
||||
} |
||||
|
||||
redisPostCB('get_dbkey_list', args, function(rdata){ |
||||
if (rdata.data.status){ |
||||
var data = rdata.data.data.data; |
||||
var tbody = ''; |
||||
for (var i = 0; i < data.length; i++) { |
||||
|
||||
|
||||
tbody += '<tr>'; |
||||
tbody += "<td><input type='checkbox' class='check' name='id' title='"+data[i].name+"' onclick='checkSelect();' value='"+data[i].name+"'></td>"; |
||||
tbody += '<td style="width:100px;">'+data[i].name+'</td>'; |
||||
tbody += '<td><span style="width:100px;" class="size_ellipsis">'+data[i].val+'</span><span data-index="'+i+'" class="ico-copy cursor copy ml5" title="复制值"></span></td>'; |
||||
tbody += '<td>'+data[i].type+'</td>'; |
||||
tbody += '<td>'+data[i].len+'</td>'; |
||||
|
||||
|
||||
if (data[i].endtime == -1){ |
||||
tbody += '<td>永久</td>'; |
||||
} else { |
||||
tbody += '<td>'+data[i].endtime+'</td>'; |
||||
} |
||||
|
||||
|
||||
tbody += '<td style="text-align:right; color:#bbb">\ |
||||
<a href="javascript:;" data-index="'+i+'" class="btlink edit" title="编辑">编辑</a> | \ |
||||
<a href="javascript:;" class="btlink" onclick="redisDeleteKey(\''+data[i].name+'\')">删除</a>\ |
||||
</td>'; |
||||
|
||||
tbody += '</tr>'; |
||||
} |
||||
|
||||
// console.log(tbody);
|
||||
$('.redis_table_content tbody').html(tbody); |
||||
$('.redis_list_page').html(rdata.data.data.page); |
||||
|
||||
|
||||
$('.edit').click(function(){ |
||||
var i = $(this).data('index'); |
||||
redisEditKv(data[i].name,data[i].val,data[i].endtime); |
||||
}); |
||||
|
||||
$('.copy').click(function(){ |
||||
var i = $(this).data('index'); |
||||
copyText(data[i].val); |
||||
}); |
||||
|
||||
} |
||||
}); |
||||
} |
||||
|
||||
function redisDeleteKey(name){ |
||||
layer.confirm('确定要删除?', {btn: ['确定', '取消']}, function(){ |
||||
var data = {}; |
||||
data['idx'] = redisGetIdx(); |
||||
data['sid'] = redisGetSid(); |
||||
data['name'] = name; |
||||
redisPostCB('del_val', data, function(rdata){ |
||||
if (rdata.data.status){ |
||||
showMsg(rdata.data.msg,function(){ |
||||
redisGetList(); |
||||
},{icon: rdata.data.status ? 1 : 2}, 2000); |
||||
} |
||||
}); |
||||
}); |
||||
} |
||||
|
||||
function redisAdd(){ |
||||
layer.open({ |
||||
type: 1, |
||||
area: '480px', |
||||
title: '添加Key至服务器', |
||||
closeBtn: 1, |
||||
shift: 0, |
||||
shadeClose: false, |
||||
btn:['确定','取消'], |
||||
content: "<form class='bt-form pd20'>\ |
||||
<div class='line'>\ |
||||
<span class='tname'>数据库</span>\ |
||||
<div class='info-r c4'>\ |
||||
<select name='idx' class='bt-input-text' style='width:260px;'>\ |
||||
<option value='0'>DB(0)</option>\ |
||||
</select>\ |
||||
</div>\ |
||||
</div>\ |
||||
<div class='line'>\ |
||||
<span class='tname'>键</span>\ |
||||
<div class='info-r c4'>\ |
||||
<input class='bt-input-text' type='text' name='key' placeholder='键' style='width:260px;'/>\ |
||||
</div>\ |
||||
</div>\ |
||||
<div class='line'>\ |
||||
<span class='tname'>值</span>\ |
||||
<div class='info-r c4'>\ |
||||
<textarea class='bt-input-text' name='val' style='width:260px;height:100px;'/></textarea>\ |
||||
</div>\ |
||||
</div>\ |
||||
<div class='line'>\ |
||||
<span class='tname'>有效期</span>\ |
||||
<div class='info-r c4'>\ |
||||
<input class='bt-input-text mr5' type='number' name='endtime' value='60' style='width:260px;'/>\ |
||||
</div>\ |
||||
</div>\ |
||||
<div class='line'>\ |
||||
<div>\ |
||||
<ul class='help-info-text c7' style='margin-left:30px;'><li>有效期为0表示永久</li>\ |
||||
</div>\ |
||||
</div>\ |
||||
</form>", |
||||
success:function(){ |
||||
var db_list = $('#redis_list_tab .tab-nav span'); |
||||
var db_list_count = db_list.length; |
||||
|
||||
var idx_html = ''; |
||||
for (var i = 0; i < db_list_count; i++) { |
||||
idx_html += "<option value='"+i+"'>DB("+i+")</option>"; |
||||
} |
||||
$('select[name=idx]').html(idx_html); |
||||
}, |
||||
yes: function(index){ |
||||
var data = {}; |
||||
data['idx'] = $('select[name=idx]').val(); |
||||
data['sid'] = redisGetSid(); |
||||
data['name'] = $('input[name="key"]').val(); |
||||
data['val'] = $('textarea[name="val"]').val(); |
||||
data['endtime'] = $('input[name="endtime"]').val(); |
||||
|
||||
redisPostCB('set_kv', data ,function(rdata){ |
||||
if (rdata.data.status){ |
||||
showMsg(rdata.data.msg,function(){ |
||||
layer.close(index); |
||||
redisGetList(); |
||||
},{icon: rdata.data.status ? 1 : 2}, 1000); |
||||
} |
||||
}); |
||||
} |
||||
}); |
||||
} |
||||
|
||||
function redisEditKv(name, val, endtime){ |
||||
layer.open({ |
||||
type: 1, |
||||
area: '480px', |
||||
title: '编辑['+name+']Key', |
||||
closeBtn: 1, |
||||
shift: 0, |
||||
shadeClose: false, |
||||
btn:['确定','取消'], |
||||
content: "<form class='bt-form pd20'>\ |
||||
<div class='line'>\ |
||||
<span class='tname'>数据库</span>\ |
||||
<div class='info-r c4'>\ |
||||
<select name='idx' class='bt-input-text' style='width:260px;'>\ |
||||
<option value='0'>DB(0)</option>\ |
||||
</select>\ |
||||
</div>\ |
||||
</div>\ |
||||
<div class='line'>\ |
||||
<span class='tname'>键</span>\ |
||||
<div class='info-r c4'>\ |
||||
<input class='bt-input-text' type='text' name='key' placeholder='键' style='width:260px;'/>\ |
||||
</div>\ |
||||
</div>\ |
||||
<div class='line'>\ |
||||
<span class='tname'>值</span>\ |
||||
<div class='info-r c4'>\ |
||||
<textarea class='bt-input-text' name='val' style='width:260px;height:100px;'/></textarea>\ |
||||
</div>\ |
||||
</div>\ |
||||
<div class='line'>\ |
||||
<span class='tname'>有效期</span>\ |
||||
<div class='info-r c4'>\ |
||||
<input class='bt-input-text mr5' type='number' name='endtime' value='60' style='width:260px;'/>\ |
||||
</div>\ |
||||
</div>\ |
||||
<div class='line'>\ |
||||
<div>\ |
||||
<ul class='help-info-text c7' style='margin-left:30px;'><li>有效期为0表示永久</li>\ |
||||
</div>\ |
||||
</div>\ |
||||
</form>", |
||||
success:function(){ |
||||
var idx = redisGetIdx(); |
||||
var idx_html = "<option value='"+idx+"'>DB("+idx+")</option>"; |
||||
$('select[name=idx]').html(idx_html).attr('readonly','readonly'); |
||||
$('input[name="key"]').val(name).attr('readonly','readonly'); |
||||
$('textarea[name="val"]').val(val); |
||||
|
||||
if (endtime == -1){ |
||||
$('input[name="endtime"]').val(0); |
||||
} else { |
||||
$('input[name="endtime"]').val(endtime); |
||||
}
|
||||
}, |
||||
yes: function(index){ |
||||
var data = {}; |
||||
data['idx'] = $('select[name=idx]').val(); |
||||
data['sid'] = redisGetSid(); |
||||
data['name'] = $('input[name="key"]').val(); |
||||
data['val'] = $('textarea[name="val"]').val(); |
||||
data['endtime'] = $('input[name="endtime"]').val(); |
||||
|
||||
redisPostCB('set_kv', data ,function(rdata){ |
||||
if (rdata.data.status){ |
||||
showMsg(rdata.data.msg,function(){ |
||||
layer.close(index); |
||||
redisGetList(); |
||||
},{icon: rdata.data.status ? 1 : 2}, 1000); |
||||
} |
||||
}); |
||||
} |
||||
}); |
||||
} |
||||
|
||||
function redisBatchDel(){ |
||||
var keys = []; |
||||
$('input[type="checkbox"].check:checked').each(function () { |
||||
keys.push($(this).val()); |
||||
}); |
||||
if (keys.length == 0){ |
||||
layer.msg('没有选中数据!',{icon:7}); |
||||
return; |
||||
}
|
||||
|
||||
layer.confirm('确定要批量删除?', {btn: ['确定', '取消']}, function(){ |
||||
var data = {}; |
||||
data['idx'] = redisGetIdx(); |
||||
data['sid'] = redisGetSid(); |
||||
data['keys'] = keys; |
||||
redisPostCB('batch_del_val', data, function(rdata){ |
||||
if (rdata.data.status){ |
||||
showMsg(rdata.data.msg,function(){ |
||||
if (rdata.data.status){ |
||||
redisGetList();
|
||||
} |
||||
},{icon: rdata.data.status ? 1 : 2}, 2000); |
||||
} |
||||
}); |
||||
}); |
||||
} |
||||
|
||||
function redisBatchClear(){ |
||||
var xm_db_list; |
||||
layer.open({ |
||||
type: 1, |
||||
area: ['480px','180px'], |
||||
title: '清空【本地服务器】数据库', |
||||
closeBtn: 1, |
||||
shift: 0, |
||||
shadeClose: false, |
||||
btn:['确定','取消'], |
||||
content: "<form class='bt-form pd20'>\ |
||||
<div class='line'>\ |
||||
<span class='tname'>选择数据库</span>\ |
||||
<div class='info-r'>\ |
||||
<div id='select_db'></div>\ |
||||
</div>\ |
||||
</div>\ |
||||
</form>", |
||||
success:function(l,i){ |
||||
var db_list = $('#redis_list_tab .tab-nav span'); |
||||
var db_list_count = db_list.length; |
||||
|
||||
var idx_db = []; |
||||
for (var i = 0; i < db_list_count; i++) { |
||||
var t = {}; |
||||
t['name'] = "DB("+i+")"; |
||||
t['value'] = i; |
||||
idx_db.push(t); |
||||
} |
||||
|
||||
xm_db_list = xmSelect.render({ |
||||
el: '#select_db',
|
||||
repeat: true, |
||||
toolbar: {show: true}, |
||||
data: idx_db, |
||||
}); |
||||
|
||||
$(l).find('.layui-layer-content').css('overflow','visible'); |
||||
}, |
||||
yes: function(index){ |
||||
var xm_db_val = xm_db_list.getValue('value'); |
||||
layer.confirm('确定要批量清空?', {btn: ['确定', '取消']}, function(){ |
||||
var data = {}; |
||||
data['sid'] = redisGetSid(); |
||||
data['idxs'] = xm_db_val; |
||||
redisPostCB('clear_flushdb', data, function(rdata){ |
||||
showMsg(rdata.data.msg,function(){ |
||||
if (rdata.data.status){ |
||||
redisGetList(); |
||||
layer.close(index); |
||||
} |
||||
},{icon: rdata.data.status ? 1 : 2}, 2000); |
||||
}); |
||||
}); |
||||
} |
||||
}); |
||||
} |
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
Loading…
Reference in new issue