2024-01-08 22:50:47 +08:00

605 lines
21 KiB
Python
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# File : admin.py
# Author: DaShenHan&道长-----先苦后甜,任凭晚风拂柳颜------
# Date : 2022/9/6
import os
import ujson
from flask import Blueprint, abort, request, render_template, render_template_string, jsonify, make_response, redirect
from controllers.service import storage_service, rules_service, parse_service
from base.R import R
from base.database import db
from utils.log import logger
import shutil
from utils.update import zipfile, getLocalVer, getOnlineVer, download_new_version, download_lives, copy_to_update
from utils import parser
from utils.env import get_env, update_env
from utils.web import getParmas, verfy_token
from js.rules import getRules, getCacheCount
from utils.parser import runJScode
from werkzeug.utils import secure_filename
from utils.web import md5
from utils.common_api import js_render
from utils.files import get_jar_list, get_jsd_list, get_drop_js
from quickjs import Function, Context
admin = Blueprint("admin", __name__)
# @admin.route("/",methods=['get'])
# def index():
# return R.ok(msg='欢迎进入首页',data=None)
# @admin.route("/info",methods=['get'])
# def info_all():
# data = storage_service.query_all()
# return R.ok(data=data)
@admin.route('/')
def admin_index(): # 管理员界面
if not verfy_token():
return render_template('login.html')
lsg = storage_service()
live_url = lsg.getItem('LIVE_URL')
use_py = lsg.getItem('USE_PY')
force_up = lsg.getItem('FORCE_UP')
js0_password = lsg.getItem('JS0_PASSWORD')
# print(f'live_url:', live_url)
rules = getRules('js')
# print(rules)
cache_count = getCacheCount()
# print(cache_count)
return render_template('admin.html', js0_password=js0_password, pystate=use_py,force_up=force_up, rules=rules,
cache_count=cache_count, ver=getLocalVer(), live_url=live_url)
@admin.route('/settings')
def admin_settings(): # 管理员界面
if not verfy_token():
return render_template('login.html')
lsg = storage_service()
# conf_list = 'LIVE_URL|USE_PY|PLAY_URL|PLAY_DISABLE|LAZYPARSE_MODE|WALL_PAPER_ENABLE|WALL_PAPER|UNAME|PWD|LIVE_MODE|LIVE_URL|CATE_EXCLUDE|TAB_EXCLUDE'.split('|')
conf_lists = lsg.getStoreConf()
# print(conf_lists)
jar_lists = get_jar_list()
SPIDER_JAR = lsg.getItem('SPIDER_JAR', 'custom_spider.jar')
ZB_PLAYER = lsg.getItem('ZB_PLAYER', '1')
# print('ZB_PLAYER:',ZB_PLAYER)
return render_template('settings.html', conf_lists=conf_lists, jar_lists=jar_lists, jar_now=SPIDER_JAR,player_now=ZB_PLAYER,
ver=getLocalVer())
@admin.route('/save_conf', methods=['POST'])
def admin_save_conf(): # 管理员界面
if not verfy_token():
# return render_template('login.html')
return R.error('请登录后再试')
key = getParmas('key')
value = getParmas('value')
print(f'key:{key},value:{value}')
lsg = storage_service()
res_id = lsg.setItem(key, value)
return R.success(f'修改成功,记录ID为:{res_id}')
@admin.route('/update_env', methods=['POST'])
def admin_update_env(): # 更新环境变量中的某个值
if not verfy_token():
# return render_template('login.html')
return R.error('请登录后再试')
key = getParmas('key')
value = getParmas('value')
print(f'key:{key},value:{value}')
ENV = update_env(key, value)
return R.success(f'修改成功,最新的完整ENV见data', data=ENV)
@admin.route("/edit/<name>", methods=['GET'])
def admin_edit_rule(name):
# print(name)
if not verfy_token():
return render_template('login.html')
return render_template('edit_rule.html', name=name)
@admin.route("/edit2/<name>", methods=['GET'])
def admin_edit2_rule(name):
# print(name)
if not verfy_token():
return render_template('login.html')
return render_template('edit_rule_mobile.html', name=name)
@admin.route("/save_edit/<name>", methods=['POST'])
def admin_save_edit_rule(name):
# print(name)
if not verfy_token():
return R.error('请登录后再试')
code = getParmas('code')
file_path = os.path.abspath(f'js/{name}')
if 'var rule' not in code and name != '模板.js':
return R.error(f'文件{name}保存失败,未检测到关键词:var rule')
if not os.path.exists(file_path):
return R.error('服务端没有此文件!' + file_path)
logger.info(f'待保存文件路径:{file_path}')
with open(file_path, mode='w+', encoding='utf-8') as f:
f.write(code)
return R.success(f'保存成功')
@admin.route("/view/<name>", methods=['GET'])
def admin_view_rule(name):
return js_render(name)
# if not name or not name.split('.')[-1] in ['js','txt','py','json']:
# return R.error(f'非法猥亵,未指定文件名。必须包含js|txt|json|py')
# try:
# env = get_env()
# # print(env)
# if env.get('js_proxy'):
# js_proxy = env['js_proxy']
# burl = request.base_url
# if '=>' in js_proxy:
# oldsrc = js_proxy.split('=>')[0]
# if oldsrc in burl:
# newsrc = js_proxy.split('=>')[1]
# # print(f'js1源代理已启用,全局替换{oldsrc}为{newsrc}')
# rurl = burl.replace(oldsrc, newsrc)
# if burl != rurl:
# jscode = parser.getJs(name, 'js')
# # rjscode = render_template_string(jscode, env=env)
# rjscode = jscode
# for k in env:
# # print(f'${k}', f'{env[k]}')
# if f'${k}' in rjscode:
# rjscode = rjscode.replace(f'${k}', f'{env[k]}')
# # rjscode = render_template_string(jscode, **env)
# if rjscode.strip() == jscode.strip(): # 无需渲染才代理
# return redirect(rurl)
# else:
# logger.info(f'{name}由于存在环境变量无法被依赖代理')
#
# return parser.toJs(name,'js',env)
# except Exception as e:
# return R.error(f'非法猥亵\n{e}')
@admin.route('/clear/<name>')
def admin_clear_rule(name):
if not name or not name.split('.')[-1] in ['js', 'txt', 'py', 'json']:
return R.error(f'非法猥亵,未指定文件名。必须包含js|txt|json|py')
if not verfy_token():
return render_template('login.html')
file_path = os.path.abspath(f'js/{name}')
print(file_path)
if not os.path.exists(file_path):
return R.error('服务端没有此文件!' + file_path)
os.remove(file_path)
return R.ok('成功删除文件:' + file_path)
@admin.route('/get_ver')
def admin_get_ver():
if not verfy_token():
# return render_template('login.html')
return R.error('请登录后再试')
lsg = storage_service()
update_proxy = lsg.getItem('UPDATE_PROXY')
online_ver, msg = getOnlineVer(update_proxy)
return jsonify({'local_ver': getLocalVer(), 'online_ver': online_ver, 'msg': msg})
@admin.route('/update_db')
def admin_update_db():
if not verfy_token():
# return render_template('login.html')
return R.error('请登录后再试')
old_dbfile = 'migrations'
if os.path.exists(old_dbfile):
logger.info(f'开始删除历史数据库迁移文件:{old_dbfile}')
shutil.rmtree(old_dbfile)
db.session.execute('drop table if exists alembic_version')
cmd = 'flask db migrate && flask db upgrade'
if not os.path.exists('migrations'):
cmd = 'flask db init && ' + cmd
logger.info(f'开始执行cmd:{cmd}')
result = os.system(cmd)
logger.info(f'cmd执行结果:{result}')
return R.success('数据库升级完毕')
@admin.route('/update_ver')
def admin_update_ver():
if not verfy_token():
return R.failed('请登录后再试')
lsg = storage_service()
update_proxy = lsg.getItem('UPDATE_PROXY')
force_up = lsg.getItem('FORCE_UP')
msg = download_new_version(update_proxy,force_up)
return R.success(msg)
@admin.route('/rule_state/<int:state>', methods=['POST'])
def admin_rule_state(state=0): # 管理员修改规则状态
if not verfy_token():
return R.error('请登录后再试')
names = getParmas('names')
if not names:
return R.success(f'修改失败,没有传递names参数')
rule_list = names.split(',')
rules = rules_service()
# print(rules.query_all())
# print(rules.getState(rule_list[0]))
# print(rule_list)
success_list = []
for rule in rule_list:
try:
res_id = rules.setState(rule, state)
success_list.append(f'{rule}:{res_id}')
except:
success_list.append(rule)
return R.success(f'修改成功,服务器反馈信息为:{success_list}')
@admin.route('/rule_order/<int:order>', methods=['POST'])
def admin_rule_order(order=0): # 管理员修改规则顺序
if not verfy_token():
return R.error('请登录后再试')
names = getParmas('names')
if not names:
return R.success(f'修改失败,没有传递names参数')
rule_list = names.split(',')
rules = rules_service()
# print(rules.query_all())
# print(rules.getState(rule_list[0]))
# print(rule_list)
success_list = []
rule_list.reverse() # 倒序解决时间多重排序问题
for rule in rule_list:
try:
res_id = rules.setOrder(rule, order)
success_list.append(f'{rule}:{res_id}')
except:
success_list.append(rule)
return R.success(f'修改成功,服务器反馈信息为:{success_list}')
@admin.route('/parse/save_data', methods=['POST'])
def admin_parse_save_data(): # 管理员保存拖拽排序后的解析数据
if not verfy_token():
return R.error('请登录后再试')
data = getParmas('data')
if not data:
return R.success(f'修改失败,没有传递data参数')
parse = parse_service()
success_list = []
data = ujson.loads(data)
new_list = []
new_data = []
for nd in data:
if not nd.get('url') and nd.get('name') != '🌐Ⓤ':
continue
if nd['url'] not in new_list:
new_data.append(nd)
new_list.append(nd['url'])
print(f'去重前:{len(data)},去重后:{len(new_data)}')
for i in range(len(new_data)):
d = new_data[i]
# if not d.get('url') and d.get('name') != '🌐Ⓤ':
# continue
obj = {
'name': d.get('name', ''),
'url': d.get('url', ''),
'state': d.get('state', 1),
'type': d.get('state', 0),
'order': i + 1,
'ext': d.get('ext', ''),
'header': d.get('header', ''),
}
# print(obj)
try:
parse.saveData(obj)
success_list.append(f'parse:{d["url"]}')
# print(obj)
# print(200,obj)
except Exception as e:
success_list.append(d["url"])
print(f'{d["url"]}失败:{e}')
# print(len(success_list))
return R.success(f'修改成功,服务器反馈信息为:{success_list}')
@admin.route('/force_update')
def admin_force_update():
if not verfy_token():
return R.failed('请登录后再试')
ret = copy_to_update()
if ret:
msg = '升级成功'
return R.success(msg)
else:
msg = '升级失败。具体原因只能去看实时日志(通过9001端口)'
return R.failed(msg)
@admin.route('/update_lives')
def admin_update_lives():
url = getParmas('url')
if not url:
return R.failed('未提供被同步的直播源远程地址!')
if not verfy_token():
return R.failed('请登录后再试')
live_url = url
success = download_lives(live_url)
if success:
return R.success(f'直播源{live_url}同步成功')
else:
return R.failed(f'直播源{live_url}同步失败')
@admin.route('/write_live_url')
def admin_write_live_url():
url = getParmas('url')
if not url:
return R.failed('未提供修改后的直播源地址!')
if not verfy_token():
return R.failed('请登录后再试')
lsg = storage_service()
id = lsg.setItem('LIVE_URL', url)
msg = f'已修改的配置记录id为:{id}'
return R.success(msg)
@admin.route('/change_use_py')
def admin_change_use_py():
if not verfy_token():
return R.failed('请登录后再试')
lsg = storage_service()
use_py = lsg.getItem('USE_PY')
new_use_py = '' if use_py else '1'
state = '开启' if new_use_py else '关闭'
id = lsg.setItem('USE_PY', new_use_py)
msg = f'已修改的配置记录id为:{id},结果为{state}'
return R.success(msg)
@admin.route('/change_force_up')
def admin_change_force_up():
if not verfy_token():
return R.failed('请登录后再试')
lsg = storage_service()
force_up = lsg.getItem('FORCE_UP')
new_force_up = '' if force_up else '1'
state = '开启' if new_force_up else '关闭'
id = lsg.setItem('FORCE_UP', new_force_up)
msg = f'已修改的配置记录id为:{id},结果为{state}'
return R.success(msg)
@admin.route('/clear_drop')
def admin_clear_drop():
if not verfy_token():
return R.failed('请登录后再试')
jsd_list = get_jsd_list()
logger.info(f'jsd文件列表:{jsd_list}')
js_list = get_drop_js(jsd_list)
rm_list = []
for i in range(len(js_list)):
js_file = js_list[i]
# shutil.rmtree(js_file, ignore_errors=False, onerror=None)
if os.path.exists(js_file):
os.remove(js_file)
rm_list.append(jsd_list[i][:-1])
logger.info(f'待删除js文件列表:{rm_list}')
rm_str = ','.join(rm_list)
msg = f'清理完毕,本次共计清理{len(rm_list)}\n {rm_str}'
return R.success(msg)
# @admin.route('/get_use_py')
# def admin_get_use_py():
# if not verfy_token():
# return R.failed('请登录后再试')
# lsg = storage_service()
# use_py = lsg.getItem('USE_PY')
# state = 1 if use_py else 0
# return R.success(state)
def get_size(fobj):
if fobj.content_length:
return fobj.content_length
try:
pos = fobj.tell()
fobj.seek(0, 2) # seek to end
size = fobj.tell()
fobj.seek(pos) # back to original position
return size
except (AttributeError, IOError):
pass
# in-memory file object that doesn't support seeking or tell
return 0
@admin.route('/upload', methods=['POST'])
def upload_file():
args = request.args
force = args.get('force')
if not verfy_token():
return render_template('login.html')
if request.method == 'POST':
try:
file = request.files['file']
lsg = storage_service()
js_max_len = lsg.getItem('JS_MAX_LENGTH', 0.1 * 1024 * 1024)
if get_size(file) > float(js_max_len):
logger.info(f'文件体积过大,禁止上传。当前体积:{get_size(file)},源体积限制:{js_max_len}')
abort(413) # request entity too large
filename = secure_filename(file.filename)
logger.info(f'推荐安全文件命名:{filename}')
savePath = f'js/{file.filename}'
# print(savePath)
if os.path.exists(savePath) and not force:
return R.failed(f'上传失败,文件已存在,请先查看删除再试')
with open('js/模板.js', encoding='utf-8') as f2:
before = f2.read().split('export')[0]
end_code = """\nif (rule.模板 && muban.hasOwnProperty(rule.模板)) {rule = Object.assign(muban[rule.模板], rule);}"""
upcode = file.stream.read().decode('utf-8')
check_to_run = before + upcode + end_code
# print(check_to_run)
# try:
# loader, _ = runJScode(check_to_run)
# rule = loader.eval('rule')
# if not rule:
# return R.failed('文件上传失败,检测到上传的文件不是drpy框架支持的源代码')
# except Exception as e:
# logger.info(f'上传文件发生了错误:{e}')
# return R.failed('文件上传失败,检测到上传的文件不是drpy框架支持的源代码')
try:
ctx = Context()
ctx.eval(check_to_run)
js_ret = ctx.get('rule')
rule_json = js_ret.json() # 规则的json字符串
ruleDict = ujson.loads(rule_json)
if not ruleDict:
return R.failed('文件上传失败,检测到上传的文件不是drpy框架支持的源代码')
except Exception as e:
logger.info(f'上传文件发生了错误:{e}')
return R.failed('文件上传失败,检测到上传的文件不是drpy框架支持的源代码')
# print(savePath)
file.seek(0) # 读取后变成空文件,重新赋能
file.save(savePath)
return R.success('文件上传成功')
except Exception as e:
return R.failed(f'文件上传失败!{e}')
else:
# return render_template('upload.html')
return R.failed('文件上传失败')
@admin.route('/upload_update', methods=['POST'])
def upload_update():
args = request.args
force = args.get('force')
print('force:', force)
if not verfy_token():
return render_template('login.html')
if request.method == 'POST':
try:
file = request.files['file']
filename = secure_filename(file.filename)
logger.info(f'推荐安全文件命名:{filename}')
savePath = f'tmp/dr_py.zip'
file.seek(0) # 读取后变成空文件,重新赋能
file.save(savePath)
logger.info(f'开始解压文件:{savePath}')
f = zipfile.ZipFile(savePath, 'r') # 压缩文件位置
for file in f.namelist():
f.extract(file, 'tmp') # 解压位置
f.close()
# print('解压完毕,开始升级')
logger.info('解压完毕,开始升级')
# ret = copy_to_update()
return R.success('升级文件上传成功,请确认drpy目录内是否存在/tmp/dr_py-main/文件夹如果ok你可以点击强制升级按钮升级刚才上传的文件')
except Exception as e:
return R.failed(f'升级文件上传失败!{e}')
@admin.route('/login', methods=['GET', 'POST'])
def login_api():
username = getParmas('username')
password = getParmas('password')
autologin = getParmas('autologin')
if not all([username, password]):
return R.failed('账号密码字段必填')
token = md5(f'{username};{password}')
check = verfy_token(token=token)
if check:
# response = make_response(redirect('/admin'))
response = make_response(R.success('登录成功'))
response.set_cookie('token', token)
return response
else:
return R.failed('登录失败,用户名或密码错误')
@admin.route('/logtail')
def admin_logtail():
if not verfy_token():
return R.failed('请登录后再试')
return render_template('logtail.html')
@admin.route('/lives')
def admin_lives():
if not verfy_token():
return R.failed('请登录后再试')
# print(dir(request))
# 完整地址: request.base_url url
# 带http的前缀 host_url root_url
# 不带http的前缀 host
# 当前路径 path
host_url = request.host_url
def get_lives():
base_path = os.path.dirname(os.path.abspath(__file__)) # 当前文件所在目录
# print(base_path)
live_path = os.path.join(base_path, '../txt/lives')
# print(live_path)
files = os.listdir(live_path)
# print(files)
# files = list(filter(lambda x: str(x).endswith('.txt') and str(x).find('模板') < 0, files))
files = list(
filter(lambda x: str(x).split('.')[-1] in ['txt', 'json', 'm3u'] and str(x).find('模板') < 0, files))
files = [f'{host_url}lives?path=txt/lives/{file}' for file in files]
return files
files = '\n'.join(get_lives())
response = make_response(files)
response.headers['Content-Type'] = 'text/plain; charset=utf-8'
return response
@admin.route('/lives_web')
def admin_lives_web():
if not verfy_token():
return R.failed('请登录后再试')
# host_url = request.host_url
def get_lives():
base_path = os.path.dirname(os.path.abspath(os.path.dirname(__file__))) # 上级目录
live_path = os.path.join(base_path, f'base/直播.txt')
with open(live_path,encoding='utf-8') as f:
text = f.read()
return text
text = get_lives()
# response = make_response(text)
# response.headers['Content-Type'] = 'text/plain; charset=utf-8'
# return response
lives = []
for line in text.split('\n'):
if ',http' in line:
lives.append({
'title':line.split(',')[0],
'url':line.split(',')[1],
})
print(lives)
lsg = storage_service()
zb_player = lsg.getItem('ZB_PLAYER','1')
return render_template('lives.html',ver=getLocalVer(),lives=lives,zb_player=zb_player)
@admin.route('/tools')
def admin_tools():
if not verfy_token():
return R.failed('请登录后再试')
return render_template('tools.html', ver=getLocalVer())