mirror of
https://github.com/infiniflow/ragflow.git
synced 2026-05-04 17:27:50 +08:00
Refa: files /file API to RESTFul style (#13741)
### What problem does this PR solve? Files /file API to RESTFul style. ### Type of change - [x] Documentation Update - [x] Refactoring --------- Co-authored-by: writinwaters <cai.keith@gmail.com> Co-authored-by: Liu An <asiro@qq.com>
This commit is contained in:
@ -1,464 +1,464 @@
|
||||
# #
|
||||
# # Copyright 2024 The InfiniFlow Authors. All Rights Reserved.
|
||||
# #
|
||||
# # Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# # you may not use this file except in compliance with the License.
|
||||
# # You may obtain a copy of the License at
|
||||
# #
|
||||
# # http://www.apache.org/licenses/LICENSE-2.0
|
||||
# #
|
||||
# # Unless required by applicable law or agreed to in writing, software
|
||||
# # distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# # See the License for the specific language governing permissions and
|
||||
# # limitations under the License
|
||||
# #
|
||||
# import logging
|
||||
# import os
|
||||
# import pathlib
|
||||
# import re
|
||||
# from quart import request, make_response
|
||||
# from api.apps import login_required, current_user
|
||||
#
|
||||
# Copyright 2024 The InfiniFlow Authors. All Rights Reserved.
|
||||
# from api.common.check_team_permission import check_file_team_permission
|
||||
# from api.db.services.document_service import DocumentService
|
||||
# from api.db.services.file2document_service import File2DocumentService
|
||||
# from api.utils.api_utils import server_error_response, get_data_error_result, validate_request
|
||||
# from common.misc_utils import get_uuid, thread_pool_exec
|
||||
# from common.constants import RetCode, FileSource
|
||||
# from api.db import FileType
|
||||
# from api.db.services import duplicate_name
|
||||
# from api.db.services.file_service import FileService
|
||||
# from api.utils.api_utils import get_json_result, get_request_json
|
||||
# from api.utils.file_utils import filename_type
|
||||
# from api.utils.web_utils import CONTENT_TYPE_MAP, apply_safe_file_response_headers
|
||||
# from common import settings
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
# You may obtain a copy of the License at
|
||||
# @manager.route('/upload', methods=['POST']) # noqa: F821
|
||||
# @login_required
|
||||
# # @validate_request("parent_id")
|
||||
# async def upload():
|
||||
# form = await request.form
|
||||
# pf_id = form.get("parent_id")
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
# if not pf_id:
|
||||
# root_folder = FileService.get_root_folder(current_user.id)
|
||||
# pf_id = root_folder["id"]
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# See the License for the specific language governing permissions and
|
||||
# limitations under the License
|
||||
# files = await request.files
|
||||
# if 'file' not in files:
|
||||
# return get_json_result(
|
||||
# data=False, message='No file part!', code=RetCode.ARGUMENT_ERROR)
|
||||
# file_objs = files.getlist('file')
|
||||
#
|
||||
import logging
|
||||
import os
|
||||
import pathlib
|
||||
import re
|
||||
from quart import request, make_response
|
||||
from api.apps import login_required, current_user
|
||||
|
||||
from api.common.check_team_permission import check_file_team_permission
|
||||
from api.db.services.document_service import DocumentService
|
||||
from api.db.services.file2document_service import File2DocumentService
|
||||
from api.utils.api_utils import server_error_response, get_data_error_result, validate_request
|
||||
from common.misc_utils import get_uuid, thread_pool_exec
|
||||
from common.constants import RetCode, FileSource
|
||||
from api.db import FileType
|
||||
from api.db.services import duplicate_name
|
||||
from api.db.services.file_service import FileService
|
||||
from api.utils.api_utils import get_json_result, get_request_json
|
||||
from api.utils.file_utils import filename_type
|
||||
from api.utils.web_utils import CONTENT_TYPE_MAP, apply_safe_file_response_headers
|
||||
from common import settings
|
||||
|
||||
@manager.route('/upload', methods=['POST']) # noqa: F821
|
||||
@login_required
|
||||
# @validate_request("parent_id")
|
||||
async def upload():
|
||||
form = await request.form
|
||||
pf_id = form.get("parent_id")
|
||||
|
||||
if not pf_id:
|
||||
root_folder = FileService.get_root_folder(current_user.id)
|
||||
pf_id = root_folder["id"]
|
||||
|
||||
files = await request.files
|
||||
if 'file' not in files:
|
||||
return get_json_result(
|
||||
data=False, message='No file part!', code=RetCode.ARGUMENT_ERROR)
|
||||
file_objs = files.getlist('file')
|
||||
|
||||
for file_obj in file_objs:
|
||||
if file_obj.filename == '':
|
||||
return get_json_result(
|
||||
data=False, message='No file selected!', code=RetCode.ARGUMENT_ERROR)
|
||||
file_res = []
|
||||
try:
|
||||
e, pf_folder = FileService.get_by_id(pf_id)
|
||||
if not e:
|
||||
return get_data_error_result( message="Can't find this folder!")
|
||||
|
||||
async def _handle_single_file(file_obj):
|
||||
MAX_FILE_NUM_PER_USER: int = int(os.environ.get('MAX_FILE_NUM_PER_USER', 0))
|
||||
if 0 < MAX_FILE_NUM_PER_USER <= await thread_pool_exec(DocumentService.get_doc_count, current_user.id):
|
||||
return get_data_error_result( message="Exceed the maximum file number of a free user!")
|
||||
|
||||
# split file name path
|
||||
if not file_obj.filename:
|
||||
file_obj_names = [pf_folder.name, file_obj.filename]
|
||||
else:
|
||||
full_path = '/' + file_obj.filename
|
||||
file_obj_names = full_path.split('/')
|
||||
file_len = len(file_obj_names)
|
||||
|
||||
# get folder
|
||||
file_id_list = await thread_pool_exec(FileService.get_id_list_by_id, pf_id, file_obj_names, 1, [pf_id])
|
||||
len_id_list = len(file_id_list)
|
||||
|
||||
# create folder
|
||||
if file_len != len_id_list:
|
||||
e, file = await thread_pool_exec(FileService.get_by_id, file_id_list[len_id_list - 1])
|
||||
if not e:
|
||||
return get_data_error_result(message="Folder not found!")
|
||||
last_folder = await thread_pool_exec(FileService.create_folder, file, file_id_list[len_id_list - 1], file_obj_names,
|
||||
len_id_list)
|
||||
else:
|
||||
e, file = await thread_pool_exec(FileService.get_by_id, file_id_list[len_id_list - 2])
|
||||
if not e:
|
||||
return get_data_error_result(message="Folder not found!")
|
||||
last_folder = await thread_pool_exec(FileService.create_folder, file, file_id_list[len_id_list - 2], file_obj_names,
|
||||
len_id_list)
|
||||
|
||||
# file type
|
||||
filetype = filename_type(file_obj_names[file_len - 1])
|
||||
location = file_obj_names[file_len - 1]
|
||||
while await thread_pool_exec(settings.STORAGE_IMPL.obj_exist, last_folder.id, location):
|
||||
location += "_"
|
||||
blob = await thread_pool_exec(file_obj.read)
|
||||
filename = await thread_pool_exec(
|
||||
duplicate_name,
|
||||
FileService.query,
|
||||
name=file_obj_names[file_len - 1],
|
||||
parent_id=last_folder.id)
|
||||
await thread_pool_exec(settings.STORAGE_IMPL.put, last_folder.id, location, blob)
|
||||
file_data = {
|
||||
"id": get_uuid(),
|
||||
"parent_id": last_folder.id,
|
||||
"tenant_id": current_user.id,
|
||||
"created_by": current_user.id,
|
||||
"type": filetype,
|
||||
"name": filename,
|
||||
"location": location,
|
||||
"size": len(blob),
|
||||
}
|
||||
inserted = await thread_pool_exec(FileService.insert, file_data)
|
||||
return inserted.to_json()
|
||||
|
||||
for file_obj in file_objs:
|
||||
res = await _handle_single_file(file_obj)
|
||||
file_res.append(res)
|
||||
|
||||
return get_json_result(data=file_res)
|
||||
except Exception as e:
|
||||
return server_error_response(e)
|
||||
|
||||
|
||||
@manager.route('/create', methods=['POST']) # noqa: F821
|
||||
@login_required
|
||||
@validate_request("name")
|
||||
async def create():
|
||||
req = await get_request_json()
|
||||
pf_id = req.get("parent_id")
|
||||
input_file_type = req.get("type")
|
||||
if not pf_id:
|
||||
root_folder = FileService.get_root_folder(current_user.id)
|
||||
pf_id = root_folder["id"]
|
||||
|
||||
try:
|
||||
if not FileService.is_parent_folder_exist(pf_id):
|
||||
return get_json_result(
|
||||
data=False, message="Parent Folder Doesn't Exist!", code=RetCode.OPERATING_ERROR)
|
||||
if FileService.query(name=req["name"], parent_id=pf_id):
|
||||
return get_data_error_result(
|
||||
message="Duplicated folder name in the same folder.")
|
||||
|
||||
if input_file_type == FileType.FOLDER.value:
|
||||
file_type = FileType.FOLDER.value
|
||||
else:
|
||||
file_type = FileType.VIRTUAL.value
|
||||
|
||||
file = FileService.insert({
|
||||
"id": get_uuid(),
|
||||
"parent_id": pf_id,
|
||||
"tenant_id": current_user.id,
|
||||
"created_by": current_user.id,
|
||||
"name": req["name"],
|
||||
"location": "",
|
||||
"size": 0,
|
||||
"type": file_type
|
||||
})
|
||||
|
||||
return get_json_result(data=file.to_json())
|
||||
except Exception as e:
|
||||
return server_error_response(e)
|
||||
|
||||
|
||||
@manager.route('/list', methods=['GET']) # noqa: F821
|
||||
@login_required
|
||||
def list_files():
|
||||
pf_id = request.args.get("parent_id")
|
||||
|
||||
keywords = request.args.get("keywords", "")
|
||||
|
||||
page_number = int(request.args.get("page", 1))
|
||||
items_per_page = int(request.args.get("page_size", 15))
|
||||
orderby = request.args.get("orderby", "create_time")
|
||||
desc = request.args.get("desc", True)
|
||||
if not pf_id:
|
||||
root_folder = FileService.get_root_folder(current_user.id)
|
||||
pf_id = root_folder["id"]
|
||||
FileService.init_knowledgebase_docs(pf_id, current_user.id)
|
||||
try:
|
||||
e, file = FileService.get_by_id(pf_id)
|
||||
if not e:
|
||||
return get_data_error_result(message="Folder not found!")
|
||||
|
||||
files, total = FileService.get_by_pf_id(
|
||||
current_user.id, pf_id, page_number, items_per_page, orderby, desc, keywords)
|
||||
|
||||
parent_folder = FileService.get_parent_folder(pf_id)
|
||||
if not parent_folder:
|
||||
return get_json_result(message="File not found!")
|
||||
|
||||
return get_json_result(data={"total": total, "files": files, "parent_folder": parent_folder.to_json()})
|
||||
except Exception as e:
|
||||
return server_error_response(e)
|
||||
|
||||
|
||||
@manager.route('/root_folder', methods=['GET']) # noqa: F821
|
||||
@login_required
|
||||
def get_root_folder():
|
||||
try:
|
||||
root_folder = FileService.get_root_folder(current_user.id)
|
||||
return get_json_result(data={"root_folder": root_folder})
|
||||
except Exception as e:
|
||||
return server_error_response(e)
|
||||
|
||||
|
||||
@manager.route('/parent_folder', methods=['GET']) # noqa: F821
|
||||
@login_required
|
||||
def get_parent_folder():
|
||||
file_id = request.args.get("file_id")
|
||||
try:
|
||||
e, file = FileService.get_by_id(file_id)
|
||||
if not e:
|
||||
return get_data_error_result(message="Folder not found!")
|
||||
|
||||
parent_folder = FileService.get_parent_folder(file_id)
|
||||
return get_json_result(data={"parent_folder": parent_folder.to_json()})
|
||||
except Exception as e:
|
||||
return server_error_response(e)
|
||||
|
||||
|
||||
@manager.route('/all_parent_folder', methods=['GET']) # noqa: F821
|
||||
@login_required
|
||||
def get_all_parent_folders():
|
||||
file_id = request.args.get("file_id")
|
||||
try:
|
||||
e, file = FileService.get_by_id(file_id)
|
||||
if not e:
|
||||
return get_data_error_result(message="Folder not found!")
|
||||
|
||||
parent_folders = FileService.get_all_parent_folders(file_id)
|
||||
parent_folders_res = []
|
||||
for parent_folder in parent_folders:
|
||||
parent_folders_res.append(parent_folder.to_json())
|
||||
return get_json_result(data={"parent_folders": parent_folders_res})
|
||||
except Exception as e:
|
||||
return server_error_response(e)
|
||||
|
||||
|
||||
@manager.route("/rm", methods=["POST"]) # noqa: F821
|
||||
@login_required
|
||||
@validate_request("file_ids")
|
||||
async def rm():
|
||||
req = await get_request_json()
|
||||
file_ids = req["file_ids"]
|
||||
uid = current_user.id
|
||||
|
||||
try:
|
||||
def _delete_single_file(file):
|
||||
try:
|
||||
if file.location:
|
||||
settings.STORAGE_IMPL.rm(file.parent_id, file.location)
|
||||
except Exception as e:
|
||||
logging.exception(f"Fail to remove object: {file.parent_id}/{file.location}, error: {e}")
|
||||
|
||||
informs = File2DocumentService.get_by_file_id(file.id)
|
||||
for inform in informs:
|
||||
doc_id = inform.document_id
|
||||
e, doc = DocumentService.get_by_id(doc_id)
|
||||
if e and doc:
|
||||
tenant_id = DocumentService.get_tenant_id(doc_id)
|
||||
if tenant_id:
|
||||
DocumentService.remove_document(doc, tenant_id)
|
||||
File2DocumentService.delete_by_file_id(file.id)
|
||||
|
||||
FileService.delete(file)
|
||||
|
||||
def _delete_folder_recursive(folder, tenant_id):
|
||||
sub_files = FileService.list_all_files_by_parent_id(folder.id)
|
||||
for sub_file in sub_files:
|
||||
if sub_file.type == FileType.FOLDER.value:
|
||||
_delete_folder_recursive(sub_file, tenant_id)
|
||||
else:
|
||||
_delete_single_file(sub_file)
|
||||
|
||||
FileService.delete(folder)
|
||||
|
||||
def _rm_sync():
|
||||
for file_id in file_ids:
|
||||
e, file = FileService.get_by_id(file_id)
|
||||
if not e or not file:
|
||||
return get_data_error_result(message="File or Folder not found!")
|
||||
if not file.tenant_id:
|
||||
return get_data_error_result(message="Tenant not found!")
|
||||
if not check_file_team_permission(file, uid):
|
||||
return get_json_result(data=False, message="No authorization.", code=RetCode.AUTHENTICATION_ERROR)
|
||||
|
||||
if file.source_type == FileSource.KNOWLEDGEBASE:
|
||||
continue
|
||||
|
||||
if file.type == FileType.FOLDER.value:
|
||||
_delete_folder_recursive(file, uid)
|
||||
continue
|
||||
|
||||
_delete_single_file(file)
|
||||
|
||||
return get_json_result(data=True)
|
||||
|
||||
return await thread_pool_exec(_rm_sync)
|
||||
|
||||
except Exception as e:
|
||||
return server_error_response(e)
|
||||
|
||||
|
||||
@manager.route('/rename', methods=['POST']) # noqa: F821
|
||||
@login_required
|
||||
@validate_request("file_id", "name")
|
||||
async def rename():
|
||||
req = await get_request_json()
|
||||
try:
|
||||
e, file = FileService.get_by_id(req["file_id"])
|
||||
if not e:
|
||||
return get_data_error_result(message="File not found!")
|
||||
if not check_file_team_permission(file, current_user.id):
|
||||
return get_json_result(data=False, message='No authorization.', code=RetCode.AUTHENTICATION_ERROR)
|
||||
if file.type != FileType.FOLDER.value \
|
||||
and pathlib.Path(req["name"].lower()).suffix != pathlib.Path(
|
||||
file.name.lower()).suffix:
|
||||
return get_json_result(
|
||||
data=False,
|
||||
message="The extension of file can't be changed",
|
||||
code=RetCode.ARGUMENT_ERROR)
|
||||
for file in FileService.query(name=req["name"], pf_id=file.parent_id):
|
||||
if file.name == req["name"]:
|
||||
return get_data_error_result(
|
||||
message="Duplicated file name in the same folder.")
|
||||
|
||||
if not FileService.update_by_id(
|
||||
req["file_id"], {"name": req["name"]}):
|
||||
return get_data_error_result(
|
||||
message="Database error (File rename)!")
|
||||
|
||||
informs = File2DocumentService.get_by_file_id(req["file_id"])
|
||||
if informs:
|
||||
if not DocumentService.update_by_id(
|
||||
informs[0].document_id, {"name": req["name"]}):
|
||||
return get_data_error_result(
|
||||
message="Database error (Document rename)!")
|
||||
|
||||
return get_json_result(data=True)
|
||||
except Exception as e:
|
||||
return server_error_response(e)
|
||||
|
||||
|
||||
@manager.route('/get/<file_id>', methods=['GET']) # noqa: F821
|
||||
@login_required
|
||||
async def get(file_id):
|
||||
try:
|
||||
e, file = FileService.get_by_id(file_id)
|
||||
if not e:
|
||||
return get_data_error_result(message="Document not found!")
|
||||
if not check_file_team_permission(file, current_user.id):
|
||||
return get_json_result(data=False, message='No authorization.', code=RetCode.AUTHENTICATION_ERROR)
|
||||
|
||||
blob = await thread_pool_exec(settings.STORAGE_IMPL.get, file.parent_id, file.location)
|
||||
if not blob:
|
||||
b, n = File2DocumentService.get_storage_address(file_id=file_id)
|
||||
blob = await thread_pool_exec(settings.STORAGE_IMPL.get, b, n)
|
||||
|
||||
response = await make_response(blob)
|
||||
ext = re.search(r"\.([^.]+)$", file.name.lower())
|
||||
ext = ext.group(1) if ext else None
|
||||
content_type = None
|
||||
if ext:
|
||||
fallback_prefix = "image" if file.type == FileType.VISUAL.value else "application"
|
||||
content_type = CONTENT_TYPE_MAP.get(ext, f"{fallback_prefix}/{ext}")
|
||||
apply_safe_file_response_headers(response, content_type, ext)
|
||||
return response
|
||||
except Exception as e:
|
||||
return server_error_response(e)
|
||||
|
||||
|
||||
@manager.route("/mv", methods=["POST"]) # noqa: F821
|
||||
@login_required
|
||||
@validate_request("src_file_ids", "dest_file_id")
|
||||
async def move():
|
||||
req = await get_request_json()
|
||||
try:
|
||||
file_ids = req["src_file_ids"]
|
||||
dest_parent_id = req["dest_file_id"]
|
||||
|
||||
ok, dest_folder = FileService.get_by_id(dest_parent_id)
|
||||
if not ok or not dest_folder:
|
||||
return get_data_error_result(message="Parent folder not found!")
|
||||
|
||||
files = FileService.get_by_ids(file_ids)
|
||||
if not files:
|
||||
return get_data_error_result(message="Source files not found!")
|
||||
|
||||
files_dict = {f.id: f for f in files}
|
||||
|
||||
for file_id in file_ids:
|
||||
file = files_dict.get(file_id)
|
||||
if not file:
|
||||
return get_data_error_result(message="File or folder not found!")
|
||||
if not file.tenant_id:
|
||||
return get_data_error_result(message="Tenant not found!")
|
||||
if not check_file_team_permission(file, current_user.id):
|
||||
return get_json_result(
|
||||
data=False,
|
||||
message="No authorization.",
|
||||
code=RetCode.AUTHENTICATION_ERROR,
|
||||
)
|
||||
|
||||
def _move_entry_recursive(source_file_entry, dest_folder):
|
||||
if source_file_entry.type == FileType.FOLDER.value:
|
||||
existing_folder = FileService.query(name=source_file_entry.name, parent_id=dest_folder.id)
|
||||
if existing_folder:
|
||||
new_folder = existing_folder[0]
|
||||
else:
|
||||
new_folder = FileService.insert(
|
||||
{
|
||||
"id": get_uuid(),
|
||||
"parent_id": dest_folder.id,
|
||||
"tenant_id": source_file_entry.tenant_id,
|
||||
"created_by": current_user.id,
|
||||
"name": source_file_entry.name,
|
||||
"location": "",
|
||||
"size": 0,
|
||||
"type": FileType.FOLDER.value,
|
||||
}
|
||||
)
|
||||
|
||||
sub_files = FileService.list_all_files_by_parent_id(source_file_entry.id)
|
||||
for sub_file in sub_files:
|
||||
_move_entry_recursive(sub_file, new_folder)
|
||||
|
||||
FileService.delete_by_id(source_file_entry.id)
|
||||
return
|
||||
|
||||
old_parent_id = source_file_entry.parent_id
|
||||
old_location = source_file_entry.location
|
||||
filename = source_file_entry.name
|
||||
|
||||
new_location = filename
|
||||
while settings.STORAGE_IMPL.obj_exist(dest_folder.id, new_location):
|
||||
new_location += "_"
|
||||
|
||||
try:
|
||||
settings.STORAGE_IMPL.move(old_parent_id, old_location, dest_folder.id, new_location)
|
||||
except Exception as storage_err:
|
||||
raise RuntimeError(f"Move file failed at storage layer: {str(storage_err)}")
|
||||
|
||||
FileService.update_by_id(
|
||||
source_file_entry.id,
|
||||
{
|
||||
"parent_id": dest_folder.id,
|
||||
"location": new_location,
|
||||
},
|
||||
)
|
||||
|
||||
def _move_sync():
|
||||
for file in files:
|
||||
_move_entry_recursive(file, dest_folder)
|
||||
return get_json_result(data=True)
|
||||
|
||||
return await thread_pool_exec(_move_sync)
|
||||
|
||||
except Exception as e:
|
||||
return server_error_response(e)
|
||||
# for file_obj in file_objs:
|
||||
# if file_obj.filename == '':
|
||||
# return get_json_result(
|
||||
# data=False, message='No file selected!', code=RetCode.ARGUMENT_ERROR)
|
||||
# file_res = []
|
||||
# try:
|
||||
# e, pf_folder = FileService.get_by_id(pf_id)
|
||||
# if not e:
|
||||
# return get_data_error_result( message="Can't find this folder!")
|
||||
#
|
||||
# async def _handle_single_file(file_obj):
|
||||
# MAX_FILE_NUM_PER_USER: int = int(os.environ.get('MAX_FILE_NUM_PER_USER', 0))
|
||||
# if 0 < MAX_FILE_NUM_PER_USER <= await thread_pool_exec(DocumentService.get_doc_count, current_user.id):
|
||||
# return get_data_error_result( message="Exceed the maximum file number of a free user!")
|
||||
#
|
||||
# # split file name path
|
||||
# if not file_obj.filename:
|
||||
# file_obj_names = [pf_folder.name, file_obj.filename]
|
||||
# else:
|
||||
# full_path = '/' + file_obj.filename
|
||||
# file_obj_names = full_path.split('/')
|
||||
# file_len = len(file_obj_names)
|
||||
#
|
||||
# # get folder
|
||||
# file_id_list = await thread_pool_exec(FileService.get_id_list_by_id, pf_id, file_obj_names, 1, [pf_id])
|
||||
# len_id_list = len(file_id_list)
|
||||
#
|
||||
# # create folder
|
||||
# if file_len != len_id_list:
|
||||
# e, file = await thread_pool_exec(FileService.get_by_id, file_id_list[len_id_list - 1])
|
||||
# if not e:
|
||||
# return get_data_error_result(message="Folder not found!")
|
||||
# last_folder = await thread_pool_exec(FileService.create_folder, file, file_id_list[len_id_list - 1], file_obj_names,
|
||||
# len_id_list)
|
||||
# else:
|
||||
# e, file = await thread_pool_exec(FileService.get_by_id, file_id_list[len_id_list - 2])
|
||||
# if not e:
|
||||
# return get_data_error_result(message="Folder not found!")
|
||||
# last_folder = await thread_pool_exec(FileService.create_folder, file, file_id_list[len_id_list - 2], file_obj_names,
|
||||
# len_id_list)
|
||||
#
|
||||
# # file type
|
||||
# filetype = filename_type(file_obj_names[file_len - 1])
|
||||
# location = file_obj_names[file_len - 1]
|
||||
# while await thread_pool_exec(settings.STORAGE_IMPL.obj_exist, last_folder.id, location):
|
||||
# location += "_"
|
||||
# blob = await thread_pool_exec(file_obj.read)
|
||||
# filename = await thread_pool_exec(
|
||||
# duplicate_name,
|
||||
# FileService.query,
|
||||
# name=file_obj_names[file_len - 1],
|
||||
# parent_id=last_folder.id)
|
||||
# await thread_pool_exec(settings.STORAGE_IMPL.put, last_folder.id, location, blob)
|
||||
# file_data = {
|
||||
# "id": get_uuid(),
|
||||
# "parent_id": last_folder.id,
|
||||
# "tenant_id": current_user.id,
|
||||
# "created_by": current_user.id,
|
||||
# "type": filetype,
|
||||
# "name": filename,
|
||||
# "location": location,
|
||||
# "size": len(blob),
|
||||
# }
|
||||
# inserted = await thread_pool_exec(FileService.insert, file_data)
|
||||
# return inserted.to_json()
|
||||
#
|
||||
# for file_obj in file_objs:
|
||||
# res = await _handle_single_file(file_obj)
|
||||
# file_res.append(res)
|
||||
#
|
||||
# return get_json_result(data=file_res)
|
||||
# except Exception as e:
|
||||
# return server_error_response(e)
|
||||
#
|
||||
#
|
||||
# @manager.route('/create', methods=['POST']) # noqa: F821
|
||||
# @login_required
|
||||
# @validate_request("name")
|
||||
# async def create():
|
||||
# req = await get_request_json()
|
||||
# pf_id = req.get("parent_id")
|
||||
# input_file_type = req.get("type")
|
||||
# if not pf_id:
|
||||
# root_folder = FileService.get_root_folder(current_user.id)
|
||||
# pf_id = root_folder["id"]
|
||||
#
|
||||
# try:
|
||||
# if not FileService.is_parent_folder_exist(pf_id):
|
||||
# return get_json_result(
|
||||
# data=False, message="Parent Folder Doesn't Exist!", code=RetCode.OPERATING_ERROR)
|
||||
# if FileService.query(name=req["name"], parent_id=pf_id):
|
||||
# return get_data_error_result(
|
||||
# message="Duplicated folder name in the same folder.")
|
||||
#
|
||||
# if input_file_type == FileType.FOLDER.value:
|
||||
# file_type = FileType.FOLDER.value
|
||||
# else:
|
||||
# file_type = FileType.VIRTUAL.value
|
||||
#
|
||||
# file = FileService.insert({
|
||||
# "id": get_uuid(),
|
||||
# "parent_id": pf_id,
|
||||
# "tenant_id": current_user.id,
|
||||
# "created_by": current_user.id,
|
||||
# "name": req["name"],
|
||||
# "location": "",
|
||||
# "size": 0,
|
||||
# "type": file_type
|
||||
# })
|
||||
#
|
||||
# return get_json_result(data=file.to_json())
|
||||
# except Exception as e:
|
||||
# return server_error_response(e)
|
||||
#
|
||||
#
|
||||
# @manager.route('/list', methods=['GET']) # noqa: F821
|
||||
# @login_required
|
||||
# def list_files():
|
||||
# pf_id = request.args.get("parent_id")
|
||||
#
|
||||
# keywords = request.args.get("keywords", "")
|
||||
#
|
||||
# page_number = int(request.args.get("page", 1))
|
||||
# items_per_page = int(request.args.get("page_size", 15))
|
||||
# orderby = request.args.get("orderby", "create_time")
|
||||
# desc = request.args.get("desc", True)
|
||||
# if not pf_id:
|
||||
# root_folder = FileService.get_root_folder(current_user.id)
|
||||
# pf_id = root_folder["id"]
|
||||
# FileService.init_knowledgebase_docs(pf_id, current_user.id)
|
||||
# try:
|
||||
# e, file = FileService.get_by_id(pf_id)
|
||||
# if not e:
|
||||
# return get_data_error_result(message="Folder not found!")
|
||||
#
|
||||
# files, total = FileService.get_by_pf_id(
|
||||
# current_user.id, pf_id, page_number, items_per_page, orderby, desc, keywords)
|
||||
#
|
||||
# parent_folder = FileService.get_parent_folder(pf_id)
|
||||
# if not parent_folder:
|
||||
# return get_json_result(message="File not found!")
|
||||
#
|
||||
# return get_json_result(data={"total": total, "files": files, "parent_folder": parent_folder.to_json()})
|
||||
# except Exception as e:
|
||||
# return server_error_response(e)
|
||||
#
|
||||
#
|
||||
# @manager.route('/root_folder', methods=['GET']) # noqa: F821
|
||||
# @login_required
|
||||
# def get_root_folder():
|
||||
# try:
|
||||
# root_folder = FileService.get_root_folder(current_user.id)
|
||||
# return get_json_result(data={"root_folder": root_folder})
|
||||
# except Exception as e:
|
||||
# return server_error_response(e)
|
||||
#
|
||||
#
|
||||
# @manager.route('/parent_folder', methods=['GET']) # noqa: F821
|
||||
# @login_required
|
||||
# def get_parent_folder():
|
||||
# file_id = request.args.get("file_id")
|
||||
# try:
|
||||
# e, file = FileService.get_by_id(file_id)
|
||||
# if not e:
|
||||
# return get_data_error_result(message="Folder not found!")
|
||||
#
|
||||
# parent_folder = FileService.get_parent_folder(file_id)
|
||||
# return get_json_result(data={"parent_folder": parent_folder.to_json()})
|
||||
# except Exception as e:
|
||||
# return server_error_response(e)
|
||||
#
|
||||
#
|
||||
# @manager.route('/all_parent_folder', methods=['GET']) # noqa: F821
|
||||
# @login_required
|
||||
# def get_all_parent_folders():
|
||||
# file_id = request.args.get("file_id")
|
||||
# try:
|
||||
# e, file = FileService.get_by_id(file_id)
|
||||
# if not e:
|
||||
# return get_data_error_result(message="Folder not found!")
|
||||
#
|
||||
# parent_folders = FileService.get_all_parent_folders(file_id)
|
||||
# parent_folders_res = []
|
||||
# for parent_folder in parent_folders:
|
||||
# parent_folders_res.append(parent_folder.to_json())
|
||||
# return get_json_result(data={"parent_folders": parent_folders_res})
|
||||
# except Exception as e:
|
||||
# return server_error_response(e)
|
||||
#
|
||||
#
|
||||
# @manager.route("/rm", methods=["POST"]) # noqa: F821
|
||||
# @login_required
|
||||
# @validate_request("file_ids")
|
||||
# async def rm():
|
||||
# req = await get_request_json()
|
||||
# file_ids = req["file_ids"]
|
||||
# uid = current_user.id
|
||||
#
|
||||
# try:
|
||||
# def _delete_single_file(file):
|
||||
# try:
|
||||
# if file.location:
|
||||
# settings.STORAGE_IMPL.rm(file.parent_id, file.location)
|
||||
# except Exception as e:
|
||||
# logging.exception(f"Fail to remove object: {file.parent_id}/{file.location}, error: {e}")
|
||||
#
|
||||
# informs = File2DocumentService.get_by_file_id(file.id)
|
||||
# for inform in informs:
|
||||
# doc_id = inform.document_id
|
||||
# e, doc = DocumentService.get_by_id(doc_id)
|
||||
# if e and doc:
|
||||
# tenant_id = DocumentService.get_tenant_id(doc_id)
|
||||
# if tenant_id:
|
||||
# DocumentService.remove_document(doc, tenant_id)
|
||||
# File2DocumentService.delete_by_file_id(file.id)
|
||||
#
|
||||
# FileService.delete(file)
|
||||
#
|
||||
# def _delete_folder_recursive(folder, tenant_id):
|
||||
# sub_files = FileService.list_all_files_by_parent_id(folder.id)
|
||||
# for sub_file in sub_files:
|
||||
# if sub_file.type == FileType.FOLDER.value:
|
||||
# _delete_folder_recursive(sub_file, tenant_id)
|
||||
# else:
|
||||
# _delete_single_file(sub_file)
|
||||
#
|
||||
# FileService.delete(folder)
|
||||
#
|
||||
# def _rm_sync():
|
||||
# for file_id in file_ids:
|
||||
# e, file = FileService.get_by_id(file_id)
|
||||
# if not e or not file:
|
||||
# return get_data_error_result(message="File or Folder not found!")
|
||||
# if not file.tenant_id:
|
||||
# return get_data_error_result(message="Tenant not found!")
|
||||
# if not check_file_team_permission(file, uid):
|
||||
# return get_json_result(data=False, message="No authorization.", code=RetCode.AUTHENTICATION_ERROR)
|
||||
#
|
||||
# if file.source_type == FileSource.KNOWLEDGEBASE:
|
||||
# continue
|
||||
#
|
||||
# if file.type == FileType.FOLDER.value:
|
||||
# _delete_folder_recursive(file, uid)
|
||||
# continue
|
||||
#
|
||||
# _delete_single_file(file)
|
||||
#
|
||||
# return get_json_result(data=True)
|
||||
#
|
||||
# return await thread_pool_exec(_rm_sync)
|
||||
#
|
||||
# except Exception as e:
|
||||
# return server_error_response(e)
|
||||
#
|
||||
#
|
||||
# @manager.route('/rename', methods=['POST']) # noqa: F821
|
||||
# @login_required
|
||||
# @validate_request("file_id", "name")
|
||||
# async def rename():
|
||||
# req = await get_request_json()
|
||||
# try:
|
||||
# e, file = FileService.get_by_id(req["file_id"])
|
||||
# if not e:
|
||||
# return get_data_error_result(message="File not found!")
|
||||
# if not check_file_team_permission(file, current_user.id):
|
||||
# return get_json_result(data=False, message='No authorization.', code=RetCode.AUTHENTICATION_ERROR)
|
||||
# if file.type != FileType.FOLDER.value \
|
||||
# and pathlib.Path(req["name"].lower()).suffix != pathlib.Path(
|
||||
# file.name.lower()).suffix:
|
||||
# return get_json_result(
|
||||
# data=False,
|
||||
# message="The extension of file can't be changed",
|
||||
# code=RetCode.ARGUMENT_ERROR)
|
||||
# for file in FileService.query(name=req["name"], pf_id=file.parent_id):
|
||||
# if file.name == req["name"]:
|
||||
# return get_data_error_result(
|
||||
# message="Duplicated file name in the same folder.")
|
||||
#
|
||||
# if not FileService.update_by_id(
|
||||
# req["file_id"], {"name": req["name"]}):
|
||||
# return get_data_error_result(
|
||||
# message="Database error (File rename)!")
|
||||
#
|
||||
# informs = File2DocumentService.get_by_file_id(req["file_id"])
|
||||
# if informs:
|
||||
# if not DocumentService.update_by_id(
|
||||
# informs[0].document_id, {"name": req["name"]}):
|
||||
# return get_data_error_result(
|
||||
# message="Database error (Document rename)!")
|
||||
#
|
||||
# return get_json_result(data=True)
|
||||
# except Exception as e:
|
||||
# return server_error_response(e)
|
||||
#
|
||||
#
|
||||
# @manager.route('/get/<file_id>', methods=['GET']) # noqa: F821
|
||||
# @login_required
|
||||
# async def get(file_id):
|
||||
# try:
|
||||
# e, file = FileService.get_by_id(file_id)
|
||||
# if not e:
|
||||
# return get_data_error_result(message="Document not found!")
|
||||
# if not check_file_team_permission(file, current_user.id):
|
||||
# return get_json_result(data=False, message='No authorization.', code=RetCode.AUTHENTICATION_ERROR)
|
||||
#
|
||||
# blob = await thread_pool_exec(settings.STORAGE_IMPL.get, file.parent_id, file.location)
|
||||
# if not blob:
|
||||
# b, n = File2DocumentService.get_storage_address(file_id=file_id)
|
||||
# blob = await thread_pool_exec(settings.STORAGE_IMPL.get, b, n)
|
||||
#
|
||||
# response = await make_response(blob)
|
||||
# ext = re.search(r"\.([^.]+)$", file.name.lower())
|
||||
# ext = ext.group(1) if ext else None
|
||||
# content_type = None
|
||||
# if ext:
|
||||
# fallback_prefix = "image" if file.type == FileType.VISUAL.value else "application"
|
||||
# content_type = CONTENT_TYPE_MAP.get(ext, f"{fallback_prefix}/{ext}")
|
||||
# apply_safe_file_response_headers(response, content_type, ext)
|
||||
# return response
|
||||
# except Exception as e:
|
||||
# return server_error_response(e)
|
||||
#
|
||||
#
|
||||
# @manager.route("/mv", methods=["POST"]) # noqa: F821
|
||||
# @login_required
|
||||
# @validate_request("src_file_ids", "dest_file_id")
|
||||
# async def move():
|
||||
# req = await get_request_json()
|
||||
# try:
|
||||
# file_ids = req["src_file_ids"]
|
||||
# dest_parent_id = req["dest_file_id"]
|
||||
#
|
||||
# ok, dest_folder = FileService.get_by_id(dest_parent_id)
|
||||
# if not ok or not dest_folder:
|
||||
# return get_data_error_result(message="Parent folder not found!")
|
||||
#
|
||||
# files = FileService.get_by_ids(file_ids)
|
||||
# if not files:
|
||||
# return get_data_error_result(message="Source files not found!")
|
||||
#
|
||||
# files_dict = {f.id: f for f in files}
|
||||
#
|
||||
# for file_id in file_ids:
|
||||
# file = files_dict.get(file_id)
|
||||
# if not file:
|
||||
# return get_data_error_result(message="File or folder not found!")
|
||||
# if not file.tenant_id:
|
||||
# return get_data_error_result(message="Tenant not found!")
|
||||
# if not check_file_team_permission(file, current_user.id):
|
||||
# return get_json_result(
|
||||
# data=False,
|
||||
# message="No authorization.",
|
||||
# code=RetCode.AUTHENTICATION_ERROR,
|
||||
# )
|
||||
#
|
||||
# def _move_entry_recursive(source_file_entry, dest_folder):
|
||||
# if source_file_entry.type == FileType.FOLDER.value:
|
||||
# existing_folder = FileService.query(name=source_file_entry.name, parent_id=dest_folder.id)
|
||||
# if existing_folder:
|
||||
# new_folder = existing_folder[0]
|
||||
# else:
|
||||
# new_folder = FileService.insert(
|
||||
# {
|
||||
# "id": get_uuid(),
|
||||
# "parent_id": dest_folder.id,
|
||||
# "tenant_id": source_file_entry.tenant_id,
|
||||
# "created_by": current_user.id,
|
||||
# "name": source_file_entry.name,
|
||||
# "location": "",
|
||||
# "size": 0,
|
||||
# "type": FileType.FOLDER.value,
|
||||
# }
|
||||
# )
|
||||
#
|
||||
# sub_files = FileService.list_all_files_by_parent_id(source_file_entry.id)
|
||||
# for sub_file in sub_files:
|
||||
# _move_entry_recursive(sub_file, new_folder)
|
||||
#
|
||||
# FileService.delete_by_id(source_file_entry.id)
|
||||
# return
|
||||
#
|
||||
# old_parent_id = source_file_entry.parent_id
|
||||
# old_location = source_file_entry.location
|
||||
# filename = source_file_entry.name
|
||||
#
|
||||
# new_location = filename
|
||||
# while settings.STORAGE_IMPL.obj_exist(dest_folder.id, new_location):
|
||||
# new_location += "_"
|
||||
#
|
||||
# try:
|
||||
# settings.STORAGE_IMPL.move(old_parent_id, old_location, dest_folder.id, new_location)
|
||||
# except Exception as storage_err:
|
||||
# raise RuntimeError(f"Move file failed at storage layer: {str(storage_err)}")
|
||||
#
|
||||
# FileService.update_by_id(
|
||||
# source_file_entry.id,
|
||||
# {
|
||||
# "parent_id": dest_folder.id,
|
||||
# "location": new_location,
|
||||
# },
|
||||
# )
|
||||
#
|
||||
# def _move_sync():
|
||||
# for file in files:
|
||||
# _move_entry_recursive(file, dest_folder)
|
||||
# return get_json_result(data=True)
|
||||
#
|
||||
# return await thread_pool_exec(_move_sync)
|
||||
#
|
||||
# except Exception as e:
|
||||
# return server_error_response(e)
|
||||
|
||||
Reference in New Issue
Block a user