mirror of
https://git.mirrors.martin98.com/https://github.com/langgenius/dify.git
synced 2025-04-22 13:49:45 +08:00
176 lines
6.9 KiB
Python
176 lines
6.9 KiB
Python
import os
|
|
from typing import Literal, Optional
|
|
|
|
import httpx
|
|
from tenacity import retry, retry_if_exception_type, stop_before_delay, wait_fixed
|
|
|
|
from extensions.ext_database import db
|
|
from libs.helper import RateLimiter
|
|
from models.account import Account, TenantAccountJoin, TenantAccountRole
|
|
|
|
|
|
class BillingService:
|
|
base_url = os.environ.get("BILLING_API_URL", "BILLING_API_URL")
|
|
secret_key = os.environ.get("BILLING_API_SECRET_KEY", "BILLING_API_SECRET_KEY")
|
|
|
|
compliance_download_rate_limiter = RateLimiter("compliance_download_rate_limiter", 4, 60)
|
|
|
|
@classmethod
|
|
def get_info(cls, tenant_id: str):
|
|
params = {"tenant_id": tenant_id}
|
|
|
|
billing_info = cls._send_request("GET", "/subscription/info", params=params)
|
|
return billing_info
|
|
|
|
@classmethod
|
|
def get_knowledge_rate_limit(cls, tenant_id: str):
|
|
params = {"tenant_id": tenant_id}
|
|
|
|
knowledge_rate_limit = cls._send_request("GET", "/subscription/knowledge-rate-limit", params=params)
|
|
|
|
return {
|
|
"limit": knowledge_rate_limit.get("limit", 10),
|
|
"subscription_plan": knowledge_rate_limit.get("subscription_plan", "sandbox"),
|
|
}
|
|
|
|
@classmethod
|
|
def get_subscription(cls, plan: str, interval: str, prefilled_email: str = "", tenant_id: str = ""):
|
|
params = {"plan": plan, "interval": interval, "prefilled_email": prefilled_email, "tenant_id": tenant_id}
|
|
return cls._send_request("GET", "/subscription/payment-link", params=params)
|
|
|
|
@classmethod
|
|
def get_model_provider_payment_link(cls, provider_name: str, tenant_id: str, account_id: str, prefilled_email: str):
|
|
params = {
|
|
"provider_name": provider_name,
|
|
"tenant_id": tenant_id,
|
|
"account_id": account_id,
|
|
"prefilled_email": prefilled_email,
|
|
}
|
|
return cls._send_request("GET", "/model-provider/payment-link", params=params)
|
|
|
|
@classmethod
|
|
def get_invoices(cls, prefilled_email: str = "", tenant_id: str = ""):
|
|
params = {"prefilled_email": prefilled_email, "tenant_id": tenant_id}
|
|
return cls._send_request("GET", "/invoices", params=params)
|
|
|
|
@classmethod
|
|
@retry(
|
|
wait=wait_fixed(2),
|
|
stop=stop_before_delay(10),
|
|
retry=retry_if_exception_type(httpx.RequestError),
|
|
reraise=True,
|
|
)
|
|
def _send_request(cls, method: Literal["GET", "POST", "DELETE"], endpoint: str, json=None, params=None):
|
|
headers = {"Content-Type": "application/json", "Billing-Api-Secret-Key": cls.secret_key}
|
|
|
|
url = f"{cls.base_url}{endpoint}"
|
|
response = httpx.request(method, url, json=json, params=params, headers=headers)
|
|
if method == "GET" and response.status_code != httpx.codes.OK:
|
|
raise ValueError("Unable to retrieve billing information. Please try again later or contact support.")
|
|
return response.json()
|
|
|
|
@staticmethod
|
|
def is_tenant_owner_or_admin(current_user):
|
|
tenant_id = current_user.current_tenant_id
|
|
|
|
join: Optional[TenantAccountJoin] = (
|
|
db.session.query(TenantAccountJoin)
|
|
.filter(TenantAccountJoin.tenant_id == tenant_id, TenantAccountJoin.account_id == current_user.id)
|
|
.first()
|
|
)
|
|
|
|
if not join:
|
|
raise ValueError("Tenant account join not found")
|
|
|
|
if not TenantAccountRole.is_privileged_role(join.role):
|
|
raise ValueError("Only team owner or team admin can perform this action")
|
|
|
|
@classmethod
|
|
def delete_account(cls, account_id: str):
|
|
"""Delete account."""
|
|
params = {"account_id": account_id}
|
|
return cls._send_request("DELETE", "/account/", params=params)
|
|
|
|
@classmethod
|
|
def is_email_in_freeze(cls, email: str) -> bool:
|
|
params = {"email": email}
|
|
try:
|
|
response = cls._send_request("GET", "/account/in-freeze", params=params)
|
|
return bool(response.get("data", False))
|
|
except Exception:
|
|
return False
|
|
|
|
@classmethod
|
|
def update_account_deletion_feedback(cls, email: str, feedback: str):
|
|
"""Update account deletion feedback."""
|
|
json = {"email": email, "feedback": feedback}
|
|
return cls._send_request("POST", "/account/delete-feedback", json=json)
|
|
|
|
class EducationIdentity:
|
|
verification_rate_limit = RateLimiter(prefix="edu_verification_rate_limit", max_attempts=10, time_window=60)
|
|
activation_rate_limit = RateLimiter(prefix="edu_activation_rate_limit", max_attempts=10, time_window=60)
|
|
|
|
@classmethod
|
|
def verify(cls, account_id: str, account_email: str):
|
|
if cls.verification_rate_limit.is_rate_limited(account_email):
|
|
from controllers.console.error import EducationVerifyLimitError
|
|
|
|
raise EducationVerifyLimitError()
|
|
|
|
cls.verification_rate_limit.increment_rate_limit(account_email)
|
|
|
|
params = {"account_id": account_id}
|
|
return BillingService._send_request("GET", "/education/verify", params=params)
|
|
|
|
@classmethod
|
|
def is_active(cls, account_id: str):
|
|
params = {"account_id": account_id}
|
|
return BillingService._send_request("GET", "/education/status", params=params)
|
|
|
|
@classmethod
|
|
def activate(cls, account: Account, token: str, institution: str, role: str):
|
|
if cls.activation_rate_limit.is_rate_limited(account.email):
|
|
from controllers.console.error import EducationActivateLimitError
|
|
|
|
raise EducationActivateLimitError()
|
|
|
|
cls.activation_rate_limit.increment_rate_limit(account.email)
|
|
params = {"account_id": account.id, "curr_tenant_id": account.current_tenant_id}
|
|
json = {
|
|
"institution": institution,
|
|
"token": token,
|
|
"role": role,
|
|
}
|
|
return BillingService._send_request("POST", "/education/", json=json, params=params)
|
|
|
|
@classmethod
|
|
def autocomplete(cls, keywords: str, page: int = 0, limit: int = 20):
|
|
params = {"keywords": keywords, "page": page, "limit": limit}
|
|
return BillingService._send_request("GET", "/education/autocomplete", params=params)
|
|
|
|
@classmethod
|
|
def get_compliance_download_link(
|
|
cls,
|
|
doc_name: str,
|
|
account_id: str,
|
|
tenant_id: str,
|
|
ip: str,
|
|
device_info: str,
|
|
):
|
|
limiter_key = f"{account_id}:{tenant_id}"
|
|
if cls.compliance_download_rate_limiter.is_rate_limited(limiter_key):
|
|
from controllers.console.error import CompilanceRateLimitError
|
|
|
|
raise CompilanceRateLimitError()
|
|
|
|
json = {
|
|
"doc_name": doc_name,
|
|
"account_id": account_id,
|
|
"tenant_id": tenant_id,
|
|
"ip_address": ip,
|
|
"device_info": device_info,
|
|
}
|
|
res = cls._send_request("POST", "/compliance/download", json=json)
|
|
cls.compliance_download_rate_limiter.increment_rate_limit(limiter_key)
|
|
return res
|