| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327 | 
							- from enum import StrEnum
 - 
 - from pydantic import BaseModel, ConfigDict, Field
 - 
 - from configs import dify_config
 - from services.billing_service import BillingService
 - from services.enterprise.enterprise_service import EnterpriseService
 - 
 - 
 - class SubscriptionModel(BaseModel):
 -     plan: str = "sandbox"
 -     interval: str = ""
 - 
 - 
 - class BillingModel(BaseModel):
 -     enabled: bool = False
 -     subscription: SubscriptionModel = SubscriptionModel()
 - 
 - 
 - class EducationModel(BaseModel):
 -     enabled: bool = False
 -     activated: bool = False
 - 
 - 
 - class LimitationModel(BaseModel):
 -     size: int = 0
 -     limit: int = 0
 - 
 - 
 - class LicenseLimitationModel(BaseModel):
 -     """
 -     - enabled: whether this limit is enforced
 -     - size: current usage count
 -     - limit: maximum allowed count; 0 means unlimited
 -     """
 - 
 -     enabled: bool = Field(False, description="Whether this limit is currently active")
 -     size: int = Field(0, description="Number of resources already consumed")
 -     limit: int = Field(0, description="Maximum number of resources allowed; 0 means no limit")
 - 
 -     def is_available(self, required: int = 1) -> bool:
 -         """
 -         Determine whether the requested amount can be allocated.
 - 
 -         Returns True if:
 -          - this limit is not active, or
 -          - the limit is zero (unlimited), or
 -          - there is enough remaining quota.
 -         """
 -         if not self.enabled or self.limit == 0:
 -             return True
 - 
 -         return (self.limit - self.size) >= required
 - 
 - 
 - class LicenseStatus(StrEnum):
 -     NONE = "none"
 -     INACTIVE = "inactive"
 -     ACTIVE = "active"
 -     EXPIRING = "expiring"
 -     EXPIRED = "expired"
 -     LOST = "lost"
 - 
 - 
 - class LicenseModel(BaseModel):
 -     status: LicenseStatus = LicenseStatus.NONE
 -     expired_at: str = ""
 -     workspaces: LicenseLimitationModel = LicenseLimitationModel(enabled=False, size=0, limit=0)
 - 
 - 
 - class BrandingModel(BaseModel):
 -     enabled: bool = False
 -     application_title: str = ""
 -     login_page_logo: str = ""
 -     workspace_logo: str = ""
 -     favicon: str = ""
 - 
 - 
 - class WebAppAuthSSOModel(BaseModel):
 -     protocol: str = ""
 - 
 - 
 - class WebAppAuthModel(BaseModel):
 -     enabled: bool = False
 -     allow_sso: bool = False
 -     sso_config: WebAppAuthSSOModel = WebAppAuthSSOModel()
 -     allow_email_code_login: bool = False
 -     allow_email_password_login: bool = False
 - 
 - 
 - class PluginInstallationScope(StrEnum):
 -     NONE = "none"
 -     OFFICIAL_ONLY = "official_only"
 -     OFFICIAL_AND_SPECIFIC_PARTNERS = "official_and_specific_partners"
 -     ALL = "all"
 - 
 - 
 - class PluginInstallationPermissionModel(BaseModel):
 -     # Plugin installation scope – possible values:
 -     #   none: prohibit all plugin installations
 -     #   official_only: allow only Dify official plugins
 -     #   official_and_specific_partners: allow official and specific partner plugins
 -     #   all: allow installation of all plugins
 -     plugin_installation_scope: PluginInstallationScope = PluginInstallationScope.ALL
 - 
 -     # If True, restrict plugin installation to the marketplace only
 -     # Equivalent to ForceEnablePluginVerification
 -     restrict_to_marketplace_only: bool = False
 - 
 - 
 - class FeatureModel(BaseModel):
 -     billing: BillingModel = BillingModel()
 -     education: EducationModel = EducationModel()
 -     members: LimitationModel = LimitationModel(size=0, limit=1)
 -     apps: LimitationModel = LimitationModel(size=0, limit=10)
 -     vector_space: LimitationModel = LimitationModel(size=0, limit=5)
 -     knowledge_rate_limit: int = 10
 -     annotation_quota_limit: LimitationModel = LimitationModel(size=0, limit=10)
 -     documents_upload_quota: LimitationModel = LimitationModel(size=0, limit=50)
 -     docs_processing: str = "standard"
 -     can_replace_logo: bool = False
 -     model_load_balancing_enabled: bool = False
 -     dataset_operator_enabled: bool = False
 -     webapp_copyright_enabled: bool = False
 -     workspace_members: LicenseLimitationModel = LicenseLimitationModel(enabled=False, size=0, limit=0)
 -     is_allow_transfer_workspace: bool = True
 -     # pydantic configs
 -     model_config = ConfigDict(protected_namespaces=())
 - 
 - 
 - class KnowledgeRateLimitModel(BaseModel):
 -     enabled: bool = False
 -     limit: int = 10
 -     subscription_plan: str = ""
 - 
 - 
 - class SystemFeatureModel(BaseModel):
 -     sso_enforced_for_signin: bool = False
 -     sso_enforced_for_signin_protocol: str = ""
 -     enable_marketplace: bool = False
 -     max_plugin_package_size: int = dify_config.PLUGIN_MAX_PACKAGE_SIZE
 -     enable_email_code_login: bool = False
 -     enable_email_password_login: bool = True
 -     enable_social_oauth_login: bool = False
 -     is_allow_register: bool = False
 -     is_allow_create_workspace: bool = False
 -     is_email_setup: bool = False
 -     license: LicenseModel = LicenseModel()
 -     branding: BrandingModel = BrandingModel()
 -     webapp_auth: WebAppAuthModel = WebAppAuthModel()
 -     plugin_installation_permission: PluginInstallationPermissionModel = PluginInstallationPermissionModel()
 -     enable_change_email: bool = True
 - 
 - 
 - class FeatureService:
 -     @classmethod
 -     def get_features(cls, tenant_id: str) -> FeatureModel:
 -         features = FeatureModel()
 - 
 -         cls._fulfill_params_from_env(features)
 - 
 -         if dify_config.BILLING_ENABLED and tenant_id:
 -             cls._fulfill_params_from_billing_api(features, tenant_id)
 - 
 -         if dify_config.ENTERPRISE_ENABLED:
 -             features.webapp_copyright_enabled = True
 -             cls._fulfill_params_from_workspace_info(features, tenant_id)
 - 
 -         return features
 - 
 -     @classmethod
 -     def get_knowledge_rate_limit(cls, tenant_id: str):
 -         knowledge_rate_limit = KnowledgeRateLimitModel()
 -         if dify_config.BILLING_ENABLED and tenant_id:
 -             knowledge_rate_limit.enabled = True
 -             limit_info = BillingService.get_knowledge_rate_limit(tenant_id)
 -             knowledge_rate_limit.limit = limit_info.get("limit", 10)
 -             knowledge_rate_limit.subscription_plan = limit_info.get("subscription_plan", "sandbox")
 -         return knowledge_rate_limit
 - 
 -     @classmethod
 -     def get_system_features(cls) -> SystemFeatureModel:
 -         system_features = SystemFeatureModel()
 - 
 -         cls._fulfill_system_params_from_env(system_features)
 - 
 -         if dify_config.ENTERPRISE_ENABLED:
 -             system_features.branding.enabled = True
 -             system_features.webapp_auth.enabled = True
 -             system_features.enable_change_email = False
 -             cls._fulfill_params_from_enterprise(system_features)
 - 
 -         if dify_config.MARKETPLACE_ENABLED:
 -             system_features.enable_marketplace = True
 - 
 -         return system_features
 - 
 -     @classmethod
 -     def _fulfill_system_params_from_env(cls, system_features: SystemFeatureModel):
 -         system_features.enable_email_code_login = dify_config.ENABLE_EMAIL_CODE_LOGIN
 -         system_features.enable_email_password_login = dify_config.ENABLE_EMAIL_PASSWORD_LOGIN
 -         system_features.enable_social_oauth_login = dify_config.ENABLE_SOCIAL_OAUTH_LOGIN
 -         system_features.is_allow_register = dify_config.ALLOW_REGISTER
 -         system_features.is_allow_create_workspace = dify_config.ALLOW_CREATE_WORKSPACE
 -         system_features.is_email_setup = dify_config.MAIL_TYPE is not None and dify_config.MAIL_TYPE != ""
 - 
 -     @classmethod
 -     def _fulfill_params_from_env(cls, features: FeatureModel):
 -         features.can_replace_logo = dify_config.CAN_REPLACE_LOGO
 -         features.model_load_balancing_enabled = dify_config.MODEL_LB_ENABLED
 -         features.dataset_operator_enabled = dify_config.DATASET_OPERATOR_ENABLED
 -         features.education.enabled = dify_config.EDUCATION_ENABLED
 - 
 -     @classmethod
 -     def _fulfill_params_from_workspace_info(cls, features: FeatureModel, tenant_id: str):
 -         workspace_info = EnterpriseService.get_workspace_info(tenant_id)
 -         if "WorkspaceMembers" in workspace_info:
 -             features.workspace_members.size = workspace_info["WorkspaceMembers"]["used"]
 -             features.workspace_members.limit = workspace_info["WorkspaceMembers"]["limit"]
 -             features.workspace_members.enabled = workspace_info["WorkspaceMembers"]["enabled"]
 - 
 -     @classmethod
 -     def _fulfill_params_from_billing_api(cls, features: FeatureModel, tenant_id: str):
 -         billing_info = BillingService.get_info(tenant_id)
 - 
 -         features.billing.enabled = billing_info["enabled"]
 -         features.billing.subscription.plan = billing_info["subscription"]["plan"]
 -         features.billing.subscription.interval = billing_info["subscription"]["interval"]
 -         features.education.activated = billing_info["subscription"].get("education", False)
 - 
 -         if features.billing.subscription.plan != "sandbox":
 -             features.webapp_copyright_enabled = True
 -         else:
 -             features.is_allow_transfer_workspace = False
 - 
 -         if "members" in billing_info:
 -             features.members.size = billing_info["members"]["size"]
 -             features.members.limit = billing_info["members"]["limit"]
 - 
 -         if "apps" in billing_info:
 -             features.apps.size = billing_info["apps"]["size"]
 -             features.apps.limit = billing_info["apps"]["limit"]
 - 
 -         if "vector_space" in billing_info:
 -             features.vector_space.size = billing_info["vector_space"]["size"]
 -             features.vector_space.limit = billing_info["vector_space"]["limit"]
 - 
 -         if "documents_upload_quota" in billing_info:
 -             features.documents_upload_quota.size = billing_info["documents_upload_quota"]["size"]
 -             features.documents_upload_quota.limit = billing_info["documents_upload_quota"]["limit"]
 - 
 -         if "annotation_quota_limit" in billing_info:
 -             features.annotation_quota_limit.size = billing_info["annotation_quota_limit"]["size"]
 -             features.annotation_quota_limit.limit = billing_info["annotation_quota_limit"]["limit"]
 - 
 -         if "docs_processing" in billing_info:
 -             features.docs_processing = billing_info["docs_processing"]
 - 
 -         if "can_replace_logo" in billing_info:
 -             features.can_replace_logo = billing_info["can_replace_logo"]
 - 
 -         if "model_load_balancing_enabled" in billing_info:
 -             features.model_load_balancing_enabled = billing_info["model_load_balancing_enabled"]
 - 
 -         if "knowledge_rate_limit" in billing_info:
 -             features.knowledge_rate_limit = billing_info["knowledge_rate_limit"]["limit"]
 - 
 -     @classmethod
 -     def _fulfill_params_from_enterprise(cls, features: SystemFeatureModel):
 -         enterprise_info = EnterpriseService.get_info()
 - 
 -         if "SSOEnforcedForSignin" in enterprise_info:
 -             features.sso_enforced_for_signin = enterprise_info["SSOEnforcedForSignin"]
 - 
 -         if "SSOEnforcedForSigninProtocol" in enterprise_info:
 -             features.sso_enforced_for_signin_protocol = enterprise_info["SSOEnforcedForSigninProtocol"]
 - 
 -         if "EnableEmailCodeLogin" in enterprise_info:
 -             features.enable_email_code_login = enterprise_info["EnableEmailCodeLogin"]
 - 
 -         if "EnableEmailPasswordLogin" in enterprise_info:
 -             features.enable_email_password_login = enterprise_info["EnableEmailPasswordLogin"]
 - 
 -         if "IsAllowRegister" in enterprise_info:
 -             features.is_allow_register = enterprise_info["IsAllowRegister"]
 - 
 -         if "IsAllowCreateWorkspace" in enterprise_info:
 -             features.is_allow_create_workspace = enterprise_info["IsAllowCreateWorkspace"]
 - 
 -         if "Branding" in enterprise_info:
 -             features.branding.application_title = enterprise_info["Branding"].get("applicationTitle", "")
 -             features.branding.login_page_logo = enterprise_info["Branding"].get("loginPageLogo", "")
 -             features.branding.workspace_logo = enterprise_info["Branding"].get("workspaceLogo", "")
 -             features.branding.favicon = enterprise_info["Branding"].get("favicon", "")
 - 
 -         if "WebAppAuth" in enterprise_info:
 -             features.webapp_auth.allow_sso = enterprise_info["WebAppAuth"].get("allowSso", False)
 -             features.webapp_auth.allow_email_code_login = enterprise_info["WebAppAuth"].get(
 -                 "allowEmailCodeLogin", False
 -             )
 -             features.webapp_auth.allow_email_password_login = enterprise_info["WebAppAuth"].get(
 -                 "allowEmailPasswordLogin", False
 -             )
 -             features.webapp_auth.sso_config.protocol = enterprise_info.get("SSOEnforcedForWebProtocol", "")
 - 
 -         if "License" in enterprise_info:
 -             license_info = enterprise_info["License"]
 - 
 -             if "status" in license_info:
 -                 features.license.status = LicenseStatus(license_info.get("status", LicenseStatus.INACTIVE))
 - 
 -             if "expiredAt" in license_info:
 -                 features.license.expired_at = license_info["expiredAt"]
 - 
 -             if "workspaces" in license_info:
 -                 features.license.workspaces.enabled = license_info["workspaces"]["enabled"]
 -                 features.license.workspaces.limit = license_info["workspaces"]["limit"]
 -                 features.license.workspaces.size = license_info["workspaces"]["used"]
 - 
 -         if "PluginInstallationPermission" in enterprise_info:
 -             plugin_installation_info = enterprise_info["PluginInstallationPermission"]
 -             features.plugin_installation_permission.plugin_installation_scope = plugin_installation_info[
 -                 "pluginInstallationScope"
 -             ]
 -             features.plugin_installation_permission.restrict_to_marketplace_only = plugin_installation_info[
 -                 "restrictToMarketplaceOnly"
 -             ]
 
 
  |