mirror of
https://github.com/infiniflow/ragflow.git
synced 2026-06-29 15:31:05 +08:00
This reverts PR #14775 commit 5a5e766386.
This commit is contained in:
@@ -459,32 +459,21 @@ async def list_chats():
|
||||
page_number = int(request.args.get("page", 0))
|
||||
items_per_page = validate_rest_api_page_size(int(request.args.get("page_size", 0)))
|
||||
|
||||
tenants = TenantService.get_joined_tenants_by_user_id(current_user.id)
|
||||
authorized_owner_ids = {member["tenant_id"] for member in tenants}
|
||||
authorized_owner_ids.add(current_user.id)
|
||||
|
||||
if owner_ids:
|
||||
requested_owner_ids = set(owner_ids)
|
||||
unauthorized_owner_ids = requested_owner_ids - authorized_owner_ids
|
||||
if unauthorized_owner_ids:
|
||||
logging.warning(
|
||||
"Rejected list_chats request: user=%s attempted unauthorized owner_ids=%s",
|
||||
current_user.id,
|
||||
sorted(unauthorized_owner_ids),
|
||||
)
|
||||
return get_json_result(
|
||||
data=False,
|
||||
message="Only authorized owner_ids can be queried.",
|
||||
code=RetCode.OPERATING_ERROR,
|
||||
)
|
||||
effective_owner_ids = list(requested_owner_ids)
|
||||
chats, total = await thread_pool_exec(
|
||||
DialogService.get_by_tenant_ids,
|
||||
owner_ids, current_user.id, 0, 0, orderby, desc, keywords, **exact_filters,
|
||||
)
|
||||
chats = [chat for chat in chats if chat["tenant_id"] in owner_ids]
|
||||
total = len(chats)
|
||||
if page_number and items_per_page:
|
||||
start = (page_number - 1) * items_per_page
|
||||
chats = chats[start : start + items_per_page]
|
||||
else:
|
||||
effective_owner_ids = list(authorized_owner_ids)
|
||||
|
||||
chats, total = await thread_pool_exec(
|
||||
DialogService.get_by_tenant_ids,
|
||||
effective_owner_ids, current_user.id, page_number, items_per_page, orderby, desc, keywords, **exact_filters,
|
||||
)
|
||||
chats, total = await thread_pool_exec(
|
||||
DialogService.get_by_tenant_ids,
|
||||
[], current_user.id, page_number, items_per_page, orderby, desc, keywords, **exact_filters,
|
||||
)
|
||||
|
||||
return get_json_result(
|
||||
data={"chats": [_build_chat_response(chat) for chat in chats], "total": total}
|
||||
|
||||
@@ -84,31 +84,15 @@ def list_searches():
|
||||
owner_ids = request.args.getlist("owner_ids")
|
||||
|
||||
try:
|
||||
tenants = TenantService.get_joined_tenants_by_user_id(current_user.id)
|
||||
authorized_owner_ids = {member["tenant_id"] for member in tenants}
|
||||
authorized_owner_ids.add(current_user.id)
|
||||
|
||||
if owner_ids:
|
||||
requested_owner_ids = set(owner_ids)
|
||||
unauthorized_owner_ids = requested_owner_ids - authorized_owner_ids
|
||||
if unauthorized_owner_ids:
|
||||
logging.warning(
|
||||
"Rejected list_searches request: user=%s attempted unauthorized owner_ids=%s",
|
||||
current_user.id,
|
||||
sorted(unauthorized_owner_ids),
|
||||
)
|
||||
return get_json_result(
|
||||
data=False,
|
||||
message="Only authorized owner_ids can be queried.",
|
||||
code=RetCode.OPERATING_ERROR,
|
||||
)
|
||||
effective_owner_ids = list(requested_owner_ids)
|
||||
if not owner_ids:
|
||||
tenants = []
|
||||
search_apps, total = SearchService.get_by_tenant_ids(tenants, current_user.id, page_number, items_per_page, orderby, desc, keywords)
|
||||
else:
|
||||
effective_owner_ids = list(authorized_owner_ids)
|
||||
|
||||
search_apps, total = SearchService.get_by_tenant_ids(
|
||||
effective_owner_ids, current_user.id, page_number, items_per_page, orderby, desc, keywords
|
||||
)
|
||||
search_apps, total = SearchService.get_by_tenant_ids(owner_ids, current_user.id, 0, 0, orderby, desc, keywords)
|
||||
search_apps = [s for s in search_apps if s["tenant_id"] in owner_ids]
|
||||
total = len(search_apps)
|
||||
if page_number and items_per_page:
|
||||
search_apps = search_apps[(page_number - 1) * items_per_page: page_number * items_per_page]
|
||||
return get_json_result(data={"search_apps": search_apps, "total": total})
|
||||
except Exception as e:
|
||||
return server_error_response(e)
|
||||
|
||||
@@ -1251,7 +1251,7 @@ def test_chat_create_uses_direct_chat_fields_unit(monkeypatch):
|
||||
|
||||
|
||||
@pytest.mark.p2
|
||||
def test_list_chats_defaults_to_authorized_owner_ids_when_omitted_unit(monkeypatch):
|
||||
def test_list_chats_passes_empty_owner_ids_when_omitted_unit(monkeypatch):
|
||||
module = _load_chat_routes_unit_module(monkeypatch)
|
||||
captured = {}
|
||||
monkeypatch.setattr(
|
||||
@@ -1280,12 +1280,13 @@ def test_list_chats_defaults_to_authorized_owner_ids_when_omitted_unit(monkeypat
|
||||
monkeypatch.setattr(module.DialogService, "get_by_tenant_ids", _get_by_tenant_ids)
|
||||
res = _run(module.list_chats.__wrapped__())
|
||||
assert res["code"] == 0
|
||||
assert set(captured["owner_ids"]) == {"tenant-1", "team-tenant-2"}
|
||||
assert captured["owner_ids"] == []
|
||||
|
||||
|
||||
@pytest.mark.p2
|
||||
def test_list_chats_rejects_unauthorized_owner_ids_unit(monkeypatch):
|
||||
def test_list_chats_filters_by_requested_owner_ids_unit(monkeypatch):
|
||||
module = _load_chat_routes_unit_module(monkeypatch)
|
||||
captured = {}
|
||||
monkeypatch.setattr(
|
||||
module,
|
||||
"request",
|
||||
@@ -1293,20 +1294,30 @@ def test_list_chats_rejects_unauthorized_owner_ids_unit(monkeypatch):
|
||||
args=SimpleNamespace(
|
||||
get=lambda key, default=None: {
|
||||
"keywords": "",
|
||||
"page": "0",
|
||||
"page_size": "0",
|
||||
"page": "1",
|
||||
"page_size": "10",
|
||||
"orderby": "create_time",
|
||||
"desc": "true",
|
||||
"id": None,
|
||||
"name": None,
|
||||
}.get(key, default),
|
||||
getlist=lambda key: ["foreign-tenant-id"] if key == "owner_ids" else [],
|
||||
getlist=lambda key: ["team-tenant-2"] if key == "owner_ids" else [],
|
||||
)
|
||||
),
|
||||
)
|
||||
|
||||
def _get_by_tenant_ids(owner_ids, *_args, **_kwargs):
|
||||
captured["owner_ids"] = owner_ids
|
||||
team_chat = _DummyDialogRecord({"id": "team-chat", "tenant_id": "team-tenant-2", "name": "team"}).to_dict()
|
||||
own_chat = _DummyDialogRecord({"id": "own-chat", "tenant_id": "tenant-1", "name": "own"}).to_dict()
|
||||
return ([team_chat, own_chat], 2)
|
||||
|
||||
monkeypatch.setattr(module.DialogService, "get_by_tenant_ids", _get_by_tenant_ids)
|
||||
res = _run(module.list_chats.__wrapped__())
|
||||
assert res["code"] == module.RetCode.OPERATING_ERROR
|
||||
assert "authorized owner_ids" in res["message"]
|
||||
assert res["code"] == 0
|
||||
assert captured["owner_ids"] == ["team-tenant-2"]
|
||||
assert [chat["id"] for chat in res["data"]["chats"]] == ["team-chat"]
|
||||
assert res["data"]["total"] == 1
|
||||
|
||||
|
||||
@pytest.mark.p2
|
||||
|
||||
@@ -18,7 +18,7 @@ from common import batch_create_chat_assistants, delete_all_chat_assistants, get
|
||||
from utils import wait_for
|
||||
|
||||
|
||||
@wait_for(200, 1, "Document parsing timeout")
|
||||
@wait_for(30, 1, "Document parsing timeout")
|
||||
def condition(_auth, _dataset_id):
|
||||
res = list_documents(_auth, _dataset_id)
|
||||
for doc in res["data"]["docs"]:
|
||||
|
||||
@@ -201,7 +201,6 @@ def _load_chat_module(monkeypatch):
|
||||
class _StubRetCode(int, Enum):
|
||||
SUCCESS = 0
|
||||
DATA_ERROR = 102
|
||||
OPERATING_ERROR = 103
|
||||
AUTHENTICATION_ERROR = 109
|
||||
|
||||
class _StubStatusEnum(str, Enum):
|
||||
@@ -377,10 +376,6 @@ def _load_chat_module(monkeypatch):
|
||||
def get_by_id(_tenant_id):
|
||||
return True, SimpleNamespace(llm_id="glm-4")
|
||||
|
||||
@staticmethod
|
||||
def get_joined_tenants_by_user_id(_user_id):
|
||||
return [{"tenant_id": "tenant-1"}, {"tenant_id": "team-tenant-2"}]
|
||||
|
||||
class _StubUserTenantService:
|
||||
@staticmethod
|
||||
def query(**_kwargs):
|
||||
@@ -887,112 +882,6 @@ def test_list_chats_keeps_zero_pagination_semantics(monkeypatch):
|
||||
assert len(res["data"]["chats"]) == 1
|
||||
|
||||
|
||||
@pytest.mark.p2
|
||||
def test_list_chats_rejects_unauthorized_owner_ids(monkeypatch):
|
||||
module = _load_chat_module(monkeypatch)
|
||||
monkeypatch.setattr(
|
||||
module,
|
||||
"request",
|
||||
SimpleNamespace(
|
||||
args=SimpleNamespace(
|
||||
get=lambda key, default=None: {
|
||||
"keywords": "",
|
||||
"page": "0",
|
||||
"page_size": "0",
|
||||
"orderby": "create_time",
|
||||
"desc": "true",
|
||||
"id": None,
|
||||
"name": None,
|
||||
}.get(key, default),
|
||||
getlist=lambda key: ["foreign-tenant-id"] if key == "owner_ids" else [],
|
||||
)
|
||||
),
|
||||
)
|
||||
res = _run(module.list_chats.__wrapped__())
|
||||
assert res["code"] == module.RetCode.OPERATING_ERROR
|
||||
assert "authorized owner_ids" in res["message"]
|
||||
|
||||
|
||||
@pytest.mark.p2
|
||||
def test_list_chats_authorized_multi_tenant(monkeypatch):
|
||||
module = _load_chat_module(monkeypatch)
|
||||
captured = {}
|
||||
monkeypatch.setattr(
|
||||
module,
|
||||
"request",
|
||||
SimpleNamespace(
|
||||
args=SimpleNamespace(
|
||||
get=lambda key, default=None: {
|
||||
"keywords": "",
|
||||
"page": "1",
|
||||
"page_size": "10",
|
||||
"orderby": "create_time",
|
||||
"desc": "true",
|
||||
"id": None,
|
||||
"name": None,
|
||||
}.get(key, default),
|
||||
getlist=lambda key: ["tenant-1", "team-tenant-2"] if key == "owner_ids" else [],
|
||||
)
|
||||
),
|
||||
)
|
||||
|
||||
def _get_by_tenant_ids(owner_ids, user_id, *args, **kwargs):
|
||||
captured["owner_ids"] = owner_ids
|
||||
captured["user_id"] = user_id
|
||||
return (
|
||||
[
|
||||
{**_DummyDialogRecord().to_dict(), "tenant_id": "tenant-1", "id": "c1"},
|
||||
{**_DummyDialogRecord().to_dict(), "tenant_id": "team-tenant-2", "id": "c2"},
|
||||
],
|
||||
2,
|
||||
)
|
||||
|
||||
monkeypatch.setattr(module.DialogService, "get_by_tenant_ids", _get_by_tenant_ids)
|
||||
monkeypatch.setattr(module.KnowledgebaseService, "get_by_id", lambda _id: (True, _DummyKB()))
|
||||
|
||||
res = _run(module.list_chats.__wrapped__())
|
||||
assert res["code"] == 0
|
||||
assert res["data"]["total"] == 2
|
||||
assert {c["id"] for c in res["data"]["chats"]} == {"c1", "c2"}
|
||||
assert set(captured["owner_ids"]) == {"tenant-1", "team-tenant-2"}
|
||||
assert captured["user_id"] == "tenant-1"
|
||||
|
||||
|
||||
@pytest.mark.p2
|
||||
def test_list_chats_defaults_to_authorized_owner_ids_when_omitted(monkeypatch):
|
||||
module = _load_chat_module(monkeypatch)
|
||||
captured = {}
|
||||
|
||||
monkeypatch.setattr(
|
||||
module,
|
||||
"request",
|
||||
SimpleNamespace(
|
||||
args=SimpleNamespace(
|
||||
get=lambda key, default=None: {
|
||||
"keywords": "",
|
||||
"page": "1",
|
||||
"page_size": "10",
|
||||
"orderby": "create_time",
|
||||
"desc": "true",
|
||||
"id": None,
|
||||
"name": None,
|
||||
}.get(key, default),
|
||||
getlist=lambda _key: [],
|
||||
)
|
||||
),
|
||||
)
|
||||
|
||||
def _get_by_tenant_ids(owner_ids, *_args, **_kwargs):
|
||||
captured["owner_ids"] = owner_ids
|
||||
return ([], 0)
|
||||
|
||||
monkeypatch.setattr(module.DialogService, "get_by_tenant_ids", _get_by_tenant_ids)
|
||||
res = _run(module.list_chats.__wrapped__())
|
||||
|
||||
assert res["code"] == 0
|
||||
assert set(captured["owner_ids"]) == {"tenant-1", "team-tenant-2"}
|
||||
|
||||
|
||||
@pytest.mark.p2
|
||||
def test_chat_session_create_and_update_guard_matrix_unit(monkeypatch):
|
||||
module = _load_chat_module(monkeypatch)
|
||||
|
||||
@@ -225,10 +225,6 @@ def _load_search_api(monkeypatch):
|
||||
def get_by_id(_tenant_id):
|
||||
return True, SimpleNamespace(id=_tenant_id)
|
||||
|
||||
@staticmethod
|
||||
def get_joined_tenants_by_user_id(_user_id):
|
||||
return [{"tenant_id": "tenant-1"}, {"tenant_id": "team-tenant-2"}]
|
||||
|
||||
class _UserTenantService:
|
||||
@staticmethod
|
||||
def query(**_kwargs):
|
||||
@@ -495,30 +491,19 @@ def test_list_and_delete_route_matrix_unit(monkeypatch):
|
||||
module,
|
||||
{"keywords": "k", "page": "1", "page_size": "1", "orderby": "create_time", "desc": "true", "owner_ids": ["tenant-1"]},
|
||||
)
|
||||
|
||||
def _get_by_tenant_ids_filtered(tenants, _uid, page, size, _orderby, _desc, _keywords):
|
||||
all_items = [{"id": "x", "tenant_id": "tenant-1"}, {"id": "y", "tenant_id": "tenant-1"}]
|
||||
filtered = [item for item in all_items if item["tenant_id"] in set(tenants)]
|
||||
total = len(filtered)
|
||||
if page and size:
|
||||
filtered = filtered[(page - 1) * size : page * size]
|
||||
return filtered, total
|
||||
|
||||
monkeypatch.setattr(module.SearchService, "get_by_tenant_ids", _get_by_tenant_ids_filtered)
|
||||
res = module.list_searches()
|
||||
assert res["code"] == 0
|
||||
assert res["data"]["total"] == 2
|
||||
assert len(res["data"]["search_apps"]) == 1
|
||||
|
||||
# list: unauthorized owner_ids
|
||||
_set_request_args(
|
||||
monkeypatch,
|
||||
module,
|
||||
{"keywords": "", "page": "0", "page_size": "10", "orderby": "create_time", "desc": "true", "owner_ids": ["other-tenant"]},
|
||||
monkeypatch.setattr(
|
||||
module.SearchService,
|
||||
"get_by_tenant_ids",
|
||||
lambda _tenants, _uid, _page, _size, _orderby, _desc, _keywords: (
|
||||
[{"id": "x", "tenant_id": "tenant-1"}, {"id": "y", "tenant_id": "tenant-2"}],
|
||||
2,
|
||||
),
|
||||
)
|
||||
res = module.list_searches()
|
||||
assert res["code"] == module.RetCode.OPERATING_ERROR
|
||||
assert "authorized owner_ids" in res["message"]
|
||||
assert res["code"] == 0
|
||||
assert res["data"]["total"] == 1
|
||||
assert len(res["data"]["search_apps"]) == 1
|
||||
assert res["data"]["search_apps"][0]["tenant_id"] == "tenant-1"
|
||||
|
||||
# list: exception
|
||||
def _raise_list(*_args, **_kwargs):
|
||||
@@ -557,63 +542,3 @@ def test_list_and_delete_route_matrix_unit(monkeypatch):
|
||||
res = module.delete_search(search_id="search-1")
|
||||
assert res["code"] == module.RetCode.EXCEPTION_ERROR
|
||||
assert "rm boom" in res["message"]
|
||||
|
||||
|
||||
@pytest.mark.p2
|
||||
def test_list_searches_authorized_multi_tenant(monkeypatch):
|
||||
module = _load_search_api(monkeypatch)
|
||||
captured = {}
|
||||
|
||||
_set_request_args(
|
||||
monkeypatch,
|
||||
module,
|
||||
{
|
||||
"keywords": "",
|
||||
"page": "1",
|
||||
"page_size": "10",
|
||||
"orderby": "create_time",
|
||||
"desc": "true",
|
||||
"owner_ids": ["tenant-1", "team-tenant-2"],
|
||||
},
|
||||
)
|
||||
|
||||
def _get_by_tenant_ids(owner_ids, user_id, *args, **kwargs):
|
||||
captured["owner_ids"] = owner_ids
|
||||
captured["user_id"] = user_id
|
||||
return (
|
||||
[
|
||||
{"id": "s1", "tenant_id": "tenant-1"},
|
||||
{"id": "s2", "tenant_id": "team-tenant-2"},
|
||||
],
|
||||
2,
|
||||
)
|
||||
|
||||
monkeypatch.setattr(module.SearchService, "get_by_tenant_ids", _get_by_tenant_ids)
|
||||
res = module.list_searches()
|
||||
assert res["code"] == 0
|
||||
assert res["data"]["total"] == 2
|
||||
assert {s["id"] for s in res["data"]["search_apps"]} == {"s1", "s2"}
|
||||
assert set(captured["owner_ids"]) == {"tenant-1", "team-tenant-2"}
|
||||
assert captured["user_id"] == "tenant-1"
|
||||
|
||||
|
||||
@pytest.mark.p2
|
||||
def test_list_searches_defaults_to_authorized_owner_ids_when_omitted(monkeypatch):
|
||||
module = _load_search_api(monkeypatch)
|
||||
captured = {}
|
||||
|
||||
_set_request_args(
|
||||
monkeypatch,
|
||||
module,
|
||||
{"keywords": "", "page": "1", "page_size": "10", "orderby": "create_time", "desc": "true"},
|
||||
)
|
||||
|
||||
def _get_by_tenant_ids(owner_ids, *_args, **_kwargs):
|
||||
captured["owner_ids"] = owner_ids
|
||||
return ([], 0)
|
||||
|
||||
monkeypatch.setattr(module.SearchService, "get_by_tenant_ids", _get_by_tenant_ids)
|
||||
res = module.list_searches()
|
||||
|
||||
assert res["code"] == 0
|
||||
assert set(captured["owner_ids"]) == {"tenant-1", "team-tenant-2"}
|
||||
|
||||
Reference in New Issue
Block a user