mirror of
https://github.com/mark3labs/kit.git
synced 2026-06-14 03:30:26 +00:00
277 lines
9.3 KiB
Go
277 lines
9.3 KiB
Go
package auth
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"time"
|
|
)
|
|
|
|
// CredentialStore holds all stored credentials for various providers.
|
|
// Currently supports Anthropic credentials with both OAuth and API key authentication methods.
|
|
type CredentialStore struct {
|
|
Anthropic *AnthropicCredentials `json:"anthropic,omitempty"`
|
|
}
|
|
|
|
// AnthropicCredentials holds Anthropic API credentials supporting both OAuth
|
|
// and API key authentication methods. The Type field indicates which authentication
|
|
// method is being used. For OAuth, tokens are stored with expiration timestamps
|
|
// for automatic refresh. For API keys, only the key itself is stored.
|
|
type AnthropicCredentials struct {
|
|
Type string `json:"type"` // "oauth" or "api_key"
|
|
APIKey string `json:"api_key,omitempty"` // For API key auth
|
|
AccessToken string `json:"access_token,omitempty"` // For OAuth
|
|
RefreshToken string `json:"refresh_token,omitempty"` // For OAuth
|
|
ExpiresAt int64 `json:"expires_at,omitempty"` // For OAuth
|
|
CreatedAt time.Time `json:"created_at"`
|
|
}
|
|
|
|
// IsExpired checks if the OAuth token is expired based on the ExpiresAt timestamp.
|
|
// Returns false for API key authentication or if no expiration is set.
|
|
func (c *AnthropicCredentials) IsExpired() bool {
|
|
if c.Type != "oauth" || c.ExpiresAt == 0 {
|
|
return false
|
|
}
|
|
return time.Now().Unix() >= c.ExpiresAt
|
|
}
|
|
|
|
// NeedsRefresh checks if the OAuth token needs refresh, returning true if the token
|
|
// will expire within the next 5 minutes. This allows for proactive token refresh
|
|
// to avoid authentication failures during operations. Returns false for API key
|
|
// authentication or if no expiration is set.
|
|
func (c *AnthropicCredentials) NeedsRefresh() bool {
|
|
if c.Type != "oauth" || c.ExpiresAt == 0 {
|
|
return false
|
|
}
|
|
return time.Now().Unix() >= (c.ExpiresAt - 300) // 5 minutes buffer
|
|
}
|
|
|
|
// CredentialManager handles secure storage and retrieval of authentication credentials.
|
|
// It manages a JSON file stored in the user's config directory with appropriate
|
|
// file permissions for security.
|
|
type CredentialManager struct {
|
|
credentialsPath string
|
|
}
|
|
|
|
// NewCredentialManager creates a new credential manager instance. It determines
|
|
// the appropriate credentials path based on XDG_CONFIG_HOME or falls back to
|
|
// ~/.config/.mcphost/credentials.json. Returns an error if the home directory
|
|
// cannot be determined.
|
|
func NewCredentialManager() (*CredentialManager, error) {
|
|
credentialsPath, err := getCredentialsPath()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to determine credentials path: %w", err)
|
|
}
|
|
|
|
return &CredentialManager{
|
|
credentialsPath: credentialsPath,
|
|
}, nil
|
|
}
|
|
|
|
// getCredentialsPath returns the path to the credentials file
|
|
func getCredentialsPath() (string, error) {
|
|
// Try XDG_CONFIG_HOME first
|
|
if xdgConfig := os.Getenv("XDG_CONFIG_HOME"); xdgConfig != "" {
|
|
return filepath.Join(xdgConfig, ".mcphost", "credentials.json"), nil
|
|
}
|
|
|
|
// Fall back to ~/.config/.mcphost/credentials.json
|
|
homeDir, err := os.UserHomeDir()
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to get home directory: %w", err)
|
|
}
|
|
|
|
return filepath.Join(homeDir, ".config", ".mcphost", "credentials.json"), nil
|
|
}
|
|
|
|
// LoadCredentials loads credentials from the JSON file. If the file doesn't exist,
|
|
// it returns an empty CredentialStore instead of an error, allowing for graceful
|
|
// initialization. Returns an error if the file exists but cannot be read or parsed.
|
|
func (cm *CredentialManager) LoadCredentials() (*CredentialStore, error) {
|
|
// If file doesn't exist, return empty store
|
|
if _, err := os.Stat(cm.credentialsPath); os.IsNotExist(err) {
|
|
return &CredentialStore{}, nil
|
|
}
|
|
|
|
data, err := os.ReadFile(cm.credentialsPath)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to read credentials file: %w", err)
|
|
}
|
|
|
|
var store CredentialStore
|
|
if err := json.Unmarshal(data, &store); err != nil {
|
|
return nil, fmt.Errorf("failed to parse credentials file: %w", err)
|
|
}
|
|
|
|
return &store, nil
|
|
}
|
|
|
|
// SaveCredentials saves credentials to the JSON file with secure permissions (0600).
|
|
// It creates the parent directory if it doesn't exist. The file is written atomically
|
|
// to prevent corruption. Returns an error if the directory cannot be created or the
|
|
// file cannot be written.
|
|
func (cm *CredentialManager) SaveCredentials(store *CredentialStore) error {
|
|
// Ensure directory exists
|
|
dir := filepath.Dir(cm.credentialsPath)
|
|
if err := os.MkdirAll(dir, 0700); err != nil {
|
|
return fmt.Errorf("failed to create credentials directory: %w", err)
|
|
}
|
|
|
|
data, err := json.MarshalIndent(store, "", " ")
|
|
if err != nil {
|
|
return fmt.Errorf("failed to marshal credentials: %w", err)
|
|
}
|
|
|
|
// Write with restrictive permissions (read/write for owner only)
|
|
if err := os.WriteFile(cm.credentialsPath, data, 0600); err != nil {
|
|
return fmt.Errorf("failed to write credentials file: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// SetAnthropicCredentials stores Anthropic API key credentials. It validates the
|
|
// API key format before storing. The API key must start with "sk-ant-" and be
|
|
// at least 20 characters long. Returns an error if the API key is invalid or
|
|
// if storage fails.
|
|
func (cm *CredentialManager) SetAnthropicCredentials(apiKey string) error {
|
|
if err := validateAnthropicAPIKey(apiKey); err != nil {
|
|
return err
|
|
}
|
|
|
|
store, err := cm.LoadCredentials()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
store.Anthropic = &AnthropicCredentials{
|
|
Type: "api_key",
|
|
APIKey: apiKey,
|
|
CreatedAt: time.Now(),
|
|
}
|
|
|
|
return cm.SaveCredentials(store)
|
|
}
|
|
|
|
// GetAnthropicCredentials retrieves stored Anthropic credentials. Returns nil if
|
|
// no credentials are stored. The returned credentials may be either OAuth or API
|
|
// key type, check the Type field to determine which.
|
|
func (cm *CredentialManager) GetAnthropicCredentials() (*AnthropicCredentials, error) {
|
|
store, err := cm.LoadCredentials()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return store.Anthropic, nil
|
|
}
|
|
|
|
// RemoveAnthropicCredentials removes stored Anthropic credentials from storage.
|
|
// If this was the only credential stored, the entire credentials file is removed.
|
|
// Returns an error if the removal fails.
|
|
func (cm *CredentialManager) RemoveAnthropicCredentials() error {
|
|
store, err := cm.LoadCredentials()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
store.Anthropic = nil
|
|
|
|
// If store is empty, remove the file entirely
|
|
if store.Anthropic == nil {
|
|
if err := os.Remove(cm.credentialsPath); err != nil && !os.IsNotExist(err) {
|
|
return fmt.Errorf("failed to remove credentials file: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
return cm.SaveCredentials(store)
|
|
}
|
|
|
|
// HasAnthropicCredentials checks if valid Anthropic credentials are stored.
|
|
// Returns true if either a non-empty OAuth access token or API key is present,
|
|
// false otherwise. Returns an error if credentials cannot be loaded.
|
|
func (cm *CredentialManager) HasAnthropicCredentials() (bool, error) {
|
|
creds, err := cm.GetAnthropicCredentials()
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
if creds == nil {
|
|
return false, nil
|
|
}
|
|
|
|
// Check based on credential type
|
|
switch creds.Type {
|
|
case "oauth":
|
|
return creds.AccessToken != "", nil
|
|
case "api_key":
|
|
return creds.APIKey != "", nil
|
|
default:
|
|
return false, nil
|
|
}
|
|
}
|
|
|
|
// GetCredentialsPath returns the absolute path to the credentials JSON file.
|
|
// This is useful for debugging or displaying the storage location to users.
|
|
func (cm *CredentialManager) GetCredentialsPath() string {
|
|
return cm.credentialsPath
|
|
}
|
|
|
|
// validateAnthropicAPIKey validates the format of an Anthropic API key
|
|
func validateAnthropicAPIKey(apiKey string) error {
|
|
apiKey = strings.TrimSpace(apiKey)
|
|
|
|
if apiKey == "" {
|
|
return fmt.Errorf("API key cannot be empty")
|
|
}
|
|
|
|
// Anthropic API keys typically start with "sk-ant-" and are quite long
|
|
if !strings.HasPrefix(apiKey, "sk-ant-") {
|
|
return fmt.Errorf("invalid Anthropic API key format (should start with 'sk-ant-')")
|
|
}
|
|
|
|
if len(apiKey) < 20 {
|
|
return fmt.Errorf("API key appears to be too short")
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// GetAnthropicAPIKey retrieves an Anthropic API key from multiple sources in priority order:
|
|
// 1. Command-line flag value (highest priority)
|
|
// 2. Stored credentials (OAuth or API key)
|
|
// 3. ANTHROPIC_API_KEY environment variable (lowest priority)
|
|
// Returns the API key, a description of its source, and any error encountered.
|
|
// For OAuth credentials, it automatically refreshes expired tokens.
|
|
func GetAnthropicAPIKey(flagValue string) (string, string, error) {
|
|
// 1. Check flag value first (highest priority)
|
|
if flagValue != "" {
|
|
return flagValue, "command-line flag", nil
|
|
}
|
|
|
|
// 2. Check stored credentials
|
|
cm, err := NewCredentialManager()
|
|
if err == nil {
|
|
if creds, err := cm.GetAnthropicCredentials(); err == nil && creds != nil {
|
|
if creds.Type == "oauth" && creds.AccessToken != "" {
|
|
// For OAuth, get a valid access token (may refresh if needed)
|
|
token, err := cm.GetValidAccessToken()
|
|
if err != nil {
|
|
return "", "", fmt.Errorf("failed to get valid OAuth token: %w", err)
|
|
}
|
|
return token, "stored OAuth credentials", nil
|
|
} else if creds.Type == "api_key" && creds.APIKey != "" {
|
|
return creds.APIKey, "stored API key", nil
|
|
}
|
|
}
|
|
}
|
|
|
|
// 3. Fall back to environment variable
|
|
if envKey := os.Getenv("ANTHROPIC_API_KEY"); envKey != "" {
|
|
return envKey, "ANTHROPIC_API_KEY environment variable", nil
|
|
}
|
|
|
|
return "", "", fmt.Errorf("no Anthropic API key found. Use 'mcphost auth login anthropic', set ANTHROPIC_API_KEY environment variable, or use --provider-api-key flag")
|
|
}
|