* feat(devops): migrate filesystem operations to io.Local abstraction Migrate config.go: - os.ReadFile → io.Local.Read Migrate devops.go: - os.Stat → io.Local.IsFile Migrate images.go: - os.MkdirAll → io.Local.EnsureDir - os.Stat → io.Local.IsFile - os.ReadFile → io.Local.Read - os.WriteFile → io.Local.Write Migrate test.go: - os.ReadFile → io.Local.Read - os.Stat → io.Local.IsFile Migrate claude.go: - os.Stat → io.Local.IsDir Updated tests to reflect improved behavior: - Manifest.Save() now creates parent directories - hasFile() correctly returns false for directories Part of #101 (io.Medium migration tracking issue). Closes #107 Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * chore(io): migrate remaining packages to io.Local abstraction Migrate filesystem operations to use the io.Local abstraction for improved security, testability, and consistency: - pkg/cache: Replace os.ReadFile, WriteFile, Remove, RemoveAll with io.Local equivalents. io.Local.Write creates parent dirs automatically. - pkg/agentic: Migrate config.go and context.go to use io.Local for reading config files and gathering file context. - pkg/repos: Use io.Local.Read, Exists, IsDir, List for registry operations and git repo detection. - pkg/release: Use io.Local for config loading, existence checks, and artifact discovery. - pkg/devops/sources: Use io.Local.EnsureDir for CDN download. All paths are converted to absolute using filepath.Abs() before calling io.Local methods to handle relative paths correctly. Closes #104, closes #106, closes #108, closes #111 Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * chore(io): migrate pkg/cli and pkg/container to io.Local abstraction Continue io.Medium migration for the remaining packages: - pkg/cli/daemon.go: PIDFile Acquire/Release now use io.Local.Read, Delete, and Write for managing daemon PID files. - pkg/container/state.go: LoadState and SaveState use io.Local for JSON state persistence. EnsureLogsDir uses io.Local.EnsureDir. - pkg/container/templates.go: Template loading and directory scanning now use io.Local.IsFile, IsDir, Read, and List. - pkg/container/linuxkit.go: Image validation uses io.Local.IsFile, log file check uses io.Local.IsFile. Streaming log file creation (os.Create) remains unchanged as io.Local doesn't support streaming. Closes #105, closes #107 Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * docs(audit): add dependency security audit report Complete security audit of all project dependencies: - Run govulncheck: No vulnerabilities found - Run go mod verify: All modules verified - Document 15 direct dependencies and 161 indirect - Assess supply chain risks: Low risk overall - Verify lock files are committed with integrity hashes - Provide CI integration recommendations Closes #185 Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix(ci): build core CLI from source instead of downloading release The workflows were trying to download from a non-existent release URL. Now builds the CLI directly using `go build` with version injection. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * chore: trigger CI with updated workflow * chore(ci): add workflow_dispatch trigger for manual runs --------- Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com>
169 lines
3.5 KiB
Go
169 lines
3.5 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
|
|
}
|
|
|
|
// Convert to absolute path for io.Local
|
|
absBaseDir, err := filepath.Abs(baseDir)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Ensure cache directory exists
|
|
if err := io.Local.EnsureDir(absBaseDir); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
baseDir = absBaseDir
|
|
|
|
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)
|
|
|
|
content, err := io.Local.Read(path)
|
|
if err != nil {
|
|
if os.IsNotExist(err) {
|
|
return false, nil
|
|
}
|
|
return false, err
|
|
}
|
|
data := []byte(content)
|
|
|
|
var entry Entry
|
|
if err := json.Unmarshal(data, &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)
|
|
|
|
// 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
|
|
}
|
|
|
|
// io.Local.Write creates parent directories automatically
|
|
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)
|
|
|
|
content, err := io.Local.Read(path)
|
|
if err != nil {
|
|
return -1
|
|
}
|
|
data := []byte(content)
|
|
|
|
var entry Entry
|
|
if err := json.Unmarshal(data, &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")
|
|
}
|