mirror of
				https://github.com/langgenius/dify.git
				synced 2025-11-03 20:33:00 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			94 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			94 lines
		
	
	
		
			3.6 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 models.account import 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")
 | 
						|
 | 
						|
    @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_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)
 |