Replace direct os package calls with io.Local methods: - os.MkdirAll → io.Local.EnsureDir - os.ReadFile → io.Local.Read - os.WriteFile → io.Local.Write - os.Remove → io.Local.Delete - os.RemoveAll → io.Local.DeleteAll This is part of the io.Medium abstraction migration (#101). Closes #104 Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
163 lines
3.4 KiB
Go
163 lines
3.4 KiB
Go
// Package cache provides a file-based cache for GitHub API responses.
|
|
package cache
|
|
|
|
import (
|
|
"encoding/json"
|
|
"os"
|
|
"path/filepath"
|
|
"time"
|
|
|
|
"github.com/host-uk/core/pkg/io"
|
|
)
|
|
|
|
// DefaultTTL is the default cache expiry time.
|
|
const DefaultTTL = 1 * time.Hour
|
|
|
|
// Cache represents a file-based cache.
|
|
type Cache struct {
|
|
baseDir string
|
|
ttl time.Duration
|
|
}
|
|
|
|
// Entry represents a cached item with metadata.
|
|
type Entry struct {
|
|
Data json.RawMessage `json:"data"`
|
|
CachedAt time.Time `json:"cached_at"`
|
|
ExpiresAt time.Time `json:"expires_at"`
|
|
}
|
|
|
|
// New creates a new cache instance.
|
|
// If baseDir is empty, uses .core/cache in current directory
|
|
func New(baseDir string, ttl time.Duration) (*Cache, error) {
|
|
if baseDir == "" {
|
|
// Use .core/cache in current working directory
|
|
cwd, err := os.Getwd()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
baseDir = filepath.Join(cwd, ".core", "cache")
|
|
}
|
|
|
|
if ttl == 0 {
|
|
ttl = DefaultTTL
|
|
}
|
|
|
|
// Ensure cache directory exists
|
|
if err := io.Local.EnsureDir(baseDir); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &Cache{
|
|
baseDir: baseDir,
|
|
ttl: ttl,
|
|
}, nil
|
|
}
|
|
|
|
// Path returns the full path for a cache key.
|
|
func (c *Cache) Path(key string) string {
|
|
return filepath.Join(c.baseDir, key+".json")
|
|
}
|
|
|
|
// Get retrieves a cached item if it exists and hasn't expired.
|
|
func (c *Cache) Get(key string, dest interface{}) (bool, error) {
|
|
path := c.Path(key)
|
|
|
|
dataStr, err := io.Local.Read(path)
|
|
if err != nil {
|
|
if os.IsNotExist(err) {
|
|
return false, nil
|
|
}
|
|
return false, err
|
|
}
|
|
|
|
var entry Entry
|
|
if err := json.Unmarshal([]byte(dataStr), &entry); err != nil {
|
|
// Invalid cache file, treat as miss
|
|
return false, nil
|
|
}
|
|
|
|
// Check expiry
|
|
if time.Now().After(entry.ExpiresAt) {
|
|
return false, nil
|
|
}
|
|
|
|
// Unmarshal the actual data
|
|
if err := json.Unmarshal(entry.Data, dest); err != nil {
|
|
return false, err
|
|
}
|
|
|
|
return true, nil
|
|
}
|
|
|
|
// Set stores an item in the cache.
|
|
func (c *Cache) Set(key string, data interface{}) error {
|
|
path := c.Path(key)
|
|
|
|
// Ensure parent directory exists
|
|
if err := io.Local.EnsureDir(filepath.Dir(path)); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Marshal the data
|
|
dataBytes, err := json.Marshal(data)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
entry := Entry{
|
|
Data: dataBytes,
|
|
CachedAt: time.Now(),
|
|
ExpiresAt: time.Now().Add(c.ttl),
|
|
}
|
|
|
|
entryBytes, err := json.MarshalIndent(entry, "", " ")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return io.Local.Write(path, string(entryBytes))
|
|
}
|
|
|
|
// Delete removes an item from the cache.
|
|
func (c *Cache) Delete(key string) error {
|
|
path := c.Path(key)
|
|
err := io.Local.Delete(path)
|
|
if os.IsNotExist(err) {
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
|
|
// Clear removes all cached items.
|
|
func (c *Cache) Clear() error {
|
|
return io.Local.DeleteAll(c.baseDir)
|
|
}
|
|
|
|
// Age returns how old a cached item is, or -1 if not cached.
|
|
func (c *Cache) Age(key string) time.Duration {
|
|
path := c.Path(key)
|
|
|
|
dataStr, err := io.Local.Read(path)
|
|
if err != nil {
|
|
return -1
|
|
}
|
|
|
|
var entry Entry
|
|
if err := json.Unmarshal([]byte(dataStr), &entry); err != nil {
|
|
return -1
|
|
}
|
|
|
|
return time.Since(entry.CachedAt)
|
|
}
|
|
|
|
// GitHub-specific cache keys
|
|
|
|
// GitHubReposKey returns the cache key for an org's repo list.
|
|
func GitHubReposKey(org string) string {
|
|
return filepath.Join("github", org, "repos")
|
|
}
|
|
|
|
// GitHubRepoKey returns the cache key for a specific repo's metadata.
|
|
func GitHubRepoKey(org, repo string) string {
|
|
return filepath.Join("github", org, repo, "meta")
|
|
}
|