Files
Dorod-Sky/skyvern/forge/sdk/schemas/credentials.py
2026-02-16 23:43:25 -05:00

206 lines
8.6 KiB
Python

from datetime import datetime
from enum import StrEnum
from pydantic import BaseModel, ConfigDict, Field
class CredentialVaultType(StrEnum):
BITWARDEN = "bitwarden"
AZURE_VAULT = "azure_vault"
CUSTOM = "custom"
class CredentialType(StrEnum):
"""Type of credential stored in the system."""
PASSWORD = "password"
CREDIT_CARD = "credit_card"
SECRET = "secret"
class TotpType(StrEnum):
"""Type of 2FA/TOTP method used."""
AUTHENTICATOR = "authenticator"
EMAIL = "email"
TEXT = "text"
NONE = "none"
class PasswordCredentialResponse(BaseModel):
"""Response model for password credentials — non-sensitive fields only.
SECURITY: Must NEVER include password, TOTP secret, or TOTP identifier.
"""
username: str = Field(..., description="The username associated with the credential", examples=["user@example.com"])
totp_type: TotpType = Field(
TotpType.NONE,
description="Type of 2FA method used for this credential",
examples=[TotpType.AUTHENTICATOR],
)
totp_identifier: str | None = Field(
default=None,
description="Identifier (email or phone number) used to fetch TOTP codes",
examples=["user@example.com", "+14155550123"],
)
class CreditCardCredentialResponse(BaseModel):
"""Response model for credit card credentials — non-sensitive fields only.
SECURITY: Must NEVER include full card number, CVV, expiration date, or card holder name.
"""
last_four: str = Field(..., description="Last four digits of the credit card number", examples=["1234"])
brand: str = Field(..., description="Brand of the credit card", examples=["visa"])
class SecretCredentialResponse(BaseModel):
"""Response model for secret credentials — non-sensitive fields only.
SECURITY: Must NEVER include the secret_value.
"""
secret_label: str | None = Field(default=None, description="Optional label for the stored secret")
class PasswordCredential(BaseModel):
"""Base model for password credentials."""
password: str = Field(..., description="The password value", examples=["securepassword123"])
username: str = Field(..., description="The username associated with the credential", examples=["user@example.com"])
totp: str | None = Field(
None,
description="Optional TOTP (Time-based One-Time Password) string used to generate 2FA codes",
examples=["JBSWY3DPEHPK3PXP"],
)
totp_type: TotpType = Field(
TotpType.NONE,
description="Type of 2FA method used for this credential",
examples=[TotpType.AUTHENTICATOR],
)
totp_identifier: str | None = Field(
default=None,
description="Identifier (email or phone number) used to fetch TOTP codes",
examples=["user@example.com", "+14155550123"],
)
class NonEmptyPasswordCredential(PasswordCredential):
"""Password credential model that requires non-empty values."""
password: str = Field(
..., min_length=1, description="The password value (must not be empty)", examples=["securepassword123"]
)
username: str = Field(
...,
min_length=1,
description="The username associated with the credential (must not be empty)",
examples=["user@example.com"],
)
class CreditCardCredential(BaseModel):
"""Base model for credit card credentials."""
card_number: str = Field(..., description="The full credit card number", examples=["4111111111111111"])
card_cvv: str = Field(..., description="The card's CVV (Card Verification Value)", examples=["123"])
card_exp_month: str = Field(..., description="The card's expiration month", examples=["12"])
card_exp_year: str = Field(..., description="The card's expiration year", examples=["2025"])
card_brand: str = Field(..., description="The card's brand", examples=["visa"])
card_holder_name: str = Field(..., description="The name of the card holder", examples=["John Doe"])
class NonEmptyCreditCardCredential(CreditCardCredential):
"""Credit card credential model that requires non-empty values."""
card_number: str = Field(
..., min_length=1, description="The full credit card number (must not be empty)", examples=["4111111111111111"]
)
card_cvv: str = Field(..., min_length=1, description="The card's CVV (must not be empty)", examples=["123"])
card_exp_month: str = Field(
..., min_length=1, description="The card's expiration month (must not be empty)", examples=["12"]
)
card_exp_year: str = Field(
..., min_length=1, description="The card's expiration year (must not be empty)", examples=["2025"]
)
card_brand: str = Field(..., min_length=1, description="The card's brand (must not be empty)", examples=["visa"])
card_holder_name: str = Field(
..., min_length=1, description="The name of the card holder (must not be empty)", examples=["John Doe"]
)
class SecretCredential(BaseModel):
"""Generic secret credential."""
secret_value: str = Field(..., min_length=1, description="The secret value", examples=["sk-abc123"])
secret_label: str | None = Field(default=None, description="Optional label describing the secret")
class CredentialItem(BaseModel):
"""Model representing a credential item in the system."""
item_id: str = Field(..., description="Unique identifier for the credential item", examples=["cred_1234567890"])
name: str = Field(..., description="Name of the credential", examples=["Skyvern Login"])
credential_type: CredentialType = Field(..., description="Type of the credential. Eg password, credit card, etc.")
credential: PasswordCredential | CreditCardCredential | SecretCredential = Field(
..., description="The actual credential data"
)
class CreateCredentialRequest(BaseModel):
"""Request model for creating a new credential."""
name: str = Field(..., description="Name of the credential", examples=["Amazon Login"])
credential_type: CredentialType = Field(..., description="Type of credential to create")
credential: NonEmptyPasswordCredential | NonEmptyCreditCardCredential | SecretCredential = Field(
...,
description="The credential data to store",
examples=[{"username": "user@example.com", "password": "securepassword123"}],
)
class CredentialResponse(BaseModel):
"""Response model for credential operations."""
credential_id: str = Field(..., description="Unique identifier for the credential", examples=["cred_1234567890"])
credential: PasswordCredentialResponse | CreditCardCredentialResponse | SecretCredentialResponse = Field(
..., description="The credential data"
)
credential_type: CredentialType = Field(..., description="Type of the credential")
name: str = Field(..., description="Name of the credential", examples=["Amazon Login"])
class Credential(BaseModel):
"""Database model for credentials."""
model_config = ConfigDict(from_attributes=True)
credential_id: str = Field(..., description="Unique identifier for the credential", examples=["cred_1234567890"])
organization_id: str = Field(
..., description="ID of the organization that owns the credential", examples=["o_1234567890"]
)
name: str = Field(..., description="Name of the credential", examples=["Skyvern Login"])
vault_type: CredentialVaultType | None = Field(..., description="Where the secret is stored: Bitwarden vs Azure")
item_id: str = Field(..., description="ID of the associated credential item", examples=["item_1234567890"])
credential_type: CredentialType = Field(..., description="Type of the credential. Eg password, credit card, etc.")
username: str | None = Field(..., description="For password credentials: the username")
totp_type: TotpType = Field(
TotpType.NONE,
description="Type of 2FA method used for this credential",
examples=[TotpType.AUTHENTICATOR],
)
totp_identifier: str | None = Field(
default=None,
description="Identifier (email or phone number) used to fetch TOTP codes",
examples=["user@example.com", "+14155550123"],
)
card_last4: str | None = Field(..., description="For credit_card credentials: the last four digits of the card")
card_brand: str | None = Field(..., description="For credit_card credentials: the card brand")
secret_label: str | None = Field(default=None, description="For secret credentials: optional label")
created_at: datetime = Field(..., description="Timestamp when the credential was created")
modified_at: datetime = Field(..., description="Timestamp when the credential was last modified")
deleted_at: datetime | None = Field(None, description="Timestamp when the credential was deleted, if applicable")