package client

import (
	"context"
	"errors"
	"fmt"
	"io"
	"net/http"
	"strconv"
	"sync/atomic"
	"time"
)

// ErrNotFound is returned when a resource is not found (404)
// This often indicates access blocked by GitHub (remapped from 403 by proxy)
var ErrNotFound = errors.New("resource not found")

// ErrUnavailableForLegalReasons is returned when a resource is blocked due to legal reasons (451)
// This typically indicates DMCA takedowns or other legal blocks
var ErrUnavailableForLegalReasons = errors.New("unavailable for legal reasons")

// ErrUnprocessableEntity is returned when the server cannot process the request (422)
// This typically indicates the repository is missing relevant data to generate diffs
var ErrUnprocessableEntity = errors.New("unprocessable entity")

// ErrGone is returned when a resource is permanently gone (410)
// This typically indicates features disabled (e.g., issues disabled for a repo)
var ErrGone = errors.New("resource gone")

// ErrPayloadTooLarge is returned when the response payload is too large (413)
// This typically indicates the recursive tree is too large and needs non-recursive fallback
var ErrPayloadTooLarge = errors.New("payload too large")

// GithubClient handles all GitHub API interactions through a proxy
type GithubClient struct {
	httpClient        *http.Client
	baseURL           string
	rateLimitLogCount atomic.Uint64
	requestCount      atomic.Uint64
}

// NewGithubClient creates a new GitHub client configured for high concurrency
func NewGithubClient(baseURL string, maxIdleConns int) *GithubClient {
	transport := &http.Transport{
		MaxIdleConnsPerHost: maxIdleConns,
		MaxIdleConns:        maxIdleConns,
		IdleConnTimeout:     90 * time.Second,
	}

	return &GithubClient{
		httpClient: &http.Client{
			Transport: transport,
			Timeout:   30 * time.Second,
		},
		baseURL: baseURL,
	}
}

// Get performs an HTTP GET request with automatic retry logic
// Returns raw response bytes on success (200), error otherwise
func (c *GithubClient) Get(ctx context.Context, endpoint string) ([]byte, error) {
	url := c.baseURL + endpoint

	// Increment request counter
	c.requestCount.Add(1)

	for {
		select {
		case <-ctx.Done():
			return nil, ctx.Err()
		default:
		}

		req, err := http.NewRequestWithContext(ctx, "GET", url, nil)
		if err != nil {
			return nil, fmt.Errorf("failed to create request: %w", err)
		}

		resp, err := c.httpClient.Do(req)
		if err != nil {
			// Network error - retry with exponential backoff
			time.Sleep(1 * time.Second)
			continue
		}

		// Read body
		body, err := io.ReadAll(resp.Body)
		resp.Body.Close()
		if err != nil {
			return nil, fmt.Errorf("failed to read response body: %w", err)
		}

		switch resp.StatusCode {
		case http.StatusOK:
			return body, nil

		case http.StatusTooManyRequests:
			// Rate limit hit - parse Retry-After and sleep
			retryAfter := c.parseRetryAfter(resp.Header)

			// Log warning using atomic counter to prevent flooding
			if c.rateLimitLogCount.Add(1)%1000 == 1 {
				fmt.Printf("[WARN] Rate limit hit. Sleeping for %v seconds (logged every 1000 occurrences)\n", retryAfter)
			}

			time.Sleep(time.Duration(retryAfter) * time.Second)
			continue

		case http.StatusNotFound:
			// 404 often indicates access blocked (remapped from 403 by proxy)
			// Return typed error for caller to handle without logging
			return nil, fmt.Errorf("%w: %s (body: %s)", ErrNotFound, endpoint, c.truncateBody(body, 200))

		case http.StatusGone:
			// 410 indicates resource is permanently gone (e.g., issues disabled)
			// Return typed error for caller to handle without logging
			return nil, fmt.Errorf("%w: %s (body: %s)", ErrGone, endpoint, c.truncateBody(body, 200))

		case http.StatusRequestEntityTooLarge:
			// 413 indicates payload too large (e.g., recursive tree too large)
			// Return typed error for caller to handle without logging
			return nil, fmt.Errorf("%w: %s", ErrPayloadTooLarge, endpoint)

		case http.StatusUnprocessableEntity:
			// 422 indicates the repository is missing relevant data to generate diffs
			// Return typed error for caller to handle without logging or retrying
			return nil, fmt.Errorf("%w: %s", ErrUnprocessableEntity, endpoint)

		case 451: // StatusUnavailableForLegalReasons
			// 451 indicates DMCA takedown or other legal blocks
			// Return typed error for caller to handle without logging
			return nil, fmt.Errorf("%w: %s (body: %s)", ErrUnavailableForLegalReasons, endpoint, c.truncateBody(body, 200))

		case http.StatusInternalServerError, http.StatusBadGateway, http.StatusServiceUnavailable:
			// 5xx errors - exponential backoff
			backoff := time.Duration(1+resp.StatusCode%10) * time.Second
			fmt.Printf("[WARN] Server error %d for %s, retrying after %v\n", resp.StatusCode, endpoint, backoff)
			time.Sleep(backoff)
			continue

		default:
			// Include response body in error message for debugging
			return nil, fmt.Errorf("unexpected status code %d for %s (body: %s)", resp.StatusCode, endpoint, c.truncateBody(body, 200))
		}
	}
}

// truncateBody truncates the body to maxLen bytes for error messages
func (c *GithubClient) truncateBody(body []byte, maxLen int) string {
	if len(body) == 0 {
		return "<empty>"
	}
	if len(body) <= maxLen {
		return string(body)
	}
	return string(body[:maxLen]) + "..."
}

// parseRetryAfter extracts the retry duration from Retry-After header
func (c *GithubClient) parseRetryAfter(headers http.Header) int {
	retryAfter := headers.Get("Retry-After")
	if retryAfter == "" {
		return 60 // Default to 60 seconds
	}

	// Try parsing as integer (seconds)
	if seconds, err := strconv.Atoi(retryAfter); err == nil {
		return seconds
	}

	// Try parsing as HTTP date
	if t, err := time.Parse(time.RFC1123, retryAfter); err == nil {
		duration := time.Until(t)
		if duration > 0 {
			return int(duration.Seconds())
		}
	}

	return 60 // Default fallback
}

// GetRequestCount returns the total number of API requests made
func (c *GithubClient) GetRequestCount() uint64 {
	return c.requestCount.Load()
}
