566 lines
25 KiB
Python
566 lines
25 KiB
Python
|
|
from django.shortcuts import render, redirect, get_object_or_404
|
|||
|
|
import json
|
|||
|
|
from django.contrib.auth.decorators import login_required
|
|||
|
|
from django.contrib import messages
|
|||
|
|
from django.urls import reverse
|
|||
|
|
from django.db import transaction
|
|||
|
|
from django.utils import timezone
|
|||
|
|
|
|||
|
|
from .models import Domain, DomainTrafficDaily
|
|||
|
|
from .forms import AddDomainForm, DomainSettingsForm
|
|||
|
|
from core.goedge_client import GoEdgeClient
|
|||
|
|
from core.models import SystemSettings, OperationLog
|
|||
|
|
from plans.models import Plan
|
|||
|
|
from core.utils import check_cname_map, bytes_to_gb
|
|||
|
|
|
|||
|
|
|
|||
|
|
@login_required
|
|||
|
|
def list_domains(request):
|
|||
|
|
domains = Domain.objects.filter(user=request.user).order_by('-created_at')
|
|||
|
|
# 计算当月用量与配额
|
|||
|
|
today = timezone.now().date()
|
|||
|
|
month_start = today.replace(day=1)
|
|||
|
|
sys = SystemSettings.objects.order_by('id').first()
|
|||
|
|
for d in domains:
|
|||
|
|
traffic_qs = DomainTrafficDaily.objects.filter(domain=d, day__gte=month_start)
|
|||
|
|
total_bytes = sum(t.bytes for t in traffic_qs)
|
|||
|
|
used_gb = bytes_to_gb(total_bytes)
|
|||
|
|
base_quota_gb = 0
|
|||
|
|
if d.current_plan and d.current_plan.included_traffic_gb_per_domain:
|
|||
|
|
base_quota_gb = d.current_plan.included_traffic_gb_per_domain
|
|||
|
|
elif sys:
|
|||
|
|
base_quota_gb = sys.default_free_traffic_gb_per_domain
|
|||
|
|
total_quota_gb = (base_quota_gb or 0) + (d.extra_free_traffic_gb_current_cycle or 0)
|
|||
|
|
# 动态属性用于模板展示
|
|||
|
|
d.used_gb = used_gb
|
|||
|
|
d.total_quota_gb = total_quota_gb
|
|||
|
|
return render(request, 'domains/list.html', {
|
|||
|
|
'domains': domains,
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
|
|||
|
|
@login_required
|
|||
|
|
@transaction.atomic
|
|||
|
|
def add_domain(request):
|
|||
|
|
if request.method == 'POST':
|
|||
|
|
form = AddDomainForm(request.POST)
|
|||
|
|
if form.is_valid():
|
|||
|
|
name = form.cleaned_data['name'].strip()
|
|||
|
|
subdomains = form.cleaned_data['subdomains']
|
|||
|
|
origin_host = form.cleaned_data['origin_host'].strip()
|
|||
|
|
origin_protocol = form.cleaned_data['origin_protocol']
|
|||
|
|
origin_port = form.cleaned_data['origin_port']
|
|||
|
|
plan = form.cleaned_data.get('plan')
|
|||
|
|
enable_websocket = form.cleaned_data.get('enable_websocket') or False
|
|||
|
|
|
|||
|
|
# 构造域名列表:至少包含一个主域名或子域名
|
|||
|
|
domains_list = []
|
|||
|
|
if subdomains:
|
|||
|
|
domains_list.extend([f"{sub}.{name}" for sub in subdomains])
|
|||
|
|
else:
|
|||
|
|
domains_list.append(name)
|
|||
|
|
|
|||
|
|
# 构造源站地址列表
|
|||
|
|
origin_addr = f"{origin_protocol}://{origin_host}:{origin_port}"
|
|||
|
|
origin_addrs = [origin_addr]
|
|||
|
|
|
|||
|
|
# 初始化 GoEdge 客户端并创建服务
|
|||
|
|
try:
|
|||
|
|
client = GoEdgeClient()
|
|||
|
|
server_id = client.create_basic_http_server(
|
|||
|
|
domains=domains_list,
|
|||
|
|
origin_addrs=origin_addrs,
|
|||
|
|
user_id=0, # 管理员创建,不指定用户
|
|||
|
|
enable_websocket=enable_websocket,
|
|||
|
|
)
|
|||
|
|
except Exception as e:
|
|||
|
|
messages.error(request, f'创建加速服务失败:{e}')
|
|||
|
|
return render(request, 'domains/add.html', {'form': form})
|
|||
|
|
|
|||
|
|
# 保存域名记录
|
|||
|
|
sys = SystemSettings.objects.order_by('id').first()
|
|||
|
|
cname_template = (sys.cname_template if sys else '{sub}.cdn.example.com')
|
|||
|
|
cname_map = {}
|
|||
|
|
for sub in subdomains or []:
|
|||
|
|
cname_map[f"{sub}.{name}"] = cname_template.replace('{sub}', sub)
|
|||
|
|
if not cname_map:
|
|||
|
|
# 主域名CNAME场景(可选)
|
|||
|
|
cname_map[name] = cname_template.replace('{sub}', 'www')
|
|||
|
|
|
|||
|
|
# 计费周期:设置为当前自然月
|
|||
|
|
today = timezone.now().date()
|
|||
|
|
month_start = today.replace(day=1)
|
|||
|
|
# 下个月第一天
|
|||
|
|
if month_start.month == 12:
|
|||
|
|
next_month_start = month_start.replace(year=month_start.year + 1, month=1, day=1)
|
|||
|
|
else:
|
|||
|
|
next_month_start = month_start.replace(month=month_start.month + 1, day=1)
|
|||
|
|
current_cycle_end = next_month_start - timezone.timedelta(days=1)
|
|||
|
|
|
|||
|
|
domain = Domain.objects.create(
|
|||
|
|
user=request.user,
|
|||
|
|
name=name,
|
|||
|
|
status=Domain.STATUS_PENDING,
|
|||
|
|
current_plan=plan,
|
|||
|
|
current_cycle_start=month_start,
|
|||
|
|
current_cycle_end=current_cycle_end,
|
|||
|
|
cname_targets=cname_map,
|
|||
|
|
origin_config={
|
|||
|
|
'host': origin_host,
|
|||
|
|
'protocol': origin_protocol,
|
|||
|
|
'port': origin_port,
|
|||
|
|
},
|
|||
|
|
edge_server_id=server_id,
|
|||
|
|
)
|
|||
|
|
|
|||
|
|
# 记录操作日志
|
|||
|
|
try:
|
|||
|
|
OperationLog.objects.create(
|
|||
|
|
actor=request.user,
|
|||
|
|
action='create_domain',
|
|||
|
|
target=name,
|
|||
|
|
detail=f"server_id={server_id}; origin={origin_protocol}://{origin_host}:{origin_port}; subdomains={','.join(subdomains or [])}",
|
|||
|
|
)
|
|||
|
|
except Exception:
|
|||
|
|
# 日志写入失败不阻断主流程
|
|||
|
|
pass
|
|||
|
|
|
|||
|
|
messages.success(request, '域名已创建,请按提示配置DNS CNAME记录并等待生效。')
|
|||
|
|
return redirect(reverse('domains:detail', kwargs={'domain_id': domain.id}))
|
|||
|
|
else:
|
|||
|
|
form = AddDomainForm()
|
|||
|
|
|
|||
|
|
return render(request, 'domains/add.html', {'form': form})
|
|||
|
|
|
|||
|
|
@login_required
|
|||
|
|
def domain_detail(request, domain_id: int):
|
|||
|
|
domain = get_object_or_404(Domain, id=domain_id, user=request.user)
|
|||
|
|
|
|||
|
|
# 流量统计:当月(按自然月)
|
|||
|
|
today = timezone.now().date()
|
|||
|
|
month_start = today.replace(day=1)
|
|||
|
|
traffic_qs = DomainTrafficDaily.objects.filter(domain=domain, day__gte=month_start)
|
|||
|
|
total_bytes = sum(t.bytes for t in traffic_qs)
|
|||
|
|
used_gb = bytes_to_gb(total_bytes)
|
|||
|
|
|
|||
|
|
# 配额计算:Plan 或系统默认 + 域名当期额外
|
|||
|
|
sys = SystemSettings.objects.order_by('id').first()
|
|||
|
|
base_quota_gb = 0
|
|||
|
|
if domain.current_plan and domain.current_plan.included_traffic_gb_per_domain:
|
|||
|
|
base_quota_gb = domain.current_plan.included_traffic_gb_per_domain
|
|||
|
|
elif sys:
|
|||
|
|
base_quota_gb = sys.default_free_traffic_gb_per_domain
|
|||
|
|
total_quota_gb = (base_quota_gb or 0) + (domain.extra_free_traffic_gb_current_cycle or 0)
|
|||
|
|
progress_pct = 0
|
|||
|
|
if total_quota_gb > 0:
|
|||
|
|
progress_pct = int(min(100, (used_gb / total_quota_gb) * 100))
|
|||
|
|
|
|||
|
|
# 近7/30天统计与峰值(跨自然月)
|
|||
|
|
seven_start = today - timezone.timedelta(days=6)
|
|||
|
|
thirty_start = today - timezone.timedelta(days=29)
|
|||
|
|
last7_qs = DomainTrafficDaily.objects.filter(domain=domain, day__gte=seven_start, day__lte=today)
|
|||
|
|
last30_qs = DomainTrafficDaily.objects.filter(domain=domain, day__gte=thirty_start, day__lte=today)
|
|||
|
|
|
|||
|
|
last7_total_gb = bytes_to_gb(sum(t.bytes for t in last7_qs))
|
|||
|
|
last30_total_gb = bytes_to_gb(sum(t.bytes for t in last30_qs))
|
|||
|
|
# 平均值(GB/日)
|
|||
|
|
last7_count = last7_qs.count()
|
|||
|
|
last30_count = last30_qs.count()
|
|||
|
|
last7_avg_gb = round((last7_total_gb / last7_count), 2) if last7_count else 0
|
|||
|
|
last30_avg_gb = round((last30_total_gb / last30_count), 2) if last30_count else 0
|
|||
|
|
|
|||
|
|
peak_volume_record = last30_qs.order_by('-bytes').first()
|
|||
|
|
peak_bandwidth_record = last30_qs.order_by('-peak_bandwidth_mbps').first()
|
|||
|
|
peak_volume_day = None
|
|||
|
|
peak_bandwidth_day = None
|
|||
|
|
if peak_volume_record:
|
|||
|
|
peak_volume_day = {
|
|||
|
|
'day': peak_volume_record.day,
|
|||
|
|
'gb': bytes_to_gb(peak_volume_record.bytes),
|
|||
|
|
}
|
|||
|
|
if peak_bandwidth_record:
|
|||
|
|
peak_bandwidth_day = {
|
|||
|
|
'day': peak_bandwidth_record.day,
|
|||
|
|
'mbps': peak_bandwidth_record.peak_bandwidth_mbps,
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
# 近30天每日统计(GB 与峰值带宽)
|
|||
|
|
last_30 = list(last30_qs.order_by('-day'))
|
|||
|
|
traffic_rows = []
|
|||
|
|
for t in last_30:
|
|||
|
|
gb = bytes_to_gb(t.bytes)
|
|||
|
|
traffic_rows.append({'day': t.day, 'gb': gb, 'peak_mbps': t.peak_bandwidth_mbps})
|
|||
|
|
|
|||
|
|
# GoEdge 同步状态
|
|||
|
|
goedge_status = None
|
|||
|
|
if domain.edge_server_id:
|
|||
|
|
try:
|
|||
|
|
client = GoEdgeClient()
|
|||
|
|
goedge_status = client.get_server_feature_status(domain.edge_server_id)
|
|||
|
|
except Exception as e:
|
|||
|
|
goedge_status = {'error': str(e)}
|
|||
|
|
|
|||
|
|
requests_24h_total = None
|
|||
|
|
status_bins = None
|
|||
|
|
status_top = None
|
|||
|
|
if domain.edge_server_id:
|
|||
|
|
try:
|
|||
|
|
client = GoEdgeClient()
|
|||
|
|
hs = client.find_latest_server_hourly_stats(int(domain.edge_server_id), hours=24)
|
|||
|
|
requests_24h_total = sum(int(x.get('countRequests') or 0) for x in hs)
|
|||
|
|
agg = client.aggregate_status_codes(int(domain.edge_server_id))
|
|||
|
|
status_bins = agg.get('bins')
|
|||
|
|
status_top = agg.get('top')
|
|||
|
|
except Exception:
|
|||
|
|
pass
|
|||
|
|
|
|||
|
|
context = {
|
|||
|
|
'domain': domain,
|
|||
|
|
'used_gb': used_gb,
|
|||
|
|
'total_quota_gb': total_quota_gb,
|
|||
|
|
'progress_pct': progress_pct,
|
|||
|
|
'traffic_rows': traffic_rows,
|
|||
|
|
'last7_total_gb': last7_total_gb,
|
|||
|
|
'last30_total_gb': last30_total_gb,
|
|||
|
|
'last7_avg_gb': last7_avg_gb,
|
|||
|
|
'last30_avg_gb': last30_avg_gb,
|
|||
|
|
'peak_volume_day': peak_volume_day,
|
|||
|
|
'peak_bandwidth_day': peak_bandwidth_day,
|
|||
|
|
'cname_results': None,
|
|||
|
|
'goedge_status': goedge_status,
|
|||
|
|
'requests_24h_total': requests_24h_total,
|
|||
|
|
'status_bins': status_bins,
|
|||
|
|
'status_top': status_top,
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
return render(request, 'domains/detail.html', context)
|
|||
|
|
|
|||
|
|
|
|||
|
|
@login_required
|
|||
|
|
@transaction.atomic
|
|||
|
|
def check_dns(request, domain_id: int):
|
|||
|
|
domain = get_object_or_404(Domain, id=domain_id, user=request.user)
|
|||
|
|
if request.method != 'POST':
|
|||
|
|
return redirect(reverse('domains:detail', kwargs={'domain_id': domain.id}))
|
|||
|
|
|
|||
|
|
cname_map = domain.cname_targets or {}
|
|||
|
|
results = check_cname_map(cname_map)
|
|||
|
|
|
|||
|
|
ok_count = sum(1 for r in results if r.get('ok'))
|
|||
|
|
total = len(results)
|
|||
|
|
if ok_count == total and total > 0:
|
|||
|
|
messages.success(request, 'DNS 已生效,所有 CNAME 指向正确。')
|
|||
|
|
# 如果当前为等待DNS,可切换为 active
|
|||
|
|
if domain.status == Domain.STATUS_PENDING:
|
|||
|
|
domain.status = Domain.STATUS_ACTIVE
|
|||
|
|
domain.save(update_fields=['status'])
|
|||
|
|
else:
|
|||
|
|
messages.warning(request, f'DNS 检测完成:{ok_count}/{total} 条记录正确。请检查未生效的记录。')
|
|||
|
|
|
|||
|
|
# 重渲染详情页并显示检测结果
|
|||
|
|
today = timezone.now().date()
|
|||
|
|
month_start = today.replace(day=1)
|
|||
|
|
traffic_qs = DomainTrafficDaily.objects.filter(domain=domain, day__gte=month_start)
|
|||
|
|
total_bytes = sum(t.bytes for t in traffic_qs)
|
|||
|
|
used_gb = bytes_to_gb(total_bytes)
|
|||
|
|
sys = SystemSettings.objects.order_by('id').first()
|
|||
|
|
base_quota_gb = 0
|
|||
|
|
if domain.current_plan and domain.current_plan.included_traffic_gb_per_domain:
|
|||
|
|
base_quota_gb = domain.current_plan.included_traffic_gb_per_domain
|
|||
|
|
elif sys:
|
|||
|
|
base_quota_gb = sys.default_free_traffic_gb_per_domain
|
|||
|
|
total_quota_gb = (base_quota_gb or 0) + (domain.extra_free_traffic_gb_current_cycle or 0)
|
|||
|
|
progress_pct = 0
|
|||
|
|
if total_quota_gb > 0:
|
|||
|
|
progress_pct = int(min(100, (used_gb / total_quota_gb) * 100))
|
|||
|
|
|
|||
|
|
seven_start = today - timezone.timedelta(days=6)
|
|||
|
|
thirty_start = today - timezone.timedelta(days=29)
|
|||
|
|
last30_qs = DomainTrafficDaily.objects.filter(domain=domain, day__gte=thirty_start, day__lte=today)
|
|||
|
|
last7_qs = DomainTrafficDaily.objects.filter(domain=domain, day__gte=seven_start, day__lte=today)
|
|||
|
|
peak_volume_record = last30_qs.order_by('-bytes').first()
|
|||
|
|
peak_bandwidth_record = last30_qs.order_by('-peak_bandwidth_mbps').first()
|
|||
|
|
return render(request, 'domains/detail.html', {
|
|||
|
|
'domain': domain,
|
|||
|
|
'used_gb': used_gb,
|
|||
|
|
'total_quota_gb': total_quota_gb,
|
|||
|
|
'progress_pct': progress_pct,
|
|||
|
|
'traffic_rows': [{
|
|||
|
|
'day': t.day,
|
|||
|
|
'gb': bytes_to_gb(t.bytes),
|
|||
|
|
'peak_mbps': t.peak_bandwidth_mbps,
|
|||
|
|
} for t in last30_qs.order_by('-day')],
|
|||
|
|
'last7_total_gb': bytes_to_gb(sum(t.bytes for t in last7_qs)),
|
|||
|
|
'last30_total_gb': bytes_to_gb(sum(t.bytes for t in last30_qs)),
|
|||
|
|
'last7_avg_gb': (round(bytes_to_gb(sum(t.bytes for t in last7_qs)) / last7_qs.count(), 2) if last7_qs.count() else 0),
|
|||
|
|
'last30_avg_gb': (round(bytes_to_gb(sum(t.bytes for t in last30_qs)) / last30_qs.count(), 2) if last30_qs.count() else 0),
|
|||
|
|
'peak_volume_day': ({
|
|||
|
|
'day': peak_volume_record.day,
|
|||
|
|
'gb': bytes_to_gb(peak_volume_record.bytes),
|
|||
|
|
} if peak_volume_record else None),
|
|||
|
|
'peak_bandwidth_day': ({
|
|||
|
|
'day': peak_bandwidth_record.day,
|
|||
|
|
'mbps': peak_bandwidth_record.peak_bandwidth_mbps,
|
|||
|
|
} if peak_bandwidth_record else None),
|
|||
|
|
'cname_results': results,
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
# Create your views here.
|
|||
|
|
|
|||
|
|
|
|||
|
|
@login_required
|
|||
|
|
@transaction.atomic
|
|||
|
|
def upgrade_plan(request, domain_id: int):
|
|||
|
|
"""最小套餐升级流程:展示公开套餐并允许用户选择变更当前域名的套餐。"""
|
|||
|
|
domain = get_object_or_404(Domain, id=domain_id, user=request.user)
|
|||
|
|
|
|||
|
|
# 可选套餐列表(公开且激活)
|
|||
|
|
available_plans = Plan.objects.filter(is_active=True, is_public=True).order_by('base_price_per_domain')
|
|||
|
|
|
|||
|
|
if request.method == 'POST':
|
|||
|
|
plan_id = request.POST.get('plan_id')
|
|||
|
|
try:
|
|||
|
|
new_plan = available_plans.get(id=plan_id)
|
|||
|
|
except Plan.DoesNotExist:
|
|||
|
|
messages.error(request, '选择的套餐不可用或不存在。')
|
|||
|
|
return render(request, 'domains/upgrade.html', {
|
|||
|
|
'domain': domain,
|
|||
|
|
'plans': available_plans,
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
old_plan = domain.current_plan
|
|||
|
|
domain.current_plan = new_plan
|
|||
|
|
domain.save(update_fields=['current_plan', 'updated_at'])
|
|||
|
|
|
|||
|
|
# 操作日志
|
|||
|
|
try:
|
|||
|
|
OperationLog.objects.create(
|
|||
|
|
actor=request.user,
|
|||
|
|
action='upgrade_plan',
|
|||
|
|
target=domain.name,
|
|||
|
|
detail=f"from={old_plan.name if old_plan else '-'} to={new_plan.name}",
|
|||
|
|
)
|
|||
|
|
except Exception:
|
|||
|
|
pass
|
|||
|
|
|
|||
|
|
messages.success(request, f'套餐已升级为:{new_plan.name}')
|
|||
|
|
return redirect(reverse('domains:detail', kwargs={'domain_id': domain.id}))
|
|||
|
|
|
|||
|
|
return render(request, 'domains/upgrade.html', {
|
|||
|
|
'domain': domain,
|
|||
|
|
'plans': available_plans,
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
|
|||
|
|
@login_required
|
|||
|
|
@transaction.atomic
|
|||
|
|
def domain_settings(request, domain_id: int):
|
|||
|
|
"""域名功能设置页:保存到 Domain.custom_features,后续接入 GoEdge 配置更新。"""
|
|||
|
|
domain = get_object_or_404(Domain, id=domain_id)
|
|||
|
|
# 权限:仅域名所有者或staff
|
|||
|
|
if not (domain.user_id == request.user.id or request.user.is_staff):
|
|||
|
|
messages.warning(request, '无权编辑该域名设置。')
|
|||
|
|
return redirect('domains:detail', domain_id=domain.id)
|
|||
|
|
|
|||
|
|
plan_features = domain.current_plan.features if domain.current_plan else {}
|
|||
|
|
import json
|
|||
|
|
initial = {
|
|||
|
|
'waf_enabled': bool(domain.custom_features.get('waf_enabled', plan_features.get('waf_enabled', False))),
|
|||
|
|
'http3_enabled': bool(domain.custom_features.get('http3_enabled', plan_features.get('http3_enabled', False))),
|
|||
|
|
'logs_enabled': bool(domain.custom_features.get('logs_enabled', plan_features.get('logs_enabled', False))),
|
|||
|
|
'websocket_enabled': bool(domain.custom_features.get('websocket_enabled', plan_features.get('websocket_enabled', False))),
|
|||
|
|
'redirect_https_enabled': bool(domain.custom_features.get('redirect_https_enabled', False)),
|
|||
|
|
'cache_rules_json': json.dumps(domain.custom_features.get('cache_rules_json', {}), ensure_ascii=False),
|
|||
|
|
'page_rules_json': json.dumps(domain.custom_features.get('page_rules_json', {}), ensure_ascii=False),
|
|||
|
|
'ip_whitelist': ','.join(domain.custom_features.get('ip_whitelist', [])),
|
|||
|
|
'ip_blacklist': ','.join(domain.custom_features.get('ip_blacklist', [])),
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
if request.method == 'POST':
|
|||
|
|
form = DomainSettingsForm(request.POST)
|
|||
|
|
if form.is_valid():
|
|||
|
|
data = form.cleaned_data
|
|||
|
|
domain.custom_features = {
|
|||
|
|
'waf_enabled': bool(data.get('waf_enabled')),
|
|||
|
|
'http3_enabled': bool(data.get('http3_enabled')),
|
|||
|
|
'logs_enabled': bool(data.get('logs_enabled')),
|
|||
|
|
'websocket_enabled': bool(data.get('websocket_enabled')),
|
|||
|
|
'redirect_https_enabled': bool(data.get('redirect_https_enabled')),
|
|||
|
|
'cache_rules_json': data.get('cache_rules_json') or {},
|
|||
|
|
'page_rules_json': data.get('page_rules_json') or {},
|
|||
|
|
'ip_whitelist': data.get('ip_whitelist') or [],
|
|||
|
|
'ip_blacklist': data.get('ip_blacklist') or [],
|
|||
|
|
}
|
|||
|
|
domain.save(update_fields=['custom_features', 'updated_at'])
|
|||
|
|
# 记录操作日志(含策略ID与同步结果)
|
|||
|
|
log_detail = {
|
|||
|
|
'requested': {
|
|||
|
|
'waf_enabled': bool(data.get('waf_enabled')),
|
|||
|
|
'http3_enabled': bool(data.get('http3_enabled')),
|
|||
|
|
'logs_enabled': bool(data.get('logs_enabled')),
|
|||
|
|
'websocket_enabled': bool(data.get('websocket_enabled')),
|
|||
|
|
},
|
|||
|
|
'sync': {
|
|||
|
|
'accessLog': None,
|
|||
|
|
'websocket': None,
|
|||
|
|
'firewall': None,
|
|||
|
|
'ssl_http3': None,
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
# 尝试同步到 GoEdge(根据可用策略与配置)
|
|||
|
|
try:
|
|||
|
|
client = GoEdgeClient()
|
|||
|
|
web_id = client.find_server_web_id(domain.edge_server_id)
|
|||
|
|
if web_id:
|
|||
|
|
sys = SystemSettings.objects.order_by('id').first()
|
|||
|
|
# 访问日志
|
|||
|
|
client.update_http_web_access_log(
|
|||
|
|
http_web_id=web_id,
|
|||
|
|
is_on=bool(data.get('logs_enabled')),
|
|||
|
|
policy_id=(sys.default_http_access_log_policy_id if sys else None)
|
|||
|
|
)
|
|||
|
|
log_detail['sync']['accessLog'] = {
|
|||
|
|
'webId': web_id,
|
|||
|
|
'isOn': bool(data.get('logs_enabled')),
|
|||
|
|
'policyId': (sys.default_http_access_log_policy_id if sys else None)
|
|||
|
|
}
|
|||
|
|
# WebSocket
|
|||
|
|
client.update_http_web_websocket(http_web_id=web_id, is_on=bool(data.get('websocket_enabled')))
|
|||
|
|
log_detail['sync']['websocket'] = {'webId': web_id, 'isOn': bool(data.get('websocket_enabled'))}
|
|||
|
|
# WAF(如有默认策略则引用)
|
|||
|
|
client.update_http_web_firewall(
|
|||
|
|
http_web_id=web_id,
|
|||
|
|
is_on=bool(data.get('waf_enabled')),
|
|||
|
|
policy_id=(sys.default_http_firewall_policy_id if sys else None)
|
|||
|
|
)
|
|||
|
|
log_detail['sync']['firewall'] = {
|
|||
|
|
'webId': web_id,
|
|||
|
|
'isOn': bool(data.get('waf_enabled')),
|
|||
|
|
'policyId': (sys.default_http_firewall_policy_id if sys else None)
|
|||
|
|
}
|
|||
|
|
cache_conf = data.get('cache_rules_json') or {}
|
|||
|
|
if isinstance(cache_conf, dict) and cache_conf:
|
|||
|
|
client.update_http_web_cache(http_web_id=web_id, cache_conf=cache_conf)
|
|||
|
|
log_detail['sync']['cache'] = {'webId': web_id, 'applied': True}
|
|||
|
|
page_rules = data.get('page_rules_json') or {}
|
|||
|
|
if isinstance(page_rules, dict) and page_rules:
|
|||
|
|
locations_conf = page_rules.get('locations') or page_rules.get('Locations') or None
|
|||
|
|
rewrite_conf = page_rules.get('rewriteRules') or page_rules.get('RewriteRules') or None
|
|||
|
|
if locations_conf:
|
|||
|
|
client.update_http_web_locations(http_web_id=web_id, locations_conf=locations_conf)
|
|||
|
|
log_detail['sync']['locations'] = {'webId': web_id, 'count': len(locations_conf) if isinstance(locations_conf, list) else 1}
|
|||
|
|
if rewrite_conf:
|
|||
|
|
client.update_http_web_rewrite_rules(http_web_id=web_id, rewrite_conf=rewrite_conf)
|
|||
|
|
log_detail['sync']['rewrite'] = {'webId': web_id, 'count': len(rewrite_conf) if isinstance(rewrite_conf, list) else 1}
|
|||
|
|
# 强制 HTTPS 跳转
|
|||
|
|
if bool(data.get('redirect_https_enabled')):
|
|||
|
|
client.update_http_web_redirect_to_https(http_web_id=web_id, redirect_conf={'isOn': True})
|
|||
|
|
log_detail['sync']['redirectToHTTPS'] = {'webId': web_id, 'isOn': True}
|
|||
|
|
else:
|
|||
|
|
client.update_http_web_redirect_to_https(http_web_id=web_id, redirect_conf={'isOn': False})
|
|||
|
|
log_detail['sync']['redirectToHTTPS'] = {'webId': web_id, 'isOn': False}
|
|||
|
|
# HTTP/3 通过 SSL 策略更新
|
|||
|
|
ssl_policy_id = client.find_server_ssl_policy_id(domain.edge_server_id)
|
|||
|
|
if ssl_policy_id is not None:
|
|||
|
|
client.update_ssl_policy_http3(ssl_policy_id, bool(data.get('http3_enabled')))
|
|||
|
|
log_detail['sync']['ssl_http3'] = {'sslPolicyId': ssl_policy_id, 'http3Enabled': bool(data.get('http3_enabled'))}
|
|||
|
|
except Exception as e:
|
|||
|
|
messages.warning(request, f'部分功能未同步到 GoEdge:{e}')
|
|||
|
|
log_detail['sync']['error'] = str(e)
|
|||
|
|
# 写操作日志
|
|||
|
|
try:
|
|||
|
|
OperationLog.objects.create(
|
|||
|
|
actor=request.user,
|
|||
|
|
action='domain_settings_update',
|
|||
|
|
target=domain.name,
|
|||
|
|
detail=json.dumps(log_detail, ensure_ascii=False)
|
|||
|
|
)
|
|||
|
|
except Exception:
|
|||
|
|
pass
|
|||
|
|
messages.success(request, '设置已保存。部分功能需对应套餐支持方可生效。')
|
|||
|
|
return redirect('domains:settings', domain_id=domain.id)
|
|||
|
|
else:
|
|||
|
|
messages.error(request, '保存失败,请检查表单字段。')
|
|||
|
|
else:
|
|||
|
|
form = DomainSettingsForm(initial=initial)
|
|||
|
|
|
|||
|
|
return render(request, 'domains/settings.html', {
|
|||
|
|
'domain': domain,
|
|||
|
|
'form': form,
|
|||
|
|
'plan_features': plan_features,
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
|
|||
|
|
@login_required
|
|||
|
|
def domain_logs(request, domain_id: int):
|
|||
|
|
domain = get_object_or_404(Domain, id=domain_id, user=request.user)
|
|||
|
|
server_id = int(domain.edge_server_id or 0)
|
|||
|
|
logs = []
|
|||
|
|
request_id = request.GET.get('request_id') or None
|
|||
|
|
day = request.GET.get('day') or None
|
|||
|
|
hour_from = request.GET.get('hour_from') or None
|
|||
|
|
hour_to = request.GET.get('hour_to') or None
|
|||
|
|
ip = request.GET.get('ip') or None
|
|||
|
|
keyword = request.GET.get('keyword') or None
|
|||
|
|
size = int(request.GET.get('size') or 50)
|
|||
|
|
status_code = request.GET.get('status_code') or ''
|
|||
|
|
export = request.GET.get('export') or ''
|
|||
|
|
reverse = bool(request.GET.get('reverse'))
|
|||
|
|
has_more = False
|
|||
|
|
if server_id:
|
|||
|
|
try:
|
|||
|
|
client = GoEdgeClient()
|
|||
|
|
res = client.list_http_access_logs(
|
|||
|
|
server_id=server_id,
|
|||
|
|
day=day,
|
|||
|
|
size=size,
|
|||
|
|
hour_from=hour_from,
|
|||
|
|
hour_to=hour_to,
|
|||
|
|
reverse=reverse,
|
|||
|
|
ip=ip,
|
|||
|
|
keyword=keyword,
|
|||
|
|
request_id=request_id,
|
|||
|
|
)
|
|||
|
|
logs = res.get('logs') or []
|
|||
|
|
request_id = res.get('requestId') or None
|
|||
|
|
has_more = bool(res.get('hasMore'))
|
|||
|
|
except Exception as e:
|
|||
|
|
messages.warning(request, f'访问日志查询失败:{e}')
|
|||
|
|
# 过滤状态码
|
|||
|
|
if status_code:
|
|||
|
|
try:
|
|||
|
|
sc = int(status_code)
|
|||
|
|
logs = [l for l in (logs or []) if int(l.get('status') or 0) == sc]
|
|||
|
|
except Exception:
|
|||
|
|
pass
|
|||
|
|
# 导出
|
|||
|
|
if export == 'csv':
|
|||
|
|
import csv
|
|||
|
|
from django.http import HttpResponse
|
|||
|
|
resp = HttpResponse(content_type='text/csv; charset=utf-8')
|
|||
|
|
resp['Content-Disposition'] = f'attachment; filename="{domain.name}_access_logs.csv"'
|
|||
|
|
writer = csv.writer(resp)
|
|||
|
|
writer.writerow(['timeLocal', 'host', 'remoteAddr', 'method', 'requestURI', 'status', 'bytesSent', 'userAgent'])
|
|||
|
|
for l in logs:
|
|||
|
|
writer.writerow([
|
|||
|
|
l.get('timeLocal'), l.get('host'), l.get('remoteAddr'), l.get('method'), l.get('requestURI'), l.get('status'), l.get('bytesSent'), l.get('userAgent'),
|
|||
|
|
])
|
|||
|
|
return resp
|
|||
|
|
if export == 'json':
|
|||
|
|
from django.http import JsonResponse
|
|||
|
|
return JsonResponse({'domain': domain.name, 'logs': logs}, json_dumps_params={'ensure_ascii': False})
|
|||
|
|
return render(request, 'domains/logs.html', {
|
|||
|
|
'domain': domain,
|
|||
|
|
'logs': logs,
|
|||
|
|
'day': day or '',
|
|||
|
|
'hour_from': hour_from or '',
|
|||
|
|
'hour_to': hour_to or '',
|
|||
|
|
'ip': ip or '',
|
|||
|
|
'keyword': keyword or '',
|
|||
|
|
'size': size,
|
|||
|
|
'reverse': reverse,
|
|||
|
|
'request_id': request_id or '',
|
|||
|
|
'has_more': has_more,
|
|||
|
|
'status_code': status_code or '',
|
|||
|
|
})
|