package main

import (
	"container/list"
	"sync"
	"time"
)

// lruCache is a small, in-memory LRU cache intended for caching GitHub API responses
// for reruns. It is process-local and best-effort.
//
// Concurrency: safe for concurrent access.
// Eviction: LRU by access.
// TTL: optional per-entry expiry.
//
// This is intentionally minimal to avoid introducing new third-party dependencies.
type lruCache struct {
	mu       sync.Mutex
	maxItems int
	ll       *list.List
	items    map[string]*list.Element
}

type lruEntry struct {
	key       string
	value     cachedResponse
	expiresAt time.Time
}

type cachedResponse struct {
	statusCode int
	body       []byte
	// retryAfterSeconds is applied by the proxy as Retry-After when status code is 403/429.
	retryAfterSeconds int
	// contentType is returned as Content-Type when present.
	contentType string
}

func newLRUCache(maxItems int) *lruCache {
	if maxItems <= 0 {
		maxItems = 1
	}
	return &lruCache{
		maxItems: maxItems,
		ll:       list.New(),
		items:    make(map[string]*list.Element, maxItems),
	}
}

func (c *lruCache) Get(key string) (cachedResponse, bool) {
	c.mu.Lock()
	defer c.mu.Unlock()

	el, ok := c.items[key]
	if !ok {
		return cachedResponse{}, false
	}
	ent := el.Value.(*lruEntry)
	if !ent.expiresAt.IsZero() && time.Now().After(ent.expiresAt) {
		c.removeElementLocked(el)
		return cachedResponse{}, false
	}

	c.ll.MoveToFront(el)
	return ent.value, true
}

func (c *lruCache) Set(key string, value cachedResponse, ttl time.Duration) {
	c.mu.Lock()
	defer c.mu.Unlock()

	if el, ok := c.items[key]; ok {
		ent := el.Value.(*lruEntry)
		ent.value = value
		if ttl > 0 {
			ent.expiresAt = time.Now().Add(ttl)
		} else {
			ent.expiresAt = time.Time{}
		}
		c.ll.MoveToFront(el)
		return
	}

	ent := &lruEntry{key: key, value: value}
	if ttl > 0 {
		ent.expiresAt = time.Now().Add(ttl)
	}
	el := c.ll.PushFront(ent)
	c.items[key] = el

	if len(c.items) > c.maxItems {
		c.removeOldestLocked()
	}
}

func (c *lruCache) removeOldestLocked() {
	el := c.ll.Back()
	if el == nil {
		return
	}
	c.removeElementLocked(el)
}

func (c *lruCache) removeElementLocked(el *list.Element) {
	c.ll.Remove(el)
	ent := el.Value.(*lruEntry)
	delete(c.items, ent.key)
}
