MrDoc/app_doc/views.py

3356 lines
149 KiB
Python
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

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.

# coding:utf-8
from django.shortcuts import render,redirect
from django.http.response import JsonResponse,Http404,HttpResponseNotAllowed,HttpResponse
from django.http import QueryDict
from django.http import HttpResponseForbidden
from django.contrib.auth.decorators import login_required # 登录需求装饰器
from django.views.decorators.csrf import csrf_exempt
from django.views.decorators.http import require_http_methods,require_GET,require_POST # 视图请求方法装饰器
from django.core.paginator import Paginator,PageNotAnInteger,EmptyPage,InvalidPage # 后端分页
from django.core.exceptions import PermissionDenied,ObjectDoesNotExist
from django.core.serializers import serialize
from app_doc.models import Project,Doc,DocTemp
from django.contrib.auth.models import User
from rest_framework.views import APIView # 视图
from rest_framework.response import Response # 响应
from rest_framework.pagination import PageNumberPagination # 分页
from rest_framework.authentication import SessionAuthentication # 认证
from django.db.models import Q
from django.db import transaction
from django.utils.html import strip_tags
from django.utils.translation import gettext_lazy as _
from loguru import logger
from app_api.serializers_app import *
from app_doc.report_utils import *
from app_admin.models import UserOptions,SysSetting
from app_admin.decorators import check_headers,allow_report_file
from app_api.auth_app import AppAuth,AppMustAuth # 自定义认证
import datetime
import traceback
import re
import json
import random
import os.path
import base64
import hashlib
import markdown
# HTML转义
def jsonXssFilter(data):
payloads = {
'\'':''',
'"':'"',
'<':'&lt;',
'>':'&gt;'
}
if type(data) == dict:
new = {}
for key,values in data.items():
new[key] = jsonXssFilter(values)
elif type(data) == list:
new = []
for i in data:
new.append(jsonXssFilter(i))
elif type(data) == int or type(data) == float:
new = data
elif type(data) == str:
new = data
for key,value in payloads.items():
new = new.replace(key,value)
elif type(data) ==bytes:
new = data
else:
print('>>> unknown type:')
print(type(data))
new = data
return new
def html_filter(data):
if len(data) == 0:
return ""
payloads = {
'\'':'&apos;',
'"':'&quot;',
'<':'&lt;',
'>':'&gt;'
}
new = data
for key, value in payloads.items():
new = new.replace(key, value)
print(new)
return new
# 替换前端传来的非法字符
def validateTitle(title):
rstr = r"[\/\\\:\*\?\"\<\>\|\[\]]" # '/ \ : * ? " < > |'
new_title = re.sub(rstr, "_", title) # 替换为下划线
return new_title
# markdown文本生成摘要不带markdown标记
def remove_markdown_tag(docs):
for doc in docs:
try:
doc.pre_content = strip_tags(markdown.markdown(doc.pre_content))
except Exception as e:
doc.pre_content = doc.pre_content
# 获取文集的文档目录
def get_pro_toc(pro_id):
# try:
# project = Project.objects.get(id=pro_id)
# pro_toc = ProjectToc.objects.get(project=project)
# doc_list = json.loads(pro_toc.value)
# print("使用缓存")
# except:
# print("重新生成")
# 查询存在上级文档的文档
parent_id_list = Doc.objects.filter(
top_doc=pro_id,
status=1
).exclude(parent_doc=0).values_list('parent_doc',flat=True)
# 获取存在上级文档的上级文档ID
# print(parent_id_list)
doc_list = []
n = 0
# 获取一级文档
top_docs = Doc.objects.filter(top_doc=pro_id, parent_doc=0, status=1).values('id', 'name','open_children','editor_mode').order_by('sort')
# 遍历一级文档
for doc in top_docs:
top_item = {
'id': doc['id'],
'name': doc['name'],
'open_children':doc['open_children'],
'editor_mode':doc['editor_mode']
# 'spread': True,
# 'level': 1
}
# 如果一级文档存在下级文档,查询其二级文档
if doc['id'] in parent_id_list:
# 获取二级文档
sec_docs = Doc.objects.filter(
top_doc=pro_id,
parent_doc=doc['id'],
status=1
).values('id', 'name','open_children','editor_mode').order_by('sort')
top_item['children'] = []
for doc in sec_docs:
sec_item = {
'id': doc['id'],
'name': doc['name'],
'open_children': doc['open_children'],
'editor_mode': doc['editor_mode']
# 'level': 2
}
# 如果二级文档存在下级文档,查询第三级文档
if doc['id'] in parent_id_list:
# 获取三级文档
thr_docs = Doc.objects.filter(
top_doc=pro_id,
parent_doc=doc['id'],
status=1
).values('id','name','editor_mode').order_by('sort')
sec_item['children'] = []
for doc in thr_docs:
item = {
'id': doc['id'],
'name': doc['name'],
'editor_mode': doc['editor_mode']
# 'level': 3
}
sec_item['children'].append(item)
n += 1
top_item['children'].append(sec_item)
n += 1
else:
top_item['children'].append(sec_item)
n += 1
doc_list.append(top_item)
n += 1
# 如果一级文档没有下级文档,直接保存
else:
doc_list.append(top_item)
n += 1
# 将文集的大纲目录写入数据库
# ProjectToc.objects.create(
# project = project,
# value = json.dumps(doc_list)
# )
# print(doc_list,n)
# if n > 999:
# return ([],n)
# else:
return (doc_list,n)
# 文集列表(首页)
@logger.catch()
def project_list(request):
kw = request.GET.get('kw','') # 搜索词
sort = request.GET.get('sort','') # 排序,0表示按时间升序排序1表示按时间降序排序''表示按后台配置排序,默认为''
role = request.GET.get('role',-1) # 筛选文集权限,默认为显示所有可显示的文集
# 是否排序
if sort in [0,'0']:
sort_str = ''
elif sort == '':
try:
index_project_sort = SysSetting.objects.get(name='index_project_sort')
if index_project_sort.value == '-1':
sort_str = '-'
else:
sort_str = ''
except:
sort_str = ''
else:
sort_str = '-'
# 是否搜索
if kw == '':
is_kw = False
else:
is_kw = True
# 是否认证
if request.user.is_authenticated:
is_auth = True
else:
is_auth = False
# 是否筛选
if role in ['',-1,'-1']:
is_role = False
role_list = [0,3]
else:
is_role = True
# 没有搜索 and 认证用户 and 没有筛选
if (is_kw is False) and (is_auth) and (is_role is False):
colla_list = [i.project.id for i in ProjectCollaborator.objects.filter(user=request.user)] # 用户的协作文集列表
project_list = Project.objects.filter(
Q(role__in=role_list) | \
Q(role=2,role_value__contains=str(request.user.username)) | \
Q(create_user=request.user) | \
Q(id__in=colla_list)
).order_by('-is_top',"{}create_time".format(sort_str))
# 没有搜索 and 认证用户 and 有筛选
elif (is_kw is False ) and (is_auth) and (is_role):
if role in ['0',0]:
project_list = Project.objects.filter(role=0).order_by('-is_top',"{}create_time".format(sort_str))
elif role in ['1',1]:
project_list = Project.objects.filter(create_user=request.user,role=1).order_by(
'-is_top',"{}create_time".format(sort_str))
elif role in ['2',2]:
project_list = Project.objects.filter(role=2,role_value__contains=str(request.user.username)).order_by(
'-is_top',"{}create_time".format(sort_str))
elif role in ['3',3]:
project_list = Project.objects.filter(role=3).order_by('-is_top',"{}create_time".format(sort_str))
elif role in ['99',99]:
colla_list = [i.project.id for i in ProjectCollaborator.objects.filter(user=request.user)] # 用户的协作文集列表
project_list = Project.objects.filter(id__in=colla_list).order_by('-is_top',"{}create_time".format(sort_str))
else:
return render(request,'404.html')
# 没有搜索 and 游客 and 没有筛选
elif (is_kw is False) and (is_auth is False) and (is_role is False):
project_list = Project.objects.filter(role__in=[0,3]).order_by('-is_top',"{}create_time".format(sort_str))
# 没有搜索 and 游客 and 有筛选
elif (is_kw is False) and (is_auth is False) and (is_role):
if role in ['0',0]:
project_list = Project.objects.filter(role=0).order_by('-is_top',"{}create_time".format(sort_str))
elif role in ['3',3]:
project_list = Project.objects.filter(role=3).order_by('-is_top',"{}create_time".format(sort_str))
else:
return render(request,'404.html')
# 有搜索 and 认证用户 and 没有筛选
elif (is_kw) and (is_auth) and (is_role is False):
colla_list = [i.project.id for i in ProjectCollaborator.objects.filter(user=request.user)] # 用户的协作文集
# 查询所有可显示的文集
project_list = Project.objects.filter(
Q(role__in=[0, 3]) | \
Q(role=2, role_value__contains=str(request.user.username)) | \
Q(create_user=request.user) | \
Q(id__in=colla_list),
Q(name__icontains=kw) | Q(intro__icontains=kw)
).order_by('-is_top','{}create_time'.format(sort_str))
# 有搜索 and 认证用户 and 有筛选
elif (is_kw) and (is_auth) and (is_role):
if role in ['0',0]:
project_list = Project.objects.filter(
Q(name__icontains=kw)|Q(intro__icontains=kw),
role=0
).order_by('-is_top',"{}create_time".format(sort_str))
elif role in ['1',1]:
project_list = Project.objects.filter(
Q(name__icontains=kw) | Q(intro__icontains=kw),
create_user=request.user
).order_by('-is_top',"{}create_time".format(sort_str))
elif role in ['2',2]:
project_list = Project.objects.filter(
Q(name__icontains=kw) | Q(intro__icontains=kw),
role=2,
role_value__contains=str(request.user.username)
).order_by('-is_top',"{}create_time".format(sort_str))
elif role in ['3',3]:
project_list = Project.objects.filter(
Q(name__icontains=kw) | Q(intro__icontains=kw),
role=3
).order_by('-is_top',"{}create_time".format(sort_str))
elif role in ['99',99]:
colla_list = [i.project.id for i in ProjectCollaborator.objects.filter(user=request.user)] # 用户的协作文集列表
project_list = Project.objects.filter(
Q(name__icontains=kw) | Q(intro__icontains=kw),
id__in=colla_list
).order_by('-is_top',"{}create_time".format(sort_str))
else:
return render(request,'404.html')
# 有搜索 and 游客 and 没有筛选
elif (is_kw) and (is_auth is False) and (is_role is False):
project_list = Project.objects.filter(
Q(name__icontains=kw) | Q(intro__icontains=kw),
role__in=[0, 3]
).order_by('-is_top',"{}create_time".format(sort_str))
# 有搜索 and 游客 and 有筛选
elif (is_kw) and (is_auth is False) and (is_role):
if role in ['0',0]:
project_list = Project.objects.filter(
Q(name__icontains=kw) | Q(intro__icontains=kw),
role=0
).order_by('-is_top',"{}create_time".format(sort_str))
elif role in ['3',3]:
project_list = Project.objects.filter(
Q(name__icontains=kw) | Q(intro__icontains=kw),
role=3
).order_by('-is_top',"{}create_time".format(sort_str))
else:
return render(request,'404.html')
# 分页处理
paginator = Paginator(project_list, 12)
page = request.GET.get('page', 1)
try:
projects = paginator.page(page)
except PageNotAnInteger:
projects = paginator.page(1)
except EmptyPage:
projects = paginator.page(paginator.num_pages)
return render(request, 'app_doc/pro_list.html', locals())
# 创建文集
@login_required()
@require_http_methods(['POST'])
def create_project(request):
try:
name = request.POST.get('pname','')
name = validateTitle(name)
icon = request.POST.get('picon',None)
desc = request.POST.get('desc','')
role = request.POST.get('role',0)
role_list = ['0','1','2','3',0,1,2,3]
if name != '':
# 不允许用户下同名文集存在
if Project.objects.filter(name=name,create_user=request.user).exists():
return JsonResponse({'status': False, 'data': _('同名文集已存在!')})
project = Project.objects.create(
name=validateTitle(name),
icon = icon,
intro=desc[:100],
create_user=request.user,
role = int(role) if role in role_list else 0
)
project.save()
return JsonResponse({'status':True,'data':{'id':project.id,'name':project.name,'role':project.role}})
else:
return JsonResponse({'status':False,'data':_('文集名称不能为空!')})
except Exception as e:
logger.exception(_("创建文集出错"))
return JsonResponse({'status':False,'data':_('出现异常,请检查输入值!')})
# 文集页
@require_http_methods(['GET'])
@check_headers
def project_index(request,pro_id):
# 获取文集
try:
# 获取文集信息
project = Project.objects.get(id=int(pro_id))
# 获取文集最新的5篇文档
new_docs = Doc.objects.filter(top_doc=pro_id,status=1).order_by('-modify_time')[:5]
# markdown文本生成摘要不带markdown标记
remove_markdown_tag(new_docs)
# 获取文集的文档目录
toc_list,toc_cnt = get_pro_toc(pro_id)
# toc_list,toc_cnt = ([],1000)
# 获取文集的协作成员
colla_user_list = ProjectCollaborator.objects.filter(project=project)
# 获取文集收藏状态
if request.user.is_authenticated:
is_collect_pro = MyCollect.objects.filter(
collect_type=2,collect_id=pro_id,create_user=request.user).exists()
else:
is_collect_pro = False
# 获取文集的协作用户信息
if request.user.is_authenticated: # 对登陆用户查询其协作文档信息
colla_user = ProjectCollaborator.objects.filter(project=project,user=request.user).count()
else:
colla_user = 0
# 获取文集前台下载权限
try:
allow_download = ProjectReport.objects.get(project=project)
except ObjectDoesNotExist:
allow_download = False
# 私密文集并且访问者非创建者非协作者
if (project.role == 1) and (request.user != project.create_user) and (colla_user == 0):
return render(request,'404.html')
# 指定用户可见文集
elif project.role == 2:
user_list = project.role_value
if request.user.is_authenticated: # 认证用户判断是否在许可用户列表中
if (request.user.username not in user_list) and \
(request.user != project.create_user) and \
(colla_user == 0): # 访问者不在指定用户之中
return render(request, '404.html')
else:# 游客直接返回404
return render(request, '404.html')
# 访问码可见
elif project.role == 3:
# 浏览用户不为创建者、协作者
if request.user != project.create_user and colla_user == 0:
viewcode = project.role_value
viewcode_name = 'viewcode-{}'.format(project.id)
r_viewcode = request.COOKIES[viewcode_name] if viewcode_name in request.COOKIES.keys() else 0 # 从cookie中获取访问码
if viewcode != r_viewcode: # cookie中的访问码不等于文集访问码跳转到访问码认证界面
return redirect('/check_viewcode/?to={}'.format(request.path))
# 获取搜索词
kw = request.GET.get('kw','')
# 获取文集下所有一级文档
# project_docs = Doc.objects.filter(
# top_doc=int(pro_id),
# parent_doc=0,
# status=1
# ).values('id','name','top_doc').order_by('sort')
if kw != '':
search_result = Doc.objects.filter(Q(pre_content__icontains=kw) | Q(name__icontains=kw),top_doc=int(pro_id))
remove_markdown_tag(search_result)
return render(request,'app_doc/project_doc_search.html',locals())
return render(request, 'app_doc/project.html', locals())
except Exception as e:
logger.exception(_("文集页访问异常"))
return render(request,'404.html')
# 修改文集
@login_required()
@require_http_methods(['GET','POST'])
def modify_project(request):
if request.method == 'GET':
pro_id = request.GET.get('pro_id', None)
pro = Project.objects.get(id=pro_id)
project_files = ProjectReportFile.objects.filter(project=pro) # 文集的导出文件列表
# 验证用户有权限修改文集
if (request.user == pro.create_user) or request.user.is_superuser:
return render(request,'app_doc/manage/manage_project_options.html',locals())
else:
return Http404
elif request.method == 'POST':
try:
pro_id = request.POST.get('pro_id',None)
project = Project.objects.get(id=pro_id)
# 验证用户有权限修改文集
if (request.user == project.create_user) or request.user.is_superuser:
name = request.POST.get('name',None)
icon = request.POST.get('picon', None)
content = request.POST.get('desc',None)
is_watermark = request.POST.get('is_watermark',False)
if is_watermark == 'true':
is_watermark = True
else:
is_watermark = False
watermark_value = request.POST.get('watermark_value','')
project.name = validateTitle(name)
project.intro = content
project.icon = icon
project.is_watermark = is_watermark
project.watermark_value = watermark_value
project.save()
return JsonResponse({'status':True,'data':_('修改成功')})
else:
return JsonResponse({'status':False,'data':_('非法请求')})
except Exception as e:
logger.exception(_("修改文集出错"))
return JsonResponse({'status':False,'data':_('请求出错')})
# 修改文集权限
@login_required()
@require_http_methods(['GET',"POST"])
@logger.catch()
def modify_project_role(request,pro_id):
try:
pro = Project.objects.get(id=pro_id)
except ObjectDoesNotExist:
return Http404
if (pro.create_user != request.user) and (request.user.is_superuser is False):
return render(request,'403.html')
else:
if request.method == 'GET':
return render(request,'app_doc/manage/manage_project_role.html',locals())
elif request.method == 'POST':
role_type = request.POST.get('role','')
if role_type != '':
try:
if int(role_type) in [0,1]:# 公开或私密
Project.objects.filter(id=int(pro_id)).update(
role = role_type,
modify_time = datetime.datetime.now()
)
if int(role_type) == 2: # 指定用户可见
role_value = request.POST.get('tagsinput','')
Project.objects.filter(id=int(pro_id)).update(
role=role_type,
role_value = role_value,
modify_time = datetime.datetime.now()
)
if int(role_type) == 3: # 访问码可见
role_value = request.POST.get('viewcode','')
Project.objects.filter(id=int(pro_id)).update(
role=role_type,
role_value=role_value,
modify_time=datetime.datetime.now()
)
pro = Project.objects.get(id=int(pro_id))
# return render(request, 'app_doc/manage/manage_project_role.html', locals())
return JsonResponse({'status':True,'data':'ok'})
except:
return JsonResponse({'status':False,'data':_('出错')})
else:
return Http404
# 验证文集访问码
@require_http_methods(['GET',"POST"])
def check_viewcode(request):
try:
if request.method == 'GET':
project_id = request.GET.get('to','').split("/")[1].split('-')[1]
project = Project.objects.get(id=int(project_id))
return render(request,'app_doc/check_viewcode.html',locals())
else:
viewcode = request.POST.get('viewcode','')
project_id = request.POST.get('project_id','')
project = Project.objects.get(id=int(project_id))
if project.role == 3 and project.role_value == viewcode:
obj = redirect("pro_index",pro_id=project_id)
obj.set_cookie('viewcode-{}'.format(project_id),viewcode)
return obj
else:
errormsg = _("访问码错误")
return render(request, 'app_doc/check_viewcode.html', locals())
except Exception as e:
logger.exception(_("验证文集访问码出错"))
return render(request,'404.html')
# 删除文集
@login_required()
@require_http_methods(["POST"])
def del_project(request):
try:
range = request.POST.get('range','single')
pro_id = request.POST.get('pro_id','')
if pro_id != '':
if range == 'single':
pro = Project.objects.get(id=pro_id)
if (request.user == pro.create_user) or (request.user.is_superuser):
# 删除文集下的文档、文档历史、文档分享、文档标签
pro_doc_list = Doc.objects.filter(top_doc=int(pro_id))
for doc in pro_doc_list:
DocHistory.objects.filter(doc=doc).delete()
DocShare.objects.filter(doc=doc).delete()
DocTag.objects.filter(doc=doc).delete()
pro_doc_list.delete()
# 删除文集
pro.delete()
return JsonResponse({'status':True})
else:
return JsonResponse({'status':False,'data':_('非法请求')})
elif range == 'multi':
pros = pro_id.split(",")
try:
projects = Project.objects.filter(id__in=pros, create_user=request.user)
# 删除文集下的文档、文档历史、文档分享、文档标签
pro_doc_list = Doc.objects.filter(top_doc__in=[i.id for i in projects])
for doc in pro_doc_list:
DocHistory.objects.filter(doc=doc).delete()
DocShare.objects.filter(doc=doc).delete()
DocTag.objects.filter(doc=doc).delete()
pro_doc_list.delete()
projects.delete()
return JsonResponse({'status': True, 'data': 'ok'})
except Exception:
logger.exception(_("异常"))
return JsonResponse({'status': False, 'data': _('无指定内容')})
else:
return JsonResponse({'status': False, 'data': _('类型错误')})
else:
return JsonResponse({'status':False,'data':_('参数错误')})
except Exception as e:
logger.exception(_("删除文集出错"))
return JsonResponse({'status':False,'data':_('请求出错')})
# 管理文集
@login_required()
@require_http_methods(['GET','POST'])
def manage_project(request):
if request.method == 'GET':
return render(request,'app_doc/manage/manage_project.html',locals())
else:
kw = request.POST.get('kw','')
page = request.POST.get('page', 1)
limit = request.POST.get('limit', 10)
# 获取文集列表
if kw == '':
project_list = Project.objects.filter(create_user=request.user).order_by('-create_time')
else:
project_list = Project.objects.filter(
Q(intro__icontains=kw) | Q(name__icontains=kw),
create_user=request.user,
).order_by('-create_time')
paginator = Paginator(project_list, limit)
try:
pros = paginator.page(page)
except PageNotAnInteger:
pros = paginator.page(1)
except EmptyPage:
pros = paginator.page(paginator.num_pages)
table_data = []
for project in pros:
item = {
'id':project.id,
'name':project.name,
'intro':project.intro,
'doc_total':Doc.objects.filter(top_doc=project.id).count(),
'role':project.role,
'role_value':project.role_value,
'colla_total':ProjectCollaborator.objects.filter(project=project).count(),
'create_time':project.create_time,
'modify_time':project.modify_time
}
table_data.append(item)
resp_data = {
"code": 0,
"msg": "ok",
"count": project_list.count(),
"data": table_data
}
return JsonResponse(resp_data)
# 管理文集 - 文集文档排序
@login_required()
@require_http_methods(['GET','POST'])
def manage_project_doc_sort(request,pro_id):
if request.method == 'GET':
try:
# 获取文集
pro = Project.objects.get(id=pro_id)
except ObjectDoesNotExist:
return render(request, '404.html')
# 查询文集的协作者
pro_colla = ProjectCollaborator.objects.filter(project=pro,user=request.user,role=1)
# 文集的创建者和文集高级权限协作者允许操作
if (pro.create_user == request.user) or pro_colla.count() > 0:
# 查询存在上级文档的文档
parent_id_list = Doc.objects.filter(top_doc=pro_id, status=1).exclude(parent_doc=0).values_list(
'parent_doc', flat=True)
# 获取存在上级文档的上级文档ID
doc_list = []
# 获取一级文档
top_docs = Doc.objects.filter(top_doc=pro_id, parent_doc=0, status=1).values('id', 'name').order_by('sort')
# 遍历一级文档
for doc in top_docs:
top_item = {
'id': doc['id'],
'field': doc['name'],
'title': doc['name'],
'spread': True,
'level': 1
}
# 如果一级文档存在下级文档,查询其二级文档
if doc['id'] in parent_id_list:
# 获取二级文档
sec_docs = Doc.objects.filter(top_doc=pro_id, parent_doc=doc['id'], status=1).values('id',
'name').order_by(
'sort')
top_item['children'] = []
for doc in sec_docs:
sec_item = {
'id': doc['id'],
'field': doc['name'],
'title': doc['name'],
'level': 2
}
# 如果二级文档存在下级文档,查询第三级文档
if doc['id'] in parent_id_list:
# 获取三级文档
thr_docs = Doc.objects.filter(top_doc=pro_id, parent_doc=doc['id'], status=1).values('id',
'name').order_by(
'sort')
sec_item['children'] = []
for doc in thr_docs:
item = {
'id': doc['id'],
'field': doc['name'],
'title': doc['name'],
'level': 3
}
sec_item['children'].append(item)
top_item['children'].append(sec_item)
else:
top_item['children'].append(sec_item)
doc_list.append(top_item)
# 如果一级文档没有下级文档,直接保存
else:
doc_list.append(top_item)
return render(request,'app_doc/manage/manage_project_doc_sort.html',locals())
else:
return render(request, '403.html')
else:
project_id = request.POST.get('pid', None) # 文集ID
sort_data = request.POST.get('sort_data', '[]') # 文档排序列表
try:
sort_data = json.loads(sort_data)
except Exception:
return JsonResponse({'status': False, 'data': _('文档参数错误')})
try:
pro = Project.objects.get(id=project_id)
except ObjectDoesNotExist:
return JsonResponse({'status': False, 'data': _('没有匹配的文集')})
# 查询文集的协作者
pro_colla = ProjectCollaborator.objects.filter(project=pro, user=request.user, role=1)
# 文集的创建者和文集高级权限协作者允许操作
if (pro.create_user == request.user) or pro_colla.count() > 0:
# 文档排序
n = 10
# 第一级文档
for data in sort_data:
# print(data)
Doc.objects.filter(id=data['id']).update(sort=n,parent_doc=0)
n += 10
# 存在第二级文档
if 'children' in data.keys():
n1 = 10
for c1 in data['children']:
Doc.objects.filter(id=c1['id']).update(sort=n1, parent_doc=data['id'])
n1 += 10
# 存在第三级文档
if 'children' in c1.keys():
n2 = 10
for c2 in c1['children']:
Doc.objects.filter(id=c2['id']).update(sort=n2, parent_doc=c1['id'])
n2 += 10
return JsonResponse({'status': True, 'data': 'ok'})
else:
return JsonResponse({'status':False,'data':_('无权操作')})
# 修改文集前台下载权限
@login_required()
@require_http_methods(['GET',"POST"])
@logger.catch()
def modify_project_download(request,pro_id):
try:
pro = Project.objects.get(id=pro_id)
except ObjectDoesNotExist:
return Http404
if (pro.create_user != request.user) and (request.user.is_superuser is False):
return render(request,'403.html')
else:
project_files = ProjectReportFile.objects.filter(project=pro)
if request.method == 'GET':
return render(request,'app_doc/manage/manage_project_download.html',locals())
elif request.method == 'POST':
download_epub = request.POST.get('download_epub',None)
download_pdf = request.POST.get('download_pdf', None)
# print("epub状态:",download_epub)
# EPUB下载权限
if download_epub == 'on':
epub_status = 1
else:
epub_status = 0
# PDF下载权限
if download_pdf == 'on':
pdf_status = 1
else:
pdf_status = 0
# 写入数据库
ProjectReport.objects.update_or_create(
project = pro,defaults={'allow_epub':epub_status}
)
ProjectReport.objects.update_or_create(
project=pro, defaults={'allow_pdf': pdf_status}
)
# return render(request,'app_doc/manage/manage_project_download.html',locals())
return JsonResponse({'status':True,'data':'ok'})
# 文集协作管理
@login_required()
@require_http_methods(['GET',"POST"])
@logger.catch()
def manage_project_collaborator(request,pro_id):
if request.user.is_superuser:
project = Project.objects.filter(id=pro_id)
else:
project = Project.objects.filter(id=pro_id, create_user=request.user)
if project.exists() is False:
return render(request, '404.html')
if request.method == 'GET':
user_list = User.objects.filter(~Q(username=request.user.username)) # 获取用户列表
pro = project[0]
collaborator = ProjectCollaborator.objects.filter(project=pro) # 获取文集的协作者
colla_user_list = [i.user for i in collaborator] # 文集协作用户的ID
colla_docs = Doc.objects.filter(top_doc=pro.id,create_user__in=colla_user_list) # 获取文集协作用户创建的文档
return render(request, 'app_doc/manage/manage_project_collaborator.html', locals())
elif request.method == 'POST':
# type类型0表示新增协作者、1表示删除协作者、2表示修改协作者
types = request.POST.get('types','')
try:
types = int(types)
except:
return JsonResponse({'status':False,'data':_('参数错误')})
# 添加文集协作者
if int(types) == 0:
colla_user = request.POST.get('username','')
role = request.POST.get('role',0)
user = User.objects.filter(username=colla_user)
if user.exists():
if user[0] == project[0].create_user: # 用户为文集的创建者
return JsonResponse({'status':False,'data':_('文集创建者无需添加')})
elif ProjectCollaborator.objects.filter(user=user[0],project=project[0]).exists():
return JsonResponse({'status':False,'data':_('用户已存在')})
else:
ProjectCollaborator.objects.create(
project = project[0],
user = user[0],
role = role if role in ['1',1] else 0
)
return JsonResponse({'status':True,'data':_('添加成功')})
else:
return JsonResponse({'status':False,'data':_('用户不存在')})
# 删除文集协作者
elif int(types) == 1:
username = request.POST.get('username','')
try:
user = User.objects.get(username=username)
pro_colla = ProjectCollaborator.objects.get(project=project[0],user=user)
pro_colla.delete()
return JsonResponse({'status':True,'data':_('删除成功')})
except:
logger.exception(_("删除协作者出错"))
return JsonResponse({'status':False,'data':_('删除出错')})
# 修改协作权限
elif int(types) == 2:
username = request.POST.get('username', '')
role = request.POST.get('role','')
try:
user = User.objects.get(username=username)
pro_colla = ProjectCollaborator.objects.filter(project=project[0], user=user)
pro_colla.update(role=role)
return JsonResponse({'status':True,'data':_('修改成功')})
except:
logger.exception(_("修改协作权限出错"))
return JsonResponse({'status':False,'data':_('修改失败')})
else:
return JsonResponse({'status':False,'data':_('无效的类型')})
# 我协作的文集
@login_required()
@logger.catch()
def manage_pro_colla_self(request):
colla_pros = ProjectCollaborator.objects.filter(user=request.user)
return render(request,'app_doc/manage/manage_project_self_colla.html',locals())
# 我协作的文集文档列表接口
class MyCollaList(APIView):
authentication_classes = (AppAuth, SessionAuthentication)
# 获取列表
def get(self,request):
pid = request.query_params.get('pid','')
page_num = request.query_params.get('page', 1)
limit = request.query_params.get('limit', 10)
if pid == '':
doc_data = ProjectCollaborator.objects.filter(user=request.user).order_by('-create_time')
else:
project = Project.objects.get(id=pid)
doc_data = ProjectCollaborator.objects.filter(user=request.user,project=project).order_by('-create_time')
page = PageNumberPagination() # 实例化一个分页器
page.page_size = limit
page_docs = page.paginate_queryset(doc_data, request, view=self) # 进行分页查询
serializer = ProjectCollaSerializer(page_docs, many=True) # 对分页后的结果进行序列化处理
colla_doc_list = []
for s in serializer.data:
item = {
"project_id": s['project_id'],
"project_name": s['project_name'],
'role':s['role'],
"top_doc": 0,
'type':'project',
'create_time':s['create_time'],
'username':s['username'],
# "checkArr": "0"
}
colla_doc_list.append(item)
for doc in doc_data:
doc_list = Doc.objects.filter(
top_doc=doc.project.id,
create_user=request.user
).defer('content','pre_content')
if doc_list.exists():
for d in doc_list:
item = {
"project_id": d.id,
"project_name": d.name,
"top_doc": d.top_doc,
'role':None,
'type':'doc',
'create_time':d.create_time,
'username':d.create_user.username,
}
colla_doc_list.append(item)
resp = {
'code': 0,
'data': colla_doc_list,
# 'data':a,
'count': doc_data.count()
}
return Response(resp)
# 转让文集
@login_required()
@require_http_methods(['GET',"POST"])
def manage_project_transfer(request,pro_id):
try:
pro = Project.objects.get(id=pro_id)
except ObjectDoesNotExist:
return Http404
if (pro.create_user != request.user) and (request.user.is_superuser is False):
return render(request,'403.html')
else:
if request.method == 'GET':
user_list = User.objects.filter(~Q(username=request.user.username))
return render(request,'app_doc/manage/manage_project_transfer.html',locals())
elif request.method == 'POST':
user_name = request.POST.get('username',None)
try:
transfer_user = User.objects.get(username=user_name)
init_user = pro.create_user
# 修改文集的创建者
pro.create_user = transfer_user
pro.save()
# 修改文集文档的创建者
Doc.objects.filter(create_user=init_user,top_doc=pro_id).update(
create_user=transfer_user
)
return JsonResponse({'status':True,'data':'ok'})
except:
return JsonResponse({'status':False,'data':_('用户不存在')})
# 文档浏览页
@require_http_methods(['GET'])
def doc(request,pro_id,doc_id):
try:
if pro_id != '' and doc_id != '':
# 获取文集信息
doc = Doc.objects.get(id=int(doc_id),status__in=[0,1]) # 文档信息
pro_id = doc.top_doc
project = Project.objects.get(id=int(pro_id))
# 获取文集的文档目录
toc_list,toc_cnt = get_pro_toc(pro_id)
# 获取文集的协作用户信息
if request.user.is_authenticated:
colla_user = ProjectCollaborator.objects.filter(project=project,user=request.user)
if colla_user.exists():
colla_user_role = colla_user[0].role
colla_user = colla_user.count()
else:
colla_user = colla_user.count()
else:
colla_user = 0
# 获取文集收藏状态
if request.user.is_authenticated:
is_collect_pro = MyCollect.objects.filter(collect_type=2, collect_id=pro_id,
create_user=request.user).exists()
# 获取文档收藏状态
is_collect_doc = MyCollect.objects.filter(collect_type=1, collect_id=doc_id,
create_user=request.user).exists()
else:
is_collect_pro,is_collect_doc = False,False
# 私密文集且访问者非创建者、协作者 - 不能访问
if (project.role == 1) and (request.user != project.create_user) and (colla_user == 0):
return render(request, '404.html')
# 指定用户可见文集
elif project.role == 2:
user_list = project.role_value
if request.user.is_authenticated: # 认证用户判断是否在许可用户列表中
if (request.user.username not in user_list) and \
(request.user != project.create_user) and \
(colla_user == 0): # 访问者不在指定用户之中,也不是协作者
return render(request, '404.html')
else: # 游客直接返回404
return render(request, '404.html')
# 访问码可见
elif project.role == 3:
# 浏览用户不为创建者和协作者 - 需要访问码
if (request.user != project.create_user) and (colla_user == 0):
viewcode = project.role_value
viewcode_name = 'viewcode-{}'.format(project.id)
r_viewcode = request.COOKIES[
viewcode_name] if viewcode_name in request.COOKIES.keys() else 0 # 从cookie中获取访问码
if viewcode != r_viewcode: # cookie中的访问码不等于文集访问码跳转到访问码认证界面
return redirect('/check_viewcode/?to={}'.format(request.path))
# 获取文档内容
try:
doc = Doc.objects.get(id=int(doc_id),status__in=[0,1]) # 文档信息
doc_tags = DocTag.objects.filter(doc=doc) # 文档标签信息
if doc.status == 0 and doc.create_user != request.user:
raise ObjectDoesNotExist
elif doc.status == 0 and doc.create_user == request.user:
doc.name = _('【预览草稿】')+ doc.name
except ObjectDoesNotExist:
return render(request, '404.html')
# 获取文档分享信息
try:
doc_share = DocShare.objects.get(doc=doc)
is_share = True
except ObjectDoesNotExist:
is_share = False
# 获取文集下一级文档
# project_docs = Doc.objects.filter(top_doc=doc.top_doc, parent_doc=0, status=1).order_by('sort')
return render(request,'app_doc/doc.html',locals())
else:
return HttpResponse(_('参数错误'))
except Exception as e:
logger.exception(_("文集浏览出错"))
return render(request,'404.html')
# 创建文档
@login_required()
@require_http_methods(['GET',"POST"])
@logger.catch()
def create_doc(request):
# 获取用户的编辑器模式
try:
user_opt = UserOptions.objects.get(user=request.user)
editor_mode = user_opt.editor_mode
except ObjectDoesNotExist:
editor_mode = 1
if request.method == 'GET':
# 获取url切换的编辑器模式
eid = request.GET.get('eid',editor_mode)
if eid in [1,2,3,4,'1','2','3','4']:
editor_mode = int(eid)
try:
editor_type = _("新建表格") if editor_mode == 4 else _("新建文档")
pid = request.GET.get('pid',-999)
project_list = Project.objects.filter(create_user=request.user) # 自己创建的文集列表
colla_project_list = ProjectCollaborator.objects.filter(user=request.user) # 协作的文集列表
doctemp_list = DocTemp.objects.filter(create_user=request.user).values('id','name','create_time')
return render(request, 'app_doc/editor/create_doc.html', locals())
except Exception as e:
logger.exception(_("访问创建文档页面出错"))
return render(request,'404.html')
elif request.method == 'POST':
try:
project = request.POST.get('project','') # 文集ID
parent_doc = request.POST.get('parent_doc','') # 上级文档ID
doc_name = request.POST.get('doc_name','') # 文档标题
doc_tags = request.POST.get('doc_tag','') # 文档标签
doc_content = request.POST.get('content','') # 文档内容
pre_content = request.POST.get('pre_content','') # 文档Markdown内容
sort = request.POST.get('sort','') # 文档排序
editor_mode = request.POST.get('editor_mode',editor_mode) #获取文档编辑器
status = request.POST.get('status',1) # 文档状态
open_children = request.POST.get('open_children', False) # 展示下级目录
show_children = request.POST.get('show_children', False) # 展示下级目录
if open_children == 'on':
open_children = True
else:
open_children = False
if show_children == 'on':
show_children = True
else:
show_children = False
if project != '' and doc_name != '' and project != '-1':
# 验证请求者是否有文集的权限
check_project = Project.objects.filter(id=project,create_user=request.user)
colla_project = ProjectCollaborator.objects.filter(project=project,user=request.user)
if check_project.count() > 0 or colla_project.count() > 0:
# 判断文集下是否存在同名文档
if Doc.objects.filter(name=doc_name,top_doc=int(project)).exists():
return JsonResponse({'status':False,'data':_('文集内不允许同名文档')})
# 开启事务
with transaction.atomic():
save_id = transaction.savepoint()
try:
# 创建文档
doc = Doc.objects.create(
name=doc_name,
content = doc_content,
pre_content= pre_content,
parent_doc= int(parent_doc) if parent_doc != '' else 0,
top_doc= int(project),
sort = sort if sort != '' else 99,
create_user=request.user,
status = status,
editor_mode = editor_mode,
open_children = open_children,
show_children = show_children
)
# 设置文档标签
for t in doc_tags.split(","):
if t != '':
tag = Tag.objects.get_or_create(name=t,create_user=request.user)
DocTag.objects.get_or_create(tag=tag[0],doc=doc)
return JsonResponse({'status': True, 'data': {'pro': project, 'doc': doc.id}})
except Exception as e:
logger.exception(_("创建文档异常"))
# 回滚事务
transaction.savepoint_rollback(save_id)
transaction.savepoint_commit(save_id)
return JsonResponse({'status': False, 'data': _('创建失败')})
else:
return JsonResponse({'status':False,'data':_('无权操作此文集')})
else:
return JsonResponse({'status':False,'data':_('请确认文档标题、文集正确')})
except Exception as e:
logger.exception("创建文档出错")
return JsonResponse({'status':False,'data':_('请求出错')})
else:
return JsonResponse({'status':False,'data':_('方法不允许')})
# 修改文档
@login_required()
@require_http_methods(['GET',"POST"])
def modify_doc(request,doc_id):
editor_type = _("修改文档")
if request.method == 'GET':
try:
doc = Doc.objects.get(id=doc_id) # 查询文档信息
editor_mode = doc.editor_mode # 设置文档编辑器为文档上一次使用的编辑模式
eid = request.GET.get('eid',editor_mode)
if eid in [1,2,3,'1','2','3']:
editor_mode = int(eid)
doc_tags = ','.join([i.tag.name for i in DocTag.objects.filter(doc=doc)]) # 查询文档标签信息
project = Project.objects.get(id=doc.top_doc) # 查询文档所属的文集信息
pro_colla = ProjectCollaborator.objects.filter(project=project,user=request.user) # 查询用户的协作文集信息
if pro_colla.count() == 0:
is_pro_colla = False
elif pro_colla[0].role == 1:
is_pro_colla = True
else:
is_pro_colla = False
project_list = Project.objects.filter(create_user=request.user) # 自己创建的文集列表
colla_project_list = ProjectCollaborator.objects.filter(user=request.user) # 协作的文集列表
# 判断用户是否有权限进行修改
if (request.user == doc.create_user) or \
(is_pro_colla is True) or \
(request.user == project.create_user):
doc_list = Doc.objects.filter(top_doc=project.id)
doctemp_list = DocTemp.objects.filter(create_user=request.user)
history_list = DocHistory.objects.filter(doc=doc).order_by('-create_time')
return render(request, 'app_doc/editor/modify_doc.html', locals())
else:
return render(request,'403.html')
except Exception as e:
logger.exception(_("修改文档页面访问出错"))
return render(request,'404.html')
elif request.method == 'POST':
try:
doc_id = request.POST.get('doc_id','') # 文档ID
project_id = request.POST.get('project', '') # 文集ID
parent_doc = request.POST.get('parent_doc', '') # 上级文档ID
doc_name = request.POST.get('doc_name', '') # 文档名称
doc_tags = request.POST.get('doc_tag','') # 文档标签
doc_content = request.POST.get('content', '') # 文档内容
pre_content = request.POST.get('pre_content', '') # 文档Markdown格式内容
sort = request.POST.get('sort', '') # 文档排序
editor_mode = request.POST.get('editor_mode',1) #获取文档编辑器
status = request.POST.get('status',1) # 文档状态
open_children = request.POST.get('open_children',False) # 展示下级目录
show_children = request.POST.get('show_children', False) # 展示下级目录
if open_children == 'on':
open_children = True
else:
open_children = False
if show_children == 'on':
show_children = True
else:
show_children = False
if doc_id != '' and project_id != '' and doc_name != '' and project_id != '-1':
doc = Doc.objects.get(id=doc_id)
project = Project.objects.get(id=project_id)
pro_colla = ProjectCollaborator.objects.filter(project=project, user=request.user)
if pro_colla.count() == 0:
is_pro_colla = False
elif pro_colla[0].role == 1:
is_pro_colla = True
else:
is_pro_colla = False
# 验证用户有权限修改文档 - 文档的创建者或文集的高级协作者
if (request.user == doc.create_user) or (is_pro_colla is True) or (request.user == project.create_user):
# 开启事务
with transaction.atomic():
save_id = transaction.savepoint()
try:
# 将现有文档内容写入到文档历史中
DocHistory.objects.create(
doc = doc,
pre_content = doc.pre_content,
create_user = request.user
)
# 更新文档内容
Doc.objects.filter(id=int(doc_id)).update(
name=doc_name,
content=doc_content,
pre_content=pre_content,
parent_doc=int(parent_doc) if parent_doc != '' else 0,
sort=sort if sort != '' else 99,
modify_time = datetime.datetime.now(),
status = status,
editor_mode = editor_mode,
open_children = open_children,
show_children = show_children
)
# 更新文档标签
doc_tag_list = doc_tags.split(",") if doc_tags != "" else []
# print(doc_tags,doc_tag_list)
# 如果没有设置标签,则删除此文档的所有标签
if len(doc_tag_list) == 0:
DocTag.objects.filter(doc=doc).delete()
else:
current_doc_tags = [i.tag.name for i in DocTag.objects.filter(doc=doc)] # 获取当前文档的标签
# 遍历当前文档标签,如果不在新的标签列表,则删除
for tag in current_doc_tags:
if tag not in doc_tag_list:
tag = Tag.objects.get(name=tag,create_user=request.user)
DocTag.objects.filter(doc=doc,tag=tag).delete()
# 遍历新的标签列表,如果不在当前文档标签中,则创建
for t in doc_tag_list:
if t not in current_doc_tags and current_doc_tags != '':
tag = Tag.objects.get_or_create(name=t, create_user=request.user)
DocTag.objects.get_or_create(tag=tag[0], doc=doc)
return JsonResponse({'status': True, 'data': _('修改成功')})
except:
logger.exception(_("修改文档异常"))
# 回滚事务
transaction.savepoint_rollback(save_id)
transaction.savepoint_commit(save_id)
return JsonResponse({'status': False, 'data': _('修改失败')})
else:
return JsonResponse({'status':False,'data':_('未授权请求')})
else:
return JsonResponse({'status': False,'data':_('参数错误')})
except Exception as e:
logger.exception(_("修改文档出错"))
return JsonResponse({'status':False,'data':_('请求出错')})
# 删除文档 - 软删除 - 进入回收站
@login_required()
@require_http_methods(["POST"])
def del_doc(request):
try:
# 获取文档ID
doc_id = request.POST.get('doc_id',None)
range = request.POST.get('range', 'single')
if doc_id:
if range == 'single':
# 查询文档
try:
doc = Doc.objects.get(id=doc_id)
try:
project = Project.objects.get(id=doc.top_doc) # 查询文档所属的文集
except ObjectDoesNotExist:
logger.error(_("文档{}的所属文集不存在。".format(doc_id)))
project = 0
# 获取文档所属文集的协作信息
pro_colla = ProjectCollaborator.objects.filter(project=project,user=request.user)
if pro_colla.exists():
colla_user_role = pro_colla[0].role
else:
colla_user_role = 0
except ObjectDoesNotExist:
return JsonResponse({'status': False, 'data': '文档不存在'})
# 如果请求用户为站点管理员、文档创建者、高级权限的协作者、文集的创建者,可以删除
if (request.user == doc.create_user) \
or (colla_user_role == 1) \
or (request.user == project.create_user)\
or (request.user.is_superuser):
# 修改状态为删除
doc.status = 3
doc.modify_time = datetime.datetime.now()
doc.save()
# 修改其下级所有文档状态为删除
chr_doc = Doc.objects.filter(parent_doc=doc_id) # 获取下级文档
chr_doc_ids = chr_doc.values_list('id',flat=True) # 提取下级文档的ID
chr_doc.update(status=3,modify_time=datetime.datetime.now()) # 修改下级文档的状态为删除
Doc.objects.filter(parent_doc__in=list(chr_doc_ids)).update(status=3,modify_time=datetime.datetime.now()) # 修改下级文档的下级文档状态
return JsonResponse({'status': True, 'data': _('删除完成')})
else:
return JsonResponse({'status': False, 'data': _('非法请求')})
elif range == 'multi':
docs = doc_id.split(",")
try:
Doc.objects.filter(id__in=docs,create_user=request.user).update(status=3,modify_time=datetime.datetime.now())
Doc.objects.filter(parent_doc__in=docs).update(status=3,modify_time=datetime.datetime.now())
return JsonResponse({'status': True, 'data': _('删除完成')})
except:
return JsonResponse({'status': False, 'data': _('非法请求')})
else:
return JsonResponse({'status': False, 'data': _('类型错误')})
else:
return JsonResponse({'status':False,'data':_('参数错误')})
except Exception as e:
logger.exception(_("删除文档出错"))
return JsonResponse({'status':False,'data':_('请求出错')})
# 管理文档
@login_required()
@require_http_methods(['GET','POST'])
def manage_doc(request):
if request.method == 'GET':
# 文集列表
project_list = Project.objects.filter(create_user=request.user) # 自己创建的文集列表
colla_project_list = ProjectCollaborator.objects.filter(user=request.user) # 协作的文集列表
# 文档数量
# 已发布文档数量
published_doc_cnt = Doc.objects.filter(create_user=request.user, status=1).count()
# 草稿文档数量
draft_doc_cnt = Doc.objects.filter(create_user=request.user, status=0).count()
# 所有文档数量
all_cnt = published_doc_cnt + draft_doc_cnt
return render(request,'app_doc/manage/manage_doc.html',locals())
else:
kw = request.POST.get('kw', '')
project = request.POST.get('project','')
status = request.POST.get('status','')
if status == '-1': # 全部文档
q_status = [0,1]
elif status in ['0','1']:
q_status = [int(status)]
else:
q_status = [0, 1]
if project == '':
project_list = Project.objects.filter(create_user=request.user).values_list('id',flat=True) # 自己创建的文集列表
colla_project_list = ProjectCollaborator.objects.filter(user=request.user).values_list('project__id',flat=True) # 协作的文集列表
q_project = list(project_list) + list(colla_project_list)
else:
q_project = [project]
page = request.POST.get('page', 1)
limit = request.POST.get('limit', 10)
# 没有搜索
if kw == '':
doc_list = Doc.objects.filter(
create_user=request.user,
status__in=q_status,
top_doc__in=q_project
).order_by('-modify_time')
# 有搜索
else:
doc_list = Doc.objects.filter(
Q(content__icontains=kw) | Q(name__icontains=kw),
create_user=request.user,status__in=q_status,top_doc__in=q_project
).order_by('-modify_time')
# 文集列表
project_list = Project.objects.filter(create_user=request.user) # 自己创建的文集列表
colla_project_list = ProjectCollaborator.objects.filter(user=request.user) # 协作的文集列表
# 文档数量
# 已发布文档数量
published_doc_cnt = Doc.objects.filter(create_user=request.user, status=1).count()
# 草稿文档数量
draft_doc_cnt = Doc.objects.filter(create_user=request.user, status=0).count()
# 所有文档数量
all_cnt = published_doc_cnt + draft_doc_cnt
# 分页处理
paginator = Paginator(doc_list, limit)
page = request.GET.get('page', page)
try:
docs = paginator.page(page)
except PageNotAnInteger:
docs = paginator.page(1)
except EmptyPage:
docs = paginator.page(paginator.num_pages)
table_data = []
for doc in docs:
item = {
'id': doc.id,
'name': doc.name,
'parent':Doc.objects.get(id=doc.parent_doc).name if doc.parent_doc != 0 else '',
'project_id': Project.objects.get(id=doc.top_doc).id,
'project_name':Project.objects.get(id=doc.top_doc).name,
'status':doc.status,
'editor_mode':doc.editor_mode,
'open_children':doc.open_children,
'create_time': doc.create_time,
'modify_time': doc.modify_time
}
table_data.append(item)
resp_data = {
"code": 0,
"msg": "ok",
"count": doc_list.count(),
"data": jsonXssFilter(table_data)
}
return JsonResponse(resp_data)
# 移动文档
@login_required()
@require_http_methods(['POST'])
def move_doc(request):
doc_id = request.POST.get('doc_id','') # 文档ID
pro_id = request.POST.get('pro_id','') # 移动的文集ID
move_type = request.POST.get('move_type','') # 移动的类型 0复制 1移动 2连同下级文档移动
parent_id = request.POST.get('parent_id',0)
# 判断文集是否存在且有权限
try:
project = Project.objects.get(id=int(pro_id)) # 自己的文集
colla = ProjectCollaborator.objects.filter(project=project, user=request.user) # 协作文集
if (project.create_user != request.user) and (colla.count() == 0) : # 文集创建者
print(project.create_user,request.user,colla.count())
return JsonResponse({'status':False,'data':_('文集无权限')})
except ObjectDoesNotExist:
return JsonResponse({'status':False,'data':_('文集不存在')})
# 判断文档是否存在
try:
doc = Doc.objects.get(id=int(doc_id),create_user=request.user)
except ObjectDoesNotExist:
return JsonResponse({'status':False,'data':_('文档不存在')})
# 判断上级文档是否存在
try:
if parent_id != '0':
parent = Doc.objects.get(id=int(parent_id),create_user=request.user)
except ObjectDoesNotExist:
return JsonResponse({'status':False,'data':_('上级文档不存在')})
# 复制文档
if move_type == '0':
copy_doc = Doc.objects.create(
name = doc.name,
pre_content = doc.pre_content,
content = doc.content,
parent_doc = parent_id,
top_doc = int(pro_id),
editor_mode = doc.editor_mode,
create_user = request.user,
create_time = datetime.datetime.now(),
modify_time = datetime.datetime.now(),
# 文档状态说明0表示草稿状态1表示发布状态
status = doc.status
)
return JsonResponse({'status':True,'data':{'pro_id':pro_id,'doc_id':copy_doc.id}})
# 移动文档,下级文档更改到根目录
elif move_type == '1':
try:
# 修改文档的所属文集和上级文档实现移动文档
Doc.objects.filter(id=int(doc_id)).update(parent_doc=int(parent_id),top_doc=int(pro_id))
# 修改其子文档为顶级文档
Doc.objects.filter(parent_doc=doc_id).update(parent_doc=0)
return JsonResponse({'status':True,'data':{'pro_id':pro_id,'doc_id':doc_id}})
except:
logger.exception(_("移动文档异常"))
return JsonResponse({'status':False,'data':_('移动文档失败')})
# 包含下级文档一起移动
elif move_type == '2':
try:
# 修改文档的所属文集和上级文档实现移动文档
Doc.objects.filter(id=int(doc_id)).update(parent_doc=int(parent_id), top_doc=int(pro_id))
# 修改其子文档的文集归属
child_doc = Doc.objects.filter(parent_doc=doc_id)
child_doc.update(top_doc=int(pro_id))
# 遍历子文档,如果其存在下级文档,那么继续修改所属文集
for child in child_doc:
Doc.objects.filter(parent_doc=child.id).update(top_doc=int(pro_id))
return JsonResponse({'status': True, 'data':{'pro_id':pro_id,'doc_id':doc_id}})
except:
logger.exception(_("移动包含下级的文档异常"))
return JsonResponse({'status': False, 'data': _('移动文档失败')})
else:
return JsonResponse({'status':False,'data':_('移动类型错误')})
# 查看对比文档历史版本
@login_required()
@require_http_methods(['GET',"POST"])
def diff_doc(request,doc_id,his_id):
if request.method == 'GET':
try:
doc = Doc.objects.get(id=doc_id) # 查询文档信息
project = Project.objects.get(id=doc.top_doc) # 查询文档所属的文集信息
pro_colla = ProjectCollaborator.objects.filter(project=project, user=request.user) # 查询用户的协作文集信息
if (request.user == doc.create_user) or (pro_colla[0].role == 1) or (request.user.is_superuser):
history = DocHistory.objects.get(id=his_id)
history_list = DocHistory.objects.filter(doc=doc).order_by('-create_time')
if history.doc == doc:
return render(request, 'app_doc/diff_doc.html', locals())
else:
return render(request, '403.html')
else:
return render(request, '403.html')
except Exception as e:
logger.exception(_("文档历史版本页面访问出错"))
return render(request, '404.html')
elif request.method == 'POST':
try:
doc = Doc.objects.get(id=doc_id) # 查询文档信息
project = Project.objects.get(id=doc.top_doc) # 查询文档所属的文集信息
pro_colla = ProjectCollaborator.objects.filter(project=project, user=request.user) # 查询用户的协作文集信息
if (request.user == doc.create_user) or (pro_colla[0].role == 1) or (request.user.is_superuser):
history = DocHistory.objects.get(id=his_id)
if history.doc == doc:
return JsonResponse({'status':True,'data':history.pre_content})
else:
return JsonResponse({'status': False, 'data': _('非法请求')})
else:
return JsonResponse({'status':False,'data':_('非法请求')})
except Exception as e:
logger.exception(_("文档历史版本获取出错"))
return JsonResponse({'status':False,'data':_('获取异常')})
# 管理文档历史版本
@login_required()
@require_http_methods(['GET',"POST"])
def manage_doc_history(request,doc_id):
if request.method == 'GET':
try:
doc = Doc.objects.get(id=doc_id,create_user=request.user)
history_list = DocHistory.objects.filter(create_user=request.user,doc=doc_id).order_by('-create_time')
paginator = Paginator(history_list, 15)
page = request.GET.get('page', 1)
try:
historys = paginator.page(page)
except PageNotAnInteger:
historys = paginator.page(1)
except EmptyPage:
historys = paginator.page(paginator.num_pages)
return render(request, 'app_doc/manage/manage_doc_history.html', locals())
except Exception as e:
logger.exception(_("管理文档历史版本页面访问出错"))
return render(request, '404.html')
elif request.method == 'POST':
try:
history_id = request.POST.get('history_id','')
DocHistory.objects.filter(id=history_id,doc=doc_id,create_user=request.user).delete()
return JsonResponse({'status':True,'data':_('删除成功')})
except:
logger.exception(_("操作文档历史版本出错"))
return JsonResponse({'status':False,'data':_('出现异常')})
# 文档回收站
@login_required()
@require_http_methods(['GET','POST'])
def doc_recycle(request):
if request.method == 'GET':
# 获取状态为删除的文档
doc_list = Doc.objects.filter(status=3,create_user=request.user).order_by('-modify_time')
# 分页处理
paginator = Paginator(doc_list, 15)
page = request.GET.get('page', 1)
try:
docs = paginator.page(page)
except PageNotAnInteger:
docs = paginator.page(1)
except EmptyPage:
docs = paginator.page(paginator.num_pages)
return render(request,'app_doc/manage/manage_doc_recycle.html',locals())
elif request.method == 'POST':
try:
# 获取参数
doc_id = request.POST.get('doc_id', None) # 文档ID
types = request.POST.get('type',None) # 操作类型
if doc_id:
# 查询文档
try:
doc = Doc.objects.get(id=doc_id)
project = Project.objects.get(id=doc.top_doc) # 查询文档所属的文集
# 获取文档所属文集的协作信息
pro_colla = ProjectCollaborator.objects.filter(project=project, user=request.user) #
if pro_colla.exists():
colla_user_role = pro_colla[0].role
else:
colla_user_role = 0
except ObjectDoesNotExist:
return JsonResponse({'status': False, 'data': _('文档不存在')})
# 如果请求用户为文档创建者、高级权限的协作者、文集的创建者,可以操作
if (request.user == doc.create_user) or (colla_user_role == 1) or (request.user == project.create_user):
# 还原文档
if types == 'restore':
# 修改状态为草稿
doc.status = 0
doc.modify_time = datetime.datetime.now()
doc.save()
# 删除文档
elif types == 'del':
# 删除文档历史、分享、标签
DocHistory.objects.filter(doc=doc).delete()
DocShare.objects.filter(doc=doc).delete()
DocTag.objects.filter(doc=doc).delete()
# 删除文档
doc.delete()
else:
return JsonResponse({'status':False,'data':_('无效请求')})
return JsonResponse({'status': True, 'data': _('删除完成')})
else:
return JsonResponse({'status': False, 'data': _('非法请求')})
# 清空回收站
elif types == 'empty':
docs = Doc.objects.filter(status=3,create_user=request.user)
for doc in docs:
# 删除文档历史、分享、标签
DocHistory.objects.filter(doc=doc).delete()
DocShare.objects.filter(doc=doc).delete()
DocTag.objects.filter(doc=doc).delete()
docs.delete()
return JsonResponse({'status': True, 'data': _('清空成功')})
# 还原回收站
elif types == 'restoreAll':
Doc.objects.filter(status=3,create_user=request.user).update(status=0)
return JsonResponse({'status': True, 'data': _('还原成功')})
else:
return JsonResponse({'status': False, 'data': _('参数错误')})
except Exception as e:
logger.exception(_("处理文档出错"))
return JsonResponse({'status': False, 'data': _('请求出错')})
# 一键发布文档
@login_required()
@require_http_methods(['POST'])
def fast_publish_doc(request):
doc_id = request.POST.get('doc_id',None)
# 查询文档
try:
doc = Doc.objects.get(id=doc_id)
project = Project.objects.get(id=doc.top_doc) # 查询文档所属的文集
# 获取文档所属文集的协作信息
pro_colla = ProjectCollaborator.objects.filter(project=project, user=request.user) #
if pro_colla.exists():
colla_user_role = pro_colla[0].role
else:
colla_user_role = 0
except ObjectDoesNotExist:
return JsonResponse({'status': False, 'data': _('文档不存在')})
# 判断请求者是否有权限(文档创建者、文集创建者、文集高级协作者)
# 如果请求用户为文档创建者、高级权限的协作者、文集的创建者,可以删除
if (request.user == doc.create_user) or (colla_user_role == 1) or (request.user == project.create_user):
try:
doc.status = 1
doc.modify_time = datetime.datetime.now()
doc.save()
return JsonResponse({'status':True,'data':_('发布成功')})
except:
logger.exception(_("文档一键发布失败"))
return JsonResponse({'status':False,'data':_('发布失败')})
else:
return JsonResponse({'status':False,'data':_('非法请求')})
# 私密文档分享
@require_http_methods(['GET','POST'])
def share_doc(request):
if request.method == 'GET':
share_token = request.GET.get('token')
# 判断是否存在分享
try:
share_doc = DocShare.objects.get(token=share_token,is_enable=True)
doc = share_doc.doc
# 公开分享
if share_doc.share_type == 0:
return render(request, 'app_doc/share/share_doc.html', locals())
# 私密分享
else:
doc_id_base64 = base64.standard_b64encode(str(share_doc.doc.id).encode())
# 不存在公开分享的文档,则判断验证分享码
share_cookie_name = 'sharedoc-{}'.format(share_token)
share_value = request.COOKIES.get(share_cookie_name) if share_cookie_name in request.COOKIES.keys() else 0
if share_doc.share_value == share_value:
return render(request, 'app_doc/share/share_doc.html', locals())
else:
return redirect('/share_doc_check/?surl={}'.format(share_token))
except ObjectDoesNotExist:
return render(request,'404.html')
elif request.method == 'POST':
doc_id = request.POST.get('id')
try:
# 获取请求参数
doc = Doc.objects.get(id=doc_id,create_user=request.user)
share_type = request.POST.get('share_type',0)
share_value = request.POST.get('share_value',0)
is_enable = request.POST.get('is_enable',True)
if is_enable == 'false':
is_enable = False
else:
is_enable = True
# 生成分享文档Token
share_token = hashlib.md5()
share_token.update("{}_{}".format(doc_id,request.user.username).encode())
share_token = share_token.hexdigest()
# 创建或更新分享信息
doc_share = DocShare.objects.update_or_create(
token=share_token,
defaults={'doc': doc,
'share_type': share_type,
'share_value':share_value,
'is_enable':is_enable
}
)
if int(share_type) == 0:
data = {
'doc':share_token
}
else:
data = {
'doc': share_token,
'share_value':share_value
}
return JsonResponse({'status':True,'data':data})
except ObjectDoesNotExist:
return JsonResponse({'status':False,'data':_('文档不存在')})
# 验证文档分享码
def share_doc_check(request):
doc_token = request.GET.get('surl', '')
if request.method == 'GET':
if doc_token != '':
doc_share = DocShare.objects.get(token=doc_token)
share_cookie_name = 'sharedoc-{}'.format(doc_token)
share_value = request.COOKIES.get(share_cookie_name) if share_cookie_name in request.COOKIES.keys() else 0
if doc_share.share_value == share_value:
return redirect("/share_doc/?token={}".format(doc_token))
else:
return render(request,'app_doc/share/share_check.html',locals())
else:
return render(request,'404.html')
else:
# 接收参数值
share_value = request.POST.get('share_value','')
# 查询数据
if DocShare.objects.filter(token=doc_token,share_type=1,share_value=share_value).exists():
obj = redirect("/share_doc/?token={}".format(doc_token))
obj.set_cookie('sharedoc-{}'.format(doc_token),share_value)
return obj
else:
errormsg = _("分享码错误")
return render(request, 'app_doc/share/share_check.html', locals())
# 管理文档分享
@login_required()
@require_http_methods(['GET','POST'])
def manage_doc_share(request):
if request.method == 'GET':
return render(request, 'app_doc/manage/manage_doc_share.html', locals())
else:
types = request.POST.get('type')
# 请求类型 1获取列表 2删除 3修改
if types == '1':
page = request.POST.get('page', 1)
limit = request.POST.get('limit', 10)
# share_doc = DocShare.objects.filter(doc__create_user=request.user).order_by('-create_time')
docshare_list = DocShare.objects.filter(doc__create_user=request.user).order_by('-create_time')
paginator = Paginator(docshare_list, limit)
page = request.GET.get('page', page)
try:
docshares = paginator.page(page)
except PageNotAnInteger:
docshares = paginator.page(1)
except EmptyPage:
docshares = paginator.page(paginator.num_pages)
share_list = []
for doc in docshares:
item = {
'token':doc.token,
'doc_name':doc.doc.name,
'share_type':doc.share_type,
'share_value':doc.share_value,
'share_status':doc.is_enable,
'create_time':doc.create_time
}
share_list.append(item)
resp_data = {
"code":0,
"msg":"ok",
"count":docshare_list.count(),
"data":share_list
}
return JsonResponse(resp_data)
# 删除
elif types == '2':
range = request.POST.get("range")
token = request.POST.get("token")
if range == 'single':
try:
share = DocShare.objects.get(token=token,doc__create_user=request.user)
share.delete()
return JsonResponse({'status':True,'data':'ok'})
except:
return JsonResponse({'status':False,'data':_('无指定内容')})
elif range == "multi":
tokens = token.split(",")
try:
share = DocShare.objects.filter(token__in=tokens,doc__create_user=request.user)
share.delete()
return JsonResponse({'status':True,'data':'ok'})
except:
return JsonResponse({'status':False,'data':_('无指定内容')})
else:
return JsonResponse({'status':False,'data':_('类型错误')})
# 修改
elif types == '3':
token = request.POST.get("token",'')
name = request.POST.get('key','')
value = request.POST.get('value','')
# 修改分享状态
if name == 'share_status':
is_enable = True if value == 'true' else False
DocShare.objects.filter(token=token).update(is_enable=is_enable)
# 修改分享类型
elif name == 'share_type':
share_type = 0 if value == '0' else 1
DocShare.objects.filter(token=token).update(share_type=share_type)
else:
return JsonResponse({'status':False,'data':_('参数错误')})
return JsonResponse({'status':True,'data':'ok'})
# 创建文档模板
@login_required()
@require_http_methods(['GET',"POST"])
def create_doctemp(request):
if request.method == 'GET':
editor_type = _("新建文档模板")
# 获取用户的编辑器模式
try:
user_opt = UserOptions.objects.get(user=request.user)
editor_mode = user_opt.editor_mode
except ObjectDoesNotExist:
editor_mode = 1
doctemps = DocTemp.objects.filter(create_user=request.user)
return render(request,'app_doc/editor/create_doctemp.html',locals())
elif request.method == 'POST':
try:
name = request.POST.get('name','')
content = request.POST.get('content','')
if name != '':
doctemp = DocTemp.objects.create(
name = name,
content = content,
create_user=request.user
)
doctemp.save()
return JsonResponse({'status':True,'data':doctemp.id})
else:
return JsonResponse({'status':False,'data':_('模板标题不能为空')})
except Exception as e:
logger.exception(_("创建文档模板出错"))
return JsonResponse({'status':False,'data':_('请求出错')})
# 修改文档模板
@login_required()
@require_http_methods(['GET',"POST"])
def modify_doctemp(request,doctemp_id):
if request.method == 'GET':
editor_type = _('修改文档模板')
try:
doctemp = DocTemp.objects.get(id=doctemp_id)
if request.user.id == doctemp.create_user.id:
# 获取用户的编辑器模式
try:
user_opt = UserOptions.objects.get(user=request.user)
editor_mode = user_opt.editor_mode
except ObjectDoesNotExist:
editor_mode = 1
doctemps = DocTemp.objects.filter(create_user=request.user)
return render(request,'app_doc/editor/modify_doctemp.html',locals())
else:
return HttpResponse(_('非法请求'))
except Exception as e:
logger.exception(_("访问文档模板修改页面出错"))
return render(request, '404.html')
elif request.method == 'POST':
try:
doctemp_id = request.POST.get('doctemp_id','')
name = request.POST.get('name','')
content = request.POST.get('content','')
if doctemp_id != '' and name !='':
doctemp = DocTemp.objects.get(id=doctemp_id)
if request.user.id == doctemp.create_user.id:
doctemp.name = name
doctemp.content = content
doctemp.save()
return JsonResponse({'status':True,'data':_('修改成功')})
else:
return JsonResponse({'status':False,'data':_('非法操作')})
else:
return JsonResponse({'status':False,'data':_('参数错误')})
except Exception as e:
logger.exception(_("修改文档模板出错"))
return JsonResponse({'status':False,'data':_('请求出错')})
# 删除文档模板
@login_required()
def del_doctemp(request):
try:
doctemp_id = request.POST.get('doctemp_id','')
if doctemp_id != '':
doctemp = DocTemp.objects.get(id=doctemp_id)
if request.user.id == doctemp.create_user.id:
doctemp.delete()
return JsonResponse({'status':True,'data':_('删除完成')})
else:
return JsonResponse({'status':False,'data':_('非法请求')})
else:
return JsonResponse({'status': False, 'data': _('参数错误')})
except Exception as e:
logger.exception(_("删除文档模板出错"))
return JsonResponse({'status':False,'data':_('请求出错')})
# 管理文档模板
@login_required()
@require_http_methods(['GET'])
def manage_doctemp(request):
try:
search_kw = request.GET.get('kw', None)
if search_kw:
doctemp_list = DocTemp.objects.filter(
create_user=request.user,
content__icontains=search_kw
).order_by('-modify_time')
paginator = Paginator(doctemp_list, 10)
page = request.GET.get('page', 1)
try:
doctemps = paginator.page(page)
except PageNotAnInteger:
doctemps = paginator.page(1)
except EmptyPage:
doctemps = paginator.page(paginator.num_pages)
doctemps.kw = search_kw
else:
doctemp_list = DocTemp.objects.filter(create_user=request.user).order_by('-modify_time')
paginator = Paginator(doctemp_list, 10)
page = request.GET.get('page', 1)
try:
doctemps = paginator.page(page)
except PageNotAnInteger:
doctemps = paginator.page(1)
except EmptyPage:
doctemps = paginator.page(paginator.num_pages)
return render(request, 'app_doc/manage/manage_doctemp.html', locals())
except Exception as e:
logger.exception(_("管理文档模板页面访问出错"))
return render(request, '404.html')
# 获取指定文档模板
@login_required()
@require_http_methods(["POST"])
def get_doctemp(request):
try:
doctemp_id = request.POST.get('doctemp_id','')
if doctemp_id != '':
content = DocTemp.objects.get(id=int(doctemp_id)).serializable_value('content')
return JsonResponse({'status':True,'data':content})
else:
return JsonResponse({'status':False,'data':_('参数错误')})
except Exception as e:
logger.exception(_("获取指定文档模板出错"))
return JsonResponse({'status':False,'data':_('请求出错')})
# 获取指定文集的所有文档
@require_http_methods(["POST"])
@logger.catch()
def get_pro_doc(request):
pro_id = request.POST.get('pro_id','')
if pro_id != '':
# 获取文集所有文档的id、name和parent_doc3个字段
doc_list = Doc.objects.filter(top_doc=int(pro_id),status=1).values_list('id','name','parent_doc').order_by('parent_doc')
item_list = []
# 遍历文档
for doc in doc_list:
# 如果文档为顶级文档
if doc[2] == 0:
# 将其数据添加到列表中
item = [
doc[0],doc[1],doc[2],''
]
item_list.append(item)
# 如果文档不是顶级文档
else:
# 查询文档的上级文档
try:
parent = Doc.objects.get(id=doc[2])
except ObjectDoesNotExist:
return JsonResponse({'status':False,'data':'文档id不存在'})
# 如果文档上级文档的上级是顶级文档,那么将其添加到列表
if parent.parent_doc == 0: # 只要二级目录
item = [
doc[0],doc[1],doc[2],parent.name+' --> '
]
item_list.append(item)
return JsonResponse({'status':True,'data':list(item_list)})
else:
return JsonResponse({'status':False,'data':_('参数错误')})
# 获取指定文集的文档树数据
@require_http_methods(['POST'])
@logger.catch()
def get_pro_doc_tree(request):
pro_id = request.POST.get('pro_id', None)
if pro_id:
# 查询存在上级文档的文档
parent_id_list = Doc.objects.filter(top_doc=pro_id,status=1).exclude(parent_doc=0).values_list('parent_doc',flat=True)
# 获取存在上级文档的上级文档ID
# print(parent_id_list)
doc_list = []
# 获取一级文档
top_docs = Doc.objects.filter(top_doc=pro_id,parent_doc=0,status=1).values('id','name','modify_time').order_by('sort')
# 遍历一级文档
for doc in top_docs:
top_item = {
'id':doc['id'],
'field':doc['name'],
'title':doc['name'],
'modify_time':doc['modify_time'],
'spread':True,
'level':1
}
# 如果一级文档存在下级文档,查询其二级文档
if doc['id'] in parent_id_list:
# 获取二级文档
sec_docs = Doc.objects.filter(top_doc=pro_id,parent_doc=doc['id'],status=1).values('id','name','modify_time').order_by('sort')
top_item['children'] = []
for doc in sec_docs:
sec_item = {
'id': doc['id'],
'field': doc['name'],
'title': doc['name'],
'modify_time': doc['modify_time'],
'level':2
}
# 如果二级文档存在下级文档,查询第三级文档
if doc['id'] in parent_id_list:
# 获取三级文档
thr_docs = Doc.objects.filter(top_doc=pro_id,parent_doc=doc['id'],status=1).values('id','name','modify_time').order_by('sort')
sec_item['children'] = []
for doc in thr_docs:
item = {
'id': doc['id'],
'field': doc['name'],
'title': doc['name'],
'modify_time': doc['modify_time'],
'level': 3
}
sec_item['children'].append(item)
top_item['children'].append(sec_item)
else:
top_item['children'].append(sec_item)
doc_list.append(top_item)
# 如果一级文档没有下级文档,直接保存
else:
doc_list.append(top_item)
doc_list = jsonXssFilter(doc_list)
return JsonResponse({'status':True,'data':doc_list})
else:
return JsonResponse({'status':False,'data':_('参数错误')})
# 404页面
def handle_404(request):
return render(request,'404.html')
# 导出文集MD文件
@login_required()
@require_http_methods(["POST"])
def report_md(request):
pro_id = request.POST.get('project_id','')
types = request.POST.get('type','single')
user = request.user
if types == 'single':
try:
Project.objects.get(id=int(pro_id),create_user=user)
project_md = ReportMD(
project_id=int(pro_id)
)
md_file_path = project_md.work() # 生成并获取MD文件压缩包绝对路径
md_file_filename = os.path.split(md_file_path)[-1] # 提取文件名
md_file = "/media/reportmd_temp/"+ md_file_filename # 拼接相对链接
return JsonResponse({'status':True,'data':md_file})
except ObjectDoesNotExist as e:
return JsonResponse({'status': False, 'data': _('文集不存在')})
except Exception as e:
logger.exception(_("导出文集MD文件出错"))
return JsonResponse({'status': False, 'data': _('导出文集异常')})
elif types == 'multi':
project_list = pro_id.split(',')
for project in project_list:
try:
Project.objects.get(id=project,create_user=request.user)
except ObjectDoesNotExist:
return JsonResponse({'status':False,'data':_('无权限')})
project_md = ReportMdBatch(
project_id_list = project_list,
username = request.user.username
)
try:
md_file_path = project_md.work() # 生成并获取MD文件压缩包绝对路径
except:
logger.exception("文集导出异常")
return JsonResponse({'status': False, 'data': _('文集导出异常')})
md_file_filename = os.path.split(md_file_path)[-1] # 提取文件名
md_file = "/media/reportmd_temp/" + md_file_filename # 拼接相对链接
return JsonResponse({'status': True, 'data': md_file})
else:
return JsonResponse({'status':False,'data':_('无效参数')})
# 生成文集文件 - 个人中心 - 文集管理
@login_required()
@require_http_methods(["POST"])
def genera_project_file(request):
report_type = request.POST.get('types',None) # 获取前端传入到导出文件类型参数
# 导出EPUB文件
pro_id = request.POST.get('pro_id')
try:
project = Project.objects.get(id=int(pro_id))
# 获取文集的协作用户信息
if request.user.is_authenticated:
colla_user = ProjectCollaborator.objects.filter(project=project, user=request.user)
if colla_user.exists():
colla_user_role = colla_user[0].role
colla_user = colla_user.count()
else:
colla_user = colla_user.count()
else:
colla_user = 0
# 公开的文集 - 可以直接导出
if project.role == 0:
allow_export = True
# 私密文集 - 非创建者和协作者不可导出
elif (project.role == 1):
if (request.user != project.create_user) and (colla_user == 0):
allow_export = False
else:
allow_export = True
# 指定用户可见文集 - 指定用户、文集创建者和协作者可导出
elif project.role == 2:
user_list = project.role_value
if request.user.is_authenticated: # 认证用户判断是否在许可用户列表中
if (request.user.username not in user_list) and \
(request.user != project.create_user) and \
(colla_user == 0): # 访问者不在指定用户之中,也不是协作者
allow_export = False
else:
allow_export = True
else: # 游客直接返回404
allow_export = False
# 访问码可见文集 - 文集创建者、协作者和通过验证即可导出
elif project.role == 3:
# 浏览用户不为创建者和协作者 - 需要访问码
if (request.user != project.create_user) and (colla_user == 0):
viewcode = project.role_value
viewcode_name = 'viewcode-{}'.format(project.id)
r_viewcode = request.COOKIES[
viewcode_name] if viewcode_name in request.COOKIES.keys() else 0 # 从cookie中获取访问码
if viewcode != r_viewcode: # cookie中的访问码不等于文集访问码不可导出
allow_export = False
else:
allow_export = True
else:
allow_export = True
else:
allow_export = False
# 允许被导出
if allow_export:
# 导出EPUB
if report_type in ['epub']:
try:
report_project = ReportEPUB(
project_id=project.id
).work()
# print(report_project)
report_file_path = report_project.split('media', maxsplit=1)[-1] # 导出文件的路径
epub_file = '/media' + report_file_path + '.epub' # 文件相对路径
# 查询文集是否存在导出文件
report_cnt = ProjectReportFile.objects.filter(project=project,file_type='epub')
# 存在文件删除
if report_cnt.count() != 0:
for r in report_cnt:
is_exist = os.path.exists(settings.BASE_DIR + r.file_path)
if is_exist:
os.remove(settings.BASE_DIR + r.file_path)
report_cnt.delete() # 删除数据库记录
# 创建数据库记录
ProjectReportFile.objects.create(
project=project,
file_type='epub',
file_name=epub_file,
file_path=epub_file
)
return JsonResponse({'status': True, 'data': epub_file})
except Exception as e:
logger.exception(_("生成EPUB出错"))
return JsonResponse({'status': False, 'data': _('生成出错')})
# 导出PDF
elif report_type in ['pdf']:
try:
report_project = ReportPDF(
project_id=project.id,
user_id=request.user.id
).work()
if report_project is False:
return JsonResponse({'status':False,'data':_('生成出错')})
report_file_path = report_project.split('media', maxsplit=1)[-1] # 导出文件的路径
pdf_file = '/media' + report_file_path # 文件相对路径
# 查询文集是否存在导出文件
report_cnt = ProjectReportFile.objects.filter(project=project, file_type='pdf')
# 存在文件删除
if report_cnt.count() != 0:
for r in report_cnt:
is_exist = os.path.exists(settings.BASE_DIR + r.file_path)
if is_exist:
os.remove(settings.BASE_DIR + r.file_path)
report_cnt.delete() # 删除数据库记录
# 创建数据库记录
ProjectReportFile.objects.create(
project=project,
file_type='pdf',
file_name=pdf_file,
file_path=pdf_file
)
return JsonResponse({'status': True, 'data': pdf_file})
except Exception as e:
logger.exception(_("生成出错"))
return JsonResponse({'status': False, 'data': _('生成出错')})
else:
return JsonResponse({'status': False, 'data': _('不支持的类型')})
# 不允许被导出
else:
return JsonResponse({'status':False,'data':_('无权限导出')})
except ObjectDoesNotExist:
return JsonResponse({'status':False,'data':_('文集不存在')})
except Exception as e:
logger.exception(_("生成文集文件出错"))
return JsonResponse({'status':False,'data':_('系统异常')})
# 获取文集前台导出文件
@allow_report_file
@require_http_methods(["POST"])
def report_file(request):
report_type = request.POST.get('types',None) # 获取前端传入到导出文件类型参数
pro_id = request.POST.get('pro_id')
try:
project = Project.objects.get(id=int(pro_id))
# 获取文集的协作用户信息
if request.user.is_authenticated:
colla_user = ProjectCollaborator.objects.filter(project=project, user=request.user)
if colla_user.exists():
colla_user_role = colla_user[0].role
colla_user = colla_user.count()
else:
colla_user = colla_user.count()
else:
colla_user = 0
# 公开的文集 - 可以直接导出
if project.role == 0:
allow_export = True
# 私密文集 - 非创建者和协作者不可导出
elif (project.role == 1):
if (request.user != project.create_user) and (colla_user == 0):
allow_export = False
else:
allow_export = True
# 指定用户可见文集 - 指定用户、文集创建者和协作者可导出
elif project.role == 2:
user_list = project.role_value
if request.user.is_authenticated: # 认证用户判断是否在许可用户列表中
if (request.user.username not in user_list) and \
(request.user != project.create_user) and \
(colla_user == 0): # 访问者不在指定用户之中,也不是协作者
allow_export = False
else:
allow_export = True
else: # 游客直接返回404
allow_export = False
# 访问码可见文集 - 文集创建者、协作者和通过验证即可导出
elif project.role == 3:
# 浏览用户不为创建者和协作者 - 需要访问码
if (request.user != project.create_user) and (colla_user == 0):
viewcode = project.role_value
viewcode_name = 'viewcode-{}'.format(project.id)
r_viewcode = request.COOKIES[
viewcode_name] if viewcode_name in request.COOKIES.keys() else 0 # 从cookie中获取访问码
if viewcode != r_viewcode: # cookie中的访问码不等于文集访问码不可导出
allow_export = False
else:
allow_export = True
else:
allow_export = True
else:
allow_export = False
# return JsonResponse({'status':False,'data':'不存在的文集权限'})
if allow_export:
# 导出EPUB文件
if report_type in ['epub']:
try:
try:
report_project = ProjectReportFile.objects.get(project=project,file_type='epub')
except ObjectDoesNotExist:
return JsonResponse({'status':False,'data':_('无可用文件,请联系文集创建者')})
# print(report_project)
return JsonResponse({'status': True, 'data': report_project.file_path})
except Exception as e:
return JsonResponse({'status': False, 'data': _('导出出错')})
# 导出PDF
elif report_type in ['pdf']:
try:
try:
report_project = ProjectReportFile.objects.get(project=project,file_type='pdf')
except ObjectDoesNotExist:
return JsonResponse({'status':False,'data':_('无可用文件,请联系文集创建者')})
# print(report_project)
return JsonResponse({'status': True, 'data': report_project.file_path})
except Exception as e:
return JsonResponse({'status': False, 'data': _('导出出错')})
else:
return JsonResponse({'status': False, 'data': _('不支持的类型')})
else:
return JsonResponse({'status':False,'data':_('无权限导出')})
except ObjectDoesNotExist:
return JsonResponse({'status':False,'data':_('文集不存在')})
except Exception as e:
logger.exception(_("获取文集前台导出文件出错"))
return JsonResponse({'status':False,'data':_('系统异常')})
# 图片素材管理
@login_required()
@require_http_methods(['GET',"POST"])
def manage_image(request):
# 获取图片
if request.method == 'GET':
try:
groups = ImageGroup.objects.filter(user=request.user) # 获取所有分组
all_img_cnt = Image.objects.filter(user=request.user).count()
no_group_cnt = Image.objects.filter(user=request.user,group_id=None).count() # 获取所有未分组的图片数量
g_id = int(request.GET.get('group', 0)) # 图片分组id
if int(g_id) == 0:
image_list = Image.objects.filter(user=request.user).order_by('-create_time') # 查询所有图片
elif int(g_id) == -1:
image_list = Image.objects.filter(user=request.user,group_id=None).order_by('-create_time') # 查询未分组的图片
else:
image_list = Image.objects.filter(user=request.user,group_id=g_id).order_by('-create_time') # 查询指定分组的图片
paginator = Paginator(image_list, 18)
page = request.GET.get('page', 1)
try:
images = paginator.page(page)
except PageNotAnInteger:
images = paginator.page(1)
except EmptyPage:
images = paginator.page(paginator.num_pages)
images.group = g_id
return render(request,'app_doc/manage/manage_image.html',locals())
except:
logger.exception(_("图片素材管理出错"))
return render(request,'404.html')
elif request.method == 'POST':
try:
img_id = request.POST.get('img_id','')
types = request.POST.get('types','') # 操作类型0表示删除1表示修改2表示获取
range = request.POST.get('range','single') # 操作范围 single 表示单个图片multi表示多个图片
# 删除图片
if int(types) == 0:
if range == 'single':
img = Image.objects.get(id=img_id)
if img.user != request.user:
return JsonResponse({'status': False, 'data': _('未授权请求')})
file_path = settings.BASE_DIR+img.file_path
is_exist = os.path.exists(file_path)
if is_exist:
os.remove(file_path)
img.delete() # 删除记录
elif range == 'multi':
imgs = img_id.split(',')
for i in imgs:
img = Image.objects.get(id=i)
if img.user != request.user:
logger.error(_("图片{}非法删除".format(i)))
break
file_path = settings.BASE_DIR + img.file_path
is_exist = os.path.exists(file_path)
if is_exist:
os.remove(file_path)
img.delete() # 删除记录
return JsonResponse({'status':True,'data':_('删除完成')})
# 移动图片分组
elif int(types) == 1:
if range == 'single':
group_id = request.POST.get('group_id',None)
if group_id is None:
Image.objects.filter(id=img_id,user=request.user).update(group_id=None)
else:
group = ImageGroup.objects.get(id=group_id,user=request.user)
Image.objects.filter(id=img_id,user=request.user).update(group_id=group)
elif range == 'multi':
imgs = img_id.split(',')
group_id = request.POST.get('group_id',None)
if group_id is None:
Image.objects.filter(id__in=imgs,user=request.user).update(group_id=None)
else:
group = ImageGroup.objects.get(id=group_id,user=request.user)
Image.objects.filter(id__in=imgs,user=request.user).update(group_id=group)
return JsonResponse({'status':True,'data':_('移动完成')})
# 获取图片
elif int(types) == 2:
group_id = request.POST.get('group_id', None) # 接受分组ID参数
if group_id is None: #
return JsonResponse({'status':False,'data':_('参数错误')})
elif int(group_id) == 0:
imgs = Image.objects.filter(user=request.user).order_by('-create_time')
elif int(group_id) == -1:
imgs = Image.objects.filter(user=request.user,group_id=None).order_by('-create_time')
else:
imgs = Image.objects.filter(user=request.user,group_id=group_id).order_by('-create_time')
img_list = []
for img in imgs:
item = {
'path':img.file_path,
'name':img.file_name,
}
img_list.append(item)
return JsonResponse({'status':True,'data':img_list})
else:
return JsonResponse({'status':False,'data':_('非法参数')})
except ObjectDoesNotExist:
return JsonResponse({'status':False,'data':_('图片不存在')})
except:
logger.exception(_("操作图片素材出错"))
return JsonResponse({'status':False,'data':_('程序异常')})
# 图片分组管理
@login_required()
@require_http_methods(['GET',"POST"])
@logger.catch()
def manage_img_group(request):
if request.method == 'GET':
groups = ImageGroup.objects.filter(user=request.user)
return render(request,'app_doc/manage/manage_image_group.html',locals())
# 操作分组
elif request.method == 'POST':
types = request.POST.get('types',None) # 请求类型0表示创建分组1表示修改分组2表示删除分组3表示获取分组
# 创建分组
if int(types) == 0:
group_name = request.POST.get('group_name', '')
if group_name not in ['',_('默认分组'),_('未分组')]:
ImageGroup.objects.get_or_create(
user = request.user,
group_name = group_name
)
return JsonResponse({'status':True,'data':'ok'})
else:
return JsonResponse({'status':False,'data':_('名称无效')})
# 修改分组
elif int(types) == 1:
group_name = request.POST.get("group_name",'')
if group_name not in ['',_('默认分组'),_('未分组')]:
group_id = request.POST.get('group_id', '')
ImageGroup.objects.filter(id=group_id,user=request.user).update(group_name=group_name)
return JsonResponse({'status':True,'data':_('修改成功')})
else:
return JsonResponse({'status':False,'data':_('名称无效')})
# 删除分组
elif int(types) == 2:
try:
group_id = request.POST.get('group_id','')
group = ImageGroup.objects.get(id=group_id,user=request.user) # 查询分组
images = Image.objects.filter(group_id=group_id,user=request.user).update(group_id=None) # 移动图片到未分组
group.delete() # 删除分组
return JsonResponse({'status':True,'data':_('删除完成')})
except:
logger.exception(_("删除图片分组出错"))
return JsonResponse({'status':False,'data':_('删除错误')})
# 获取分组
elif int(types) == 3:
try:
group_list = []
all_cnt = Image.objects.filter(user=request.user).count()
non_group_cnt = Image.objects.filter(group_id=None,user=request.user).count()
group_list.append({'group_name':_('全部图片'),'group_cnt':all_cnt,'group_id':0})
group_list.append({'group_name':_('未分组'),'group_cnt':non_group_cnt,'group_id':-1})
groups = ImageGroup.objects.filter(user=request.user) # 查询所有分组
for group in groups:
group_cnt = Image.objects.filter(group_id=group).count()
item = {
'group_id':group.id,
'group_name':group.group_name,
'group_cnt':group_cnt
}
group_list.append(item)
return JsonResponse({'status':True,'data':group_list})
except:
logger.exception(_("获取图片分组出错"))
return JsonResponse({'status':False,'data':_('出现错误')})
# 附件管理
@login_required()
@csrf_exempt
@require_http_methods(['GET',"POST"])
def manage_attachment(request):
# 文件大小 字节转换
def sizeFormat(size, is_disk=False, precision=2):
'''
size format for human.
byte ---- (B)
kilobyte ---- (KB)
megabyte ---- (MB)
gigabyte ---- (GB)
terabyte ---- (TB)
petabyte ---- (PB)
exabyte ---- (EB)
zettabyte ---- (ZB)
yottabyte ---- (YB)
'''
formats = ['KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB']
unit = 1000.0 if is_disk else 1024.0
if not (isinstance(size, float) or isinstance(size, int)):
raise TypeError('a float number or an integer number is required!')
if size < 0:
raise ValueError('number must be non-negative')
for i in formats:
size /= unit
if size < unit:
r = '{}{}'.format(round(size, precision),i)
return r
if request.method == 'GET':
try:
search_kw = request.GET.get('kw', None)
# 搜索附件
if search_kw:
attachment_list = Attachment.objects.filter(
user=request.user,
file_name__icontains=search_kw
).order_by('-create_time')
paginator = Paginator(attachment_list, 15)
page = request.GET.get('page', 1)
try:
attachments = paginator.page(page)
except PageNotAnInteger:
attachments = paginator.page(1)
except EmptyPage:
attachments = paginator.page(paginator.num_pages)
attachments.kw = search_kw
# 所有附件
else:
attachment_list = Attachment.objects.filter(user=request.user).order_by('-create_time')
paginator = Paginator(attachment_list, 15)
page = request.GET.get('page', 1)
try:
attachments = paginator.page(page)
except PageNotAnInteger:
attachments = paginator.page(1)
except EmptyPage:
attachments = paginator.page(paginator.num_pages)
return render(request, 'app_doc/manage/manage_attachment.html', locals())
except Exception as e:
logger.exception(_("附件管理访问出错"))
return render(request,'404.html')
elif request.method == 'POST':
# types参数0表示上传、1表示删除、2表示获取附件列表
types = request.POST.get('types','')
if types in ['0',0]:
attachment = request.FILES.get('attachment_upload',None)
if attachment:
attachment_name = attachment.name # 获取附件文件名
attachment_size = sizeFormat(attachment.size) # 获取附件文件大小
# 限制附件大小
# 获取系统设置的附件文件大小如果不存在默认50MB
try:
allow_attachment_size = SysSetting.objects.get(types='doc',name='attachment_size')
allow_attach_size = int(allow_attachment_size.value) * 1048576
except Exception as e:
# print(repr(e))
allow_attach_size = 52428800
if attachment.size > allow_attach_size:
return JsonResponse({'status':False,'data':_('文件大小超出限制')})
# 限制附件格式
# 获取系统设置允许的附件格式如果不存在默认仅允许zip格式文件
try:
attachment_suffix_list = SysSetting.objects.get(types='doc',name='attachment_suffix')
attachment_suffix_list = attachment_suffix_list.value.split(',')
if attachment_suffix_list == ['']:
attachment_suffix_list = ['zip']
except ObjectDoesNotExist:
attachment_suffix_list = ['zip']
allow_attachment = False
for suffix in attachment_suffix_list:
if attachment_name.split('.')[-1].lower() in attachment_suffix_list:
allow_attachment = True
if allow_attachment:
a = Attachment.objects.create(
file_name = attachment_name,
file_size = attachment_size,
file_path = attachment,
user = request.user
)
return JsonResponse({'status':True,'data':{'name':attachment_name,'url':a.file_path.name}})
else:
return JsonResponse({'status':False,'data':_('不支持的格式')})
else:
return JsonResponse({'status':False,'data':_('无效文件')})
elif types in ['1',1]:
attach_id = request.POST.get('attach_id','')
attachment = Attachment.objects.filter(id=attach_id,user=request.user) # 查询附件
for a in attachment: # 遍历附件
a.file_path.delete() # 删除文件
attachment.delete() # 删除数据库记录
return JsonResponse({'status':True,'data':_('删除成功')})
elif types in [2,'2']:
attachment_list = []
attachments = Attachment.objects.filter(user=request.user).order_by('-create_time')
for a in attachments:
item = {
'filename':a.file_name,
'filesize':a.file_size,
'filepath':a.file_path.name,
'filetime':a.create_time
}
attachment_list.append(item)
return JsonResponse({'status':True,'data':attachment_list})
else:
return JsonResponse({'status':False,'data':_('无效参数')})
# 搜索
def search(request):
kw = request.GET.get('kw', None)
search_type = request.GET.get('type', 'doc') # 搜索类型默认文档doc
date_type = request.GET.get('d_type', 'recent')
date_range = request.GET.get('d_range', 'all') # 时间范围默认不限all
project_range = request.GET.get('p_range', 0) # 文集范围默认不限all
# 处理时间范围
if date_type == 'recent':
if date_range == 'recent1': # 最近1天
start_date = datetime.datetime.now() - datetime.timedelta(days=1)
elif date_range == 'recent7': # 最近7天
start_date = datetime.datetime.now() - datetime.timedelta(days=7)
elif date_range == 'recent30': # 最近30天
start_date = datetime.datetime.now() - datetime.timedelta(days=30)
elif date_range == 'recent365': # 最近一年
start_date = datetime.datetime.now() - datetime.timedelta(days=365)
else:
start_date = datetime.datetime.strptime('1900-01-01', '%Y-%m-%d')
end_date = datetime.datetime.now()
elif date_type == 'day':
try:
date_list = date_range.split('|')
start_date = datetime.datetime.strptime(date_list[0], '%Y-%m-%d')
end_date = datetime.datetime.strptime(date_list[1], '%Y-%m-%d')
except:
start_date = datetime.datetime.now() - datetime.timedelta(days=1)
end_date = datetime.datetime.now()
# 是否时间筛选
if date_range == 'all':
is_date_range = False
else:
is_date_range = True
# 是否认证
if request.user.is_authenticated:
is_auth = True
else:
is_auth = False
# 存在搜索关键词
if kw:
# 搜索文档
if search_type == 'doc':
if is_auth:
colla_list = [i.project.id for i in ProjectCollaborator.objects.filter(user=request.user)] # 用户的协作文集
open_list = [i.id for i in Project.objects.filter(
Q(role=0) | Q(create_user=request.user)
)] # 公开文集
view_list = list(set(open_list).union(set(colla_list))) # 合并上述两个文集ID列表
data_list = Doc.objects.filter(
Q(top_doc__in=view_list), # 包含用户可浏览到的文集
Q(create_time__gte=start_date, create_time__lte=end_date), # 筛选创建时间
Q(name__icontains=kw) | Q(content__icontains=kw) | Q(pre_content__icontains=kw) # 筛选文档标题和内容中包含搜索词
).order_by('-create_time')
else:
view_list = [i.id for i in Project.objects.filter(role=0)]
data_list = Doc.objects.filter(
Q(top_doc__in=view_list),
Q(create_time__gte=start_date, create_time__lte=end_date), # 筛选创建时间
Q(name__icontains=kw) | Q(content__icontains=kw) | Q(pre_content__icontains=kw) # 筛选文档标题和内容中包含搜索词
).order_by('-create_time')
# 搜索文集
elif search_type == 'pro':
# 认证用户
if is_auth:
colla_list = [i.project.id for i in ProjectCollaborator.objects.filter(user=request.user)] # 用户的协作文集
# 查询所有可显示的文集
data_list = Project.objects.filter(
Q(role=0) | \
Q(role=2, role_value__contains=str(request.user.username)) | \
Q(create_user=request.user) | \
Q(id__in=colla_list),
Q(create_time__gte=start_date, create_time__lte=end_date), # 筛选创建时间
Q(name__icontains=kw) | Q(intro__icontains=kw) # 筛选文集名称和简介包含搜索词
).order_by('-create_time')
# 游客
else:
data_list = Project.objects.filter(
Q(role=0),
Q(name__icontains=kw) | Q(intro__icontains=kw),
Q(create_time__gte=start_date, create_time__lte=end_date), # 筛选创建时间
).order_by("-create_time")
# 搜索标签
elif search_type == 'tag':
# 认证用户
if is_auth:
colla_list = [i.project.id for i in ProjectCollaborator.objects.filter(user=request.user)] # 用户的协作文集
open_list = [i.id for i in Project.objects.filter(
Q(role=0) | Q(create_user=request.user)
)] # 公开文集和自己创建的文集
view_list = list(set(open_list).union(set(colla_list))) # 合并上述两个文集ID列表
tag_list = Tag.objects.filter(name__icontains=kw) # 查询符合条件的标签
tag_doc_list = [i.doc.id for i in DocTag.objects.filter(tag__in=tag_list)] # 获取符合条件的标签文档
data_list = Doc.objects.filter(
Q(top_doc__in=view_list), # 包含用户可浏览到的文集
Q(id__in=tag_doc_list), # 包含符合条件标签的文档ID列表
Q(create_time__gte=start_date, create_time__lte=end_date), # 筛选创建时间
).order_by('-create_time')
# 游客
else:
open_list = [i.id for i in Project.objects.filter(Q(role=0))] # 公开文集
view_list = list(set(open_list))
tag_list = Tag.objects.filter(name__icontains=kw) # 查询符合条件的标签
tag_doc_list = [i.doc.id for i in DocTag.objects.filter(tag__in=tag_list)] # 获取符合条件的标签文档
data_list = Doc.objects.filter(
Q(top_doc__in=view_list), # 包含用户可浏览到的文集
Q(id__in=tag_doc_list), # 包含符合条件标签的文档ID列表
Q(create_time__gte=start_date, create_time__lte=end_date), # 筛选创建时间
).order_by('-create_time')
else:
return render(request, 'app_doc/search.html')
# 分页处理
paginator = Paginator(data_list, 12)
page = request.GET.get('page', 1)
try:
datas = paginator.page(page)
except PageNotAnInteger:
datas = paginator.page(1)
except EmptyPage:
datas = paginator.page(paginator.num_pages)
return render(request, 'app_doc/search_result.html', locals())
# 否则跳转到搜索首页
else:
return render(request,'app_doc/search.html')
# 文档Markdown文件下载
@require_http_methods(['GET',"POST"])
def download_doc_md(request,doc_id):
if request.user.is_authenticated:
if request.user.is_superuser:
try:
doc = Doc.objects.get(id=doc_id)
except ObjectDoesNotExist:
return JsonResponse({'status':False,'data':_('文档不存在')})
else:
try:
doc = Doc.objects.get(id=doc_id,create_user = request.user)
except ObjectDoesNotExist:
return JsonResponse({'status':False,'data':_('文档不存在')})
else:
return render(request,'404.html')
response = HttpResponse(content_type='text/plain')
response['Content-Disposition'] = 'attachment; filename={}.md'.format(doc.name)
response.write(doc.pre_content)
return response
# 个人中心 - 概览
@login_required()
@require_http_methods(['GET','POST'])
def manage_overview(request):
if request.method == 'GET':
pro_list = Project.objects.filter(create_user=request.user).order_by('-create_time')
colla_pro_cnt = ProjectCollaborator.objects.filter(user=request.user).count()
pro_cnt = pro_list.count() + colla_pro_cnt # 文集总数
doc_cnt = Doc.objects.filter(create_user=request.user).count() # 文档总数
total_tag_cnt = Tag.objects.filter(create_user=request.user).count()
img_cnt = Image.objects.filter(user=request.user).count()
attachment_cnt = Attachment.objects.filter(user=request.user).count()
doc_active_list = Doc.objects.filter(create_user=request.user).order_by('-modify_time')[:5]
return render(request,'app_doc/manage/manage_overview.html',locals())
else:
pass
# 个人中心 - 文档标签
@login_required()
@require_http_methods(['GET','POST'])
def manage_doc_tag(request):
if request.method == 'GET':
tags = Tag.objects.filter(create_user=request.user)
return render(request,'app_doc/manage/manage_doc_tag.html',locals())
# 操作标签
elif request.method == 'POST':
types = request.POST.get('types', None) # 请求类型0表示创建标签1表示修改标签2表示删除标签3表示获取标签
# 创建标签
if int(types) == 0:
tag_name = request.POST.get('tag_name', '')
if tag_name != '':
Tag.objects.create(
user=request.user,
name=tag_name
)
return JsonResponse({'status': True, 'data': 'ok'})
else:
return JsonResponse({'status': False, 'data': _('名称无效')})
# 修改标签
elif int(types) == 1:
try:
tag_name = request.POST.get('tag_name', '')
if tag_name != "":
tag_id = request.POST.get('tag_id', '')
if tag_id != "":
print(tag_id,tag_name)
Tag.objects.filter(id=tag_id, create_user=request.user).update(name=tag_name)
return JsonResponse({'status': True, 'data': _('修改成功')})
else:
return JsonResponse({'status': False, 'data': _('标签ID无效')})
else:
return JsonResponse({'status': False, 'data': _('名称无效')})
except Exception as e:
logger.exception(_("修改异常"))
return JsonResponse({'status': False, 'data': _('异常错误')})
# 删除标签
elif int(types) == 2:
try:
tag_id = request.POST.get('tag_id', '')
tag = Tag.objects.get(id=tag_id, create_user=request.user) # 查询分组
tag.delete() # 删除标签
return JsonResponse({'status': True, 'data': _('删除完成')})
except:
logger.exception(_("删除标签出错"))
return JsonResponse({'status': False, 'data': _('删除错误')})
# 获取标签
elif int(types) == 3:
try:
tag_list = []
return JsonResponse({'status': True, 'data': tag_list})
except:
logger.exception(_("获取文档标签出错"))
return JsonResponse({'status': False, 'data': _('出现错误')})
# 标签文档关系页
def tag_docs(request,tag_id):
# 获取标签
try:
# 颜色列表
color_list = ['#37a2da', '#32c5e9', '#67e0e3', '#9fe6b8', '#ffdb5c', '#ff9f7f', '#fb7293', '#e062ae', '#e062ae']
# 获取标签信息
tag = Tag.objects.get(id=int(tag_id))
# 获取标签的文档信息
# 如果访问者已经登录
if request.user.is_authenticated:
# 判断是否为标签的创建者
if request.user == tag.create_user:
# 获取标签的所有文档
view_list = [i.id for i in Project.objects.filter(create_user=request.user)]
docs = DocTag.objects.filter(tag=tag,doc__status=1)
else:
# 获取有权限的文档
colla_list = [i.project.id for i in ProjectCollaborator.objects.filter(user=request.user)] # 用户的协作文集
open_list = [i.id for i in Project.objects.filter(
Q(role=0) | Q(create_user=tag.create_user)
)] # 公开文集
view_list = list(set(open_list).union(set(colla_list))) # 合并上述两个文集ID列表
# 查询可浏览文集的文档
doc_list = [i for i in Doc.objects.filter(top_doc__in=view_list,status=1)]
# 筛选可浏览的文档的标签文档
docs = DocTag.objects.filter(tag=tag,doc__in=doc_list)
else:
# 查询标签创建者的公开文集
open_list = [i.id for i in Project.objects.filter(
role=0,create_user=tag.create_user
)]
view_list = open_list
doc_list = [i for i in Doc.objects.filter(top_doc__in=view_list,status=1)]
docs = DocTag.objects.filter(tag=tag,doc__in=doc_list)
# 获取文档的其他标签信息
current_link_list = [] # 文档的所有标签ID列表
for doc in docs:
other_tags = [str(i.tag.id) for i in DocTag.objects.filter(~Q(tag=tag), doc=doc.doc)]
current_link_list.extend(other_tags)
# 标签的节点列表
tag_nodes_list = [
# {'id':str(tag.id),'name':tag.name,'symbolSize':50,'value':docs.count(),'itemStyle':{'color':random.choice(color_list)}}
]
# 标签的关系列表
tag_links_list = []
# 标签分类列表
tag_cate = []
# 添加用户创建的所有标签到节点列表
for t in Tag.objects.filter(create_user=tag.create_user):
tag_cate.append({'name':t.name})
if t.name == tag.name:
item = {
'id': str(t.id),
'name': t.name,
'symbolSize': 50,
'value': DocTag.objects.filter(tag=t,doc__status=1,doc__top_doc__in=view_list).count(),
'itemStyle': {'color': random.choice(color_list)}
}
else:
item = {
'id':str(t.id),
'name':t.name,
'symbolSize':25,
'value':DocTag.objects.filter(tag=t,doc__status=1,doc__top_doc__in=view_list).count(),
'itemStyle':{'color':random.choice(color_list)}
}
tag_nodes_list.append(item)
# 查询非主标签的关联标签
sub_tags = DocTag.objects.filter(tag=t,doc__status=1,doc__top_doc__in=view_list) # 获取包含t标签的文档
for sub_tag in sub_tags:
sub_docs = DocTag.objects.filter(doc=sub_tag.doc,doc__top_doc__in=view_list) # 获取包含文档的标签
for sub_doc in sub_docs:
if str(sub_tag.tag.id) != str(sub_doc.tag.id):
item = {
'source': str(sub_tag.tag.id),
'target': str(sub_doc.tag.id),
'value' : sub_doc.doc.name,
'id': sub_doc.doc.id,
'pid': sub_doc.doc.top_doc,
'label':{
'normal':{
'show':'true',
'formatter':"{c}",
'fontsize':'10px',
}
}
}
item_1 = {
'source': str(sub_doc.tag.id),
'target': str(sub_tag.tag.id),
'value': sub_doc.doc.name,
'id':sub_doc.doc.id,
'pid': sub_doc.doc.top_doc,
'label': {
'normal': {
'show': 'true',
'formatter': "{c}",
'fontsize': '10px',
}
}
}
if item_1 not in tag_links_list:
tag_links_list.append(item)
return render(request, 'app_doc/tag_docs.html', locals())
except Exception as e:
logger.exception(_("标签文档页访问异常"))
return render(request, '404.html')
# 标签文档页
@require_http_methods(['GET'])
def tag_doc(request,tag_id,doc_id):
try:
if tag_id != '' and doc_id != '':
doc = Doc.objects.get(id=int(doc_id), status=1)
# 获取文档的文集信息,以判断是否有权限访问
project = Project.objects.get(id=int(doc.top_doc))
# 获取文集的协作用户信息
if request.user.is_authenticated:
colla_user = ProjectCollaborator.objects.filter(project=project,user=request.user)
if colla_user.exists():
colla_user_role = colla_user[0].role
colla_user = colla_user.count()
else:
colla_user = colla_user.count()
else:
colla_user = 0
# 私密文集且访问者非创建者、协作者 - 不能访问
if (project.role == 1) and (request.user != project.create_user) and (colla_user == 0):
return render(request, '404.html')
# 指定用户可见文集
elif project.role == 2:
user_list = project.role_value
if request.user.is_authenticated: # 认证用户判断是否在许可用户列表中
if (request.user.username not in user_list) and \
(request.user != project.create_user) and \
(colla_user == 0): # 访问者不在指定用户之中,也不是协作者
return render(request, '404.html')
else: # 游客直接返回404
return render(request, '404.html')
# 访问码可见
elif project.role == 3:
# 浏览用户不为创建者和协作者 - 需要访问码
if (request.user != project.create_user) and (colla_user == 0):
viewcode = project.role_value
viewcode_name = 'viewcode-{}'.format(project.id)
r_viewcode = request.COOKIES[
viewcode_name] if viewcode_name in request.COOKIES.keys() else 0 # 从cookie中获取访问码
if viewcode != r_viewcode: # cookie中的访问码不等于文集访问码跳转到访问码认证界面
return redirect('/check_viewcode/?to={}'.format(request.path))
# 获取文档内容
try:
# 获取标签信息
tag = Tag.objects.get(id=int(tag_id))
# 获取标签文档信息
docs = DocTag.objects.filter(tag=tag)
# 获取文档的标签
doc_tags = DocTag.objects.filter(doc=doc)
except ObjectDoesNotExist:
return render(request, '404.html')
return render(request,'app_doc/tag_doc_single.html',locals())
else:
return HttpResponse(_('参数错误'))
except Exception as e:
logger.exception(_("文集浏览出错"))
return render(request,'404.html')
# 个人中心 - 个人设置
@login_required()
def manage_self(request):
if request.method == 'GET':
user = User.objects.get_by_natural_key(request.user)
try:
user_opt = UserOptions.objects.get(user=request.user)
except ObjectDoesNotExist:
user_opt = []
return render(request,'app_doc/manage/manage_self.html',locals())
elif request.method == 'POST':
first_name = request.POST.get('first_name','') # 昵称
email = request.POST.get('email',None) # 电子邮箱
editor_mode = request.POST.get('editor_mode',1) # 编辑器
user = User.objects.get_by_natural_key(request.user)
if len(first_name) < 2 or len(first_name) > 10:
return JsonResponse({'status': False, 'data': _('昵称长度不得小于2位大于10位')})
if User.objects.filter(first_name=first_name).count() > 0 and user.first_name != first_name:
return JsonResponse({'status':False,'data':_('昵称已被使用')})
if User.objects.filter(email=email).count() > 0 and user.email != email:
return JsonResponse({'status':False,'data':_('电子邮箱已被使用')})
if email != '' and '@' in email:
user.email = email
user.first_name = first_name
user.save()
user_opt = UserOptions.objects.update_or_create(
user = user,
defaults={'editor_mode':editor_mode}
)
return JsonResponse({'status':True,'data':'ok'})
else:
return JsonResponse({'status':False,'data':_('参数不正确')})
# 文集文档收藏
@login_required()
def my_collect(request):
if request.method == 'GET':
pass
elif request.method == 'POST':
collect_type = request.POST.get('type',None) # 收藏类型
collect_id = request.POST.get('id',None) # 收藏对象ID
if (collect_type is None) or (collect_id is None):
return JsonResponse({'status':False,'data':_('参数错误')})
else:
is_collect = MyCollect.objects.filter(collect_type=collect_type,collect_id=collect_id,create_user=request.user)
# 存在收藏
if is_collect.exists():
is_collect.delete()
return JsonResponse({'status': True, 'data': _('取消收藏成功')})
else:
MyCollect.objects.create(
collect_type = collect_type,
collect_id = collect_id,
create_user = request.user,
create_time = datetime.datetime.now()
)
return JsonResponse({'status':True,'data':_('收藏成功')})
elif request.method == 'DELETE':
pass
# 收藏管理
@login_required()
@require_http_methods(['GET','POST','DELETE'])
@csrf_exempt
def manage_collect(request):
if request.method == 'GET':
# 收藏文集数量
collect_project_cnt = MyCollect.objects.filter(create_user=request.user, collect_type=2).count()
# 收藏文档数量
collect_doc_cnt = MyCollect.objects.filter(create_user=request.user, collect_type=1).count()
# 所有收藏数量
all_cnt = collect_project_cnt + collect_doc_cnt
return render(request,'app_doc/manage/manage_collect.html',locals())
elif request.method == 'POST':
kw = request.POST.get('kw', '') # 搜索词
collect_type = request.POST.get('type', '') # 收藏类型
if collect_type in ['1', '2']:
q_type = [int(collect_type)]
else:
q_type = [1, 2]
page = request.POST.get('page', 1)
limit = request.POST.get('limit', 10)
# 没有搜索
if kw == '':
collect_list = MyCollect.objects.filter(
create_user=request.user,
collect_type__in=q_type,
).order_by('-create_time')
# 有搜索
else:
collect_list = MyCollect.objects.filter(
Q(content__icontains=kw) | Q(name__icontains=kw),
create_user=request.user, collect_type__in=q_type
).order_by('-create_time')
# 分页处理
paginator = Paginator(collect_list, limit)
page = request.GET.get('page', page)
try:
collects = paginator.page(page)
except PageNotAnInteger:
collects = paginator.page(1)
except EmptyPage:
collects = paginator.page(paginator.num_pages)
table_data = []
for collect in collects:
if collect.collect_type == 1:
item_doc = Doc.objects.get(id=collect.collect_id)
item_id = item_doc.id
item_name = item_doc.name
item_project = Project.objects.get(id=item_doc.top_doc)
item_project_name = item_project.name
item_project_id = item_project.id
else:
item_project = Project.objects.get(id=collect.collect_id)
item_id = item_project.id
item_name = item_project.name
item_project_name = ''
item_project_id = ''
item = {
'id': collect.id,
'item_id':item_id,
'item_name': item_name,
'type': collect.collect_type,
'item_project_id':item_project_id,
'item_project_name':item_project_name,
'create_time': collect.create_time,
}
table_data.append(item)
resp_data = {
"code": 0,
"msg": "ok",
"count": collect_list.count(),
"data": table_data
}
return JsonResponse(resp_data)
elif request.method == 'DELETE':
try:
# 获取收藏ID
DELETE = QueryDict(request.body)
collect_id = DELETE.get('collect_id', None)
range = DELETE.get('range', 'single')
if collect_id:
if range == 'single':
# 查询收藏
try:
collect = MyCollect.objects.get(id=collect_id)
except ObjectDoesNotExist:
return JsonResponse({'status': False, 'data': _('收藏不存在')})
# 如果请求用户为站点管理员、收藏的创建者,可以删除
if (request.user == collect.create_user) or (request.user.is_superuser):
MyCollect.objects.filter(id=collect_id).delete()
return JsonResponse({'status': True, 'data': _('删除完成')})
else:
return JsonResponse({'status': False, 'data': _('非法请求')})
elif range == 'multi':
collects = collect_id.split(",")
try:
MyCollect.objects.filter(id__in=collects, create_user=request.user).delete()
return JsonResponse({'status': True, 'data': _('删除完成')})
except:
return JsonResponse({'status': False, 'data': _('非法请求')})
else:
return JsonResponse({'status': False, 'data': _('类型错误')})
else:
return JsonResponse({'status': False, 'data': _('参数错误')})
except Exception as e:
logger.exception(_("取消收藏出错"))
return JsonResponse({'status': False, 'data': _('请求出错')})
# 获取当前版本
def get_version(request):
try:
version = settings.VERSIONS
data = {
'status':True,
'data':version
}
except:
data = {
'status':False,
'data':_('异常')
}
return JsonResponse(data)
# 用户分组用户列表接口
class UserGroupUserList(APIView):
authentication_classes = [SessionAuthentication, AppMustAuth]
def get(self,request):
user_data = User.objects.filter(is_active=True).values(
'id', 'username', 'first_name'
)
user_list = []
for user in user_data:
item = {
'name':user['username'],
'value':user['id']
}
user_list.append(item)
# serializer = UserSerializer(user_data, many=True) # 对结果进行序列化处理
resp = {
'code': 0,
'data': user_list,
'count': user_data.count()
}
return Response(resp)