2026-01-28 18:50:32 +00:00
|
|
|
package container
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"encoding/json"
|
|
|
|
|
"os"
|
|
|
|
|
"path/filepath"
|
|
|
|
|
"sync"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// State manages persistent container state.
|
|
|
|
|
type State struct {
|
|
|
|
|
// Containers is a map of container ID to Container.
|
|
|
|
|
Containers map[string]*Container `json:"containers"`
|
|
|
|
|
|
|
|
|
|
mu sync.RWMutex
|
|
|
|
|
filePath string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// DefaultStateDir returns the default directory for state files (~/.core).
|
|
|
|
|
func DefaultStateDir() (string, error) {
|
|
|
|
|
home, err := os.UserHomeDir()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
return filepath.Join(home, ".core"), nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// DefaultStatePath returns the default path for the state file.
|
|
|
|
|
func DefaultStatePath() (string, error) {
|
|
|
|
|
dir, err := DefaultStateDir()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
return filepath.Join(dir, "containers.json"), nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// DefaultLogsDir returns the default directory for container logs.
|
|
|
|
|
func DefaultLogsDir() (string, error) {
|
|
|
|
|
dir, err := DefaultStateDir()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
return filepath.Join(dir, "logs"), nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// NewState creates a new State instance.
|
|
|
|
|
func NewState(filePath string) *State {
|
|
|
|
|
return &State{
|
|
|
|
|
Containers: make(map[string]*Container),
|
|
|
|
|
filePath: filePath,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// LoadState loads the state from the given file path.
|
|
|
|
|
// If the file doesn't exist, returns an empty state.
|
|
|
|
|
func LoadState(filePath string) (*State, error) {
|
|
|
|
|
state := NewState(filePath)
|
|
|
|
|
|
|
|
|
|
data, err := os.ReadFile(filePath)
|
|
|
|
|
if err != nil {
|
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
|
return state, nil
|
|
|
|
|
}
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := json.Unmarshal(data, state); err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return state, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// SaveState persists the state to the configured file path.
|
|
|
|
|
func (s *State) SaveState() error {
|
|
|
|
|
s.mu.RLock()
|
|
|
|
|
defer s.mu.RUnlock()
|
|
|
|
|
|
|
|
|
|
// Ensure the directory exists
|
|
|
|
|
dir := filepath.Dir(s.filePath)
|
|
|
|
|
if err := os.MkdirAll(dir, 0755); err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data, err := json.MarshalIndent(s, "", " ")
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return os.WriteFile(s.filePath, data, 0644)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Add adds a container to the state and persists it.
|
|
|
|
|
func (s *State) Add(c *Container) error {
|
|
|
|
|
s.mu.Lock()
|
|
|
|
|
s.Containers[c.ID] = c
|
|
|
|
|
s.mu.Unlock()
|
|
|
|
|
|
|
|
|
|
return s.SaveState()
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-02 04:16:05 +00:00
|
|
|
// Get retrieves a copy of a container by ID.
|
|
|
|
|
// Returns a copy to prevent data races when the container is modified.
|
2026-01-28 18:50:32 +00:00
|
|
|
func (s *State) Get(id string) (*Container, bool) {
|
|
|
|
|
s.mu.RLock()
|
|
|
|
|
defer s.mu.RUnlock()
|
|
|
|
|
|
|
|
|
|
c, ok := s.Containers[id]
|
2026-02-02 04:16:05 +00:00
|
|
|
if !ok {
|
|
|
|
|
return nil, false
|
|
|
|
|
}
|
|
|
|
|
// Return a copy to prevent data races
|
|
|
|
|
copy := *c
|
|
|
|
|
return ©, true
|
2026-01-28 18:50:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Update updates a container in the state and persists it.
|
|
|
|
|
func (s *State) Update(c *Container) error {
|
|
|
|
|
s.mu.Lock()
|
|
|
|
|
s.Containers[c.ID] = c
|
|
|
|
|
s.mu.Unlock()
|
|
|
|
|
|
|
|
|
|
return s.SaveState()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Remove removes a container from the state and persists it.
|
|
|
|
|
func (s *State) Remove(id string) error {
|
|
|
|
|
s.mu.Lock()
|
|
|
|
|
delete(s.Containers, id)
|
|
|
|
|
s.mu.Unlock()
|
|
|
|
|
|
|
|
|
|
return s.SaveState()
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-02 04:16:05 +00:00
|
|
|
// All returns copies of all containers in the state.
|
|
|
|
|
// Returns copies to prevent data races when containers are modified.
|
2026-01-28 18:50:32 +00:00
|
|
|
func (s *State) All() []*Container {
|
|
|
|
|
s.mu.RLock()
|
|
|
|
|
defer s.mu.RUnlock()
|
|
|
|
|
|
|
|
|
|
containers := make([]*Container, 0, len(s.Containers))
|
|
|
|
|
for _, c := range s.Containers {
|
2026-02-02 04:16:05 +00:00
|
|
|
copy := *c
|
|
|
|
|
containers = append(containers, ©)
|
2026-01-28 18:50:32 +00:00
|
|
|
}
|
|
|
|
|
return containers
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// FilePath returns the path to the state file.
|
|
|
|
|
func (s *State) FilePath() string {
|
|
|
|
|
return s.filePath
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// LogPath returns the log file path for a given container ID.
|
|
|
|
|
func LogPath(id string) (string, error) {
|
|
|
|
|
logsDir, err := DefaultLogsDir()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
return filepath.Join(logsDir, id+".log"), nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// EnsureLogsDir ensures the logs directory exists.
|
|
|
|
|
func EnsureLogsDir() error {
|
|
|
|
|
logsDir, err := DefaultLogsDir()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
return os.MkdirAll(logsDir, 0755)
|
|
|
|
|
}
|