import json from datetime import datetime from urllib.parse import urlparse, parse_qs import httpx from typing import Union, Dict, List from fastapi import HTTPException from starlette import status # from Log import logger from app.config.config import settings # from app.service.service_token import get_admin_token from app.utils.rsa_crypto import RagflowCrypto class DifyService: def __init__(self, base_url: str): self.base_url = base_url def _handle_response(self, response: httpx.Response) -> Union[Dict, List]: if response.status_code != 200: if response.status_code == 201: return response.json() return {} data = response.json() ret_code = data.get("retcode") if ret_code == 401: raise HTTPException( status_code=status.HTTP_401_UNAUTHORIZED, detail="登录过期", ) # if ret_code != 0: # return {} # 检查返回的数据类型 if isinstance(data.get("data"), dict): return data.get("data", {}) elif isinstance(data.get("data"), list): return data.get("data", []) else: return data async def register(self, username: str, password: str, token=None): # if not token: # token = await get_admin_token() email = f"{username}@basic.com" invite_res = await self.invite_workspaces_member(token, [email], "admin") # print(invite_res) if invite_res.get("result") != "success" or not invite_res.get("invitation_results"): # logger.error(invite_res) return {} invite_token = \ parse_qs(urlparse(invite_res.get("invitation_results")[0].get("url", "")).query).get('token', [None])[0] # print(invite_token) if not invite_token: return {} await self.login(username, password, True, invite_token) activate_res = await self.activate(email, username, invite_token, "", "") activate_res["email"] = email activate_res["id"] = invite_token return activate_res async def invite_workspaces_member(self, token, emails: list, role: str): async with httpx.AsyncClient() as client: response = await client.post( f"{self.base_url}/console/api/workspaces/current/members/invite-email", headers={'Content-Type': 'application/json',"Authorization": f'Bearer {token}'}, json={"emails": emails, "language": "zh-Hans", "role": role} ) # print(response.text) return self._handle_response(response) async def login(self, username: str, password: str, remember_me=True, invite_token:str="", email="") -> str: # password = RagflowCrypto(settings.PUBLIC_KEY, settings.PRIVATE_KEY).encrypt(password) data = {"email":username if "@" in username else f"{username}@basic.com", "password": password, "remember_me": remember_me, "invite_token": invite_token, "language": "zh-Hans"} async with httpx.AsyncClient() as client: response = await client.post( f"{self.base_url}/console/api/login", headers={'Content-Type': 'application/json'}, json=data ) if response.status_code != 200: raise Exception(f"df login failed: {response.text}") data = self._handle_response(response) return data.get('access_token') async def email_check(self, token, email: str): async with httpx.AsyncClient() as client: response = await client.get( f"{self.base_url}/console/api/activate/check?email={email}&token={token}", headers={'Content-Type': 'application/json'} ) return self._handle_response(response) async def activate(self, email: str, name: str, token, workspace_id:str, access_token) -> str: # password = RagflowCrypto(settings.PUBLIC_KEY, settings.PRIVATE_KEY).encrypt(password) data = {"email": email, "name": name, "token": token, # "workspace_id": workspace_id, "interface_language": "en-US", "timezone": "Asia/Shanghai"} # print(data) async with httpx.AsyncClient() as client: response = await client.post( f"{self.base_url}/console/api/activate", headers={'Content-Type': 'application/json'}, # , 'Authorization': f'Bearer {access_token}' json=data ) if response.status_code != 200: raise Exception(f"df login failed: {response.text}") return self._handle_response(response) async def invite_member_activate(self, token:str, email: str, name: str, password:str) -> str: invite_res = await self.invite_workspaces_member(token, [email], "admin") # print(invite_res) if invite_res.get("result") != "success" or not invite_res.get("invitation_results"): # logger.error(invite_res) return {} invite_token = parse_qs(urlparse(invite_res.get("invitation_results")[0].get("url", "")).query).get('token', [None])[0] # print(invite_token) if not invite_token: return {} await self.login(email, password, True, invite_token) activate_res = await self.activate(email, name,invite_token, "", "") return activate_res async def chat(self, token: str, user_id: int, message: str, files: [], conversation_id: str, inputs: dict): target_url = f"{self.base_url}/v1/chat-messages" data = { "inputs": inputs, "query": message, "response_mode": "streaming", "conversation_id": conversation_id, "user": str(user_id), "files": files } async with httpx.AsyncClient(timeout=300.0) as client: headers = { 'Content-Type': 'application/json', 'Authorization': f'Bearer {token}' } async with client.stream("POST", target_url, data=json.dumps(data), headers=headers) as response: if response.status_code == 200: try: async for answer in response.aiter_text(): # print(f"response of ragflow chat: {answer}") yield answer except GeneratorExit as e: print(e) return else: yield f"Error: {response.status_code}" async def get_session_history(self, token: str, conversation_id: str, user: str): url = f"{self.base_url}/v1/messages" params = { 'user': user, 'conversation_id': conversation_id } headers = {"Authorization": f'Bearer {token}'} async with httpx.AsyncClient() as client: response = await client.get(url, params=params, headers=headers) # print(response.text) # print(response.status_code) # print(response.res) data = self._handle_response(response) # print("----------------data----------------------:", data) return data async def upload(self, token: str, filename: str, file: bytes, user_id) -> dict: url = f"{self.base_url}/v1/files/upload" headers = { # 'Content-Type': 'application/json', 'Authorization': f'Bearer {token}' } data = { 'user': str(user_id) } # 创建表单数据,包含文件 files = {"file": (filename, file)} async with httpx.AsyncClient() as client: response = await client.post(url, headers=headers, files=files, data=data) data = self._handle_response(response) return data async def save_images(self, url: str, filename: str): url = f"{self.base_url}{url}" async with httpx.AsyncClient() as client: response = await client.get(url) response.raise_for_status() # 打开一个文件用于写入 with open(f"app/images/{filename}", 'wb') as f: # 写入请求的内容 f.write(response.content) async def workflow(self, token: str, user_id: int, inputs: dict): target_url = f"{self.base_url}/v1/workflows/run" data = { "inputs": inputs, "response_mode": "streaming", "user": str(user_id), "files":[] } async with httpx.AsyncClient(timeout=1800) as client: headers = { 'Content-Type': 'application/json', 'Authorization': f'Bearer {token}' } async with client.stream("POST", target_url, data=json.dumps(data), headers=headers) as response: if response.status_code == 200: try: async for answer in response.aiter_text(): # print(f"response of ragflow chat: {answer}") yield answer except GeneratorExit as e: print(e) return else: yield f"Error: {response.status_code}" if __name__ == "__main__": async def a(): a = DifyService("http://192.168.20.116") # b = await a.invite_workspaces_member("eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1c2VyX2lkIjoiMjE0Y2I2ODctZTZlMC00ZTE2LWExNzUtYzcyNDNlMGRhMWEwIiwiZXhwIjoxNzMzNzI2NDA5LCJpc3MiOiJTRUxGX0hPU1RFRCIsInN1YiI6IkNvbnNvbGUgQVBJIFBhc3Nwb3J0In0.jLe1ODbcqCe79CDt6fFwnuuQL4I2FB9YTs9ynk4FeoQ", ["test05@163.com"], # "admin") # b = await a.email_check( # "ebd36739-0272-4b3f-95ab-0c6ac1639831", # "test05@163.com") b = await a.login( "zhao1234567", "zhaoqg123456") # b = await a.activate( # "test05@163.com", # "test05", "ebd36739-0272-4b3f-95ab-0c6ac1639831", "", "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1c2VyX2lkIjoiNzg5OWUzOGQtNzczOS00NGNmLTgyODItZmFlMGZhNDJlZDYwIiwiZXhwIjoxNzMzNzI5NjQyLCJpc3MiOiJTRUxGX0hPU1RFRCIsInN1YiI6IkNvbnNvbGUgQVBJIFBhc3Nwb3J0In0.YMvypPnrvvUIfqzcESj820nP46IsFdTpF_YPz8_Exso") # b = await a.invite_member_activate( # "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1c2VyX2lkIjoiMDE2NTcxNjAtZTllYi00NzVhLWIzMzYtZjlmZWJlY2I5YjczIiwiZXhwIjoxNzMzNzM0ODE0LCJpc3MiOiJTRUxGX0hPU1RFRCIsInN1YiI6IkNvbnNvbGUgQVBJIFBhc3Nwb3J0In0.khaXX3ndDe_pccEHcyTUcO2sgBCEfXCR74ZniP_b54Y", # "zhao1@df.com", # "zhao1Q", # "ZHAOQG123456") print(b) import asyncio asyncio.run(a())