Files
mcp/internal/runtime/runtime.go
Kyle Isom 8b1c89fdc9 Add mcp build command and deploy auto-build
Extends MCP to own the full build-push-deploy lifecycle. When deploying,
the CLI checks whether each component's image tag exists in the registry
and builds/pushes automatically if missing and build config is present.

- Add Build, Push, ImageExists to runtime.Runtime interface (podman impl)
- Add mcp build <service>[/<image>] command
- Add [build] section to CLI config (workspace path)
- Add path and [build.images] to service definitions
- Wire auto-build into mcp deploy before agent RPC
- Update ARCHITECTURE.md with runtime interface and deploy auto-build docs

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-27 01:34:25 -07:00

72 lines
2.2 KiB
Go

package runtime
import (
"context"
"strings"
"time"
)
// ContainerSpec describes a container to create and run.
type ContainerSpec struct {
Name string // container name, format: <service>-<component>
Image string // full image reference
Network string // docker network name
User string // container user (e.g., "0:0")
Restart string // restart policy (e.g., "unless-stopped")
Ports []string // "host:container" port mappings
Volumes []string // "host:container" volume mounts
Cmd []string // command and arguments
Env []string // environment variables (KEY=VALUE)
}
// ContainerInfo describes the observed state of a running or stopped container.
type ContainerInfo struct {
Name string
Image string
State string // "running", "stopped", "exited", etc.
Network string
User string
Restart string
Ports []string
Volumes []string
Cmd []string
Version string // extracted from image tag
Started time.Time // when the container started (zero if not running)
}
// Runtime is the container runtime abstraction. The first six methods are
// used by the agent for container lifecycle. The last three are used by the
// CLI for building and pushing images.
type Runtime interface {
Pull(ctx context.Context, image string) error
Run(ctx context.Context, spec ContainerSpec) error
Stop(ctx context.Context, name string) error
Remove(ctx context.Context, name string) error
Inspect(ctx context.Context, name string) (ContainerInfo, error)
List(ctx context.Context) ([]ContainerInfo, error)
Build(ctx context.Context, image, contextDir, dockerfile string) error
Push(ctx context.Context, image string) error
ImageExists(ctx context.Context, image string) (bool, error)
}
// ExtractVersion parses the tag from an image reference.
// Examples:
//
// "registry/img:v1.2.0" -> "v1.2.0"
// "registry/img:latest" -> "latest"
// "registry/img" -> ""
// "registry:5000/img:v1" -> "v1"
func ExtractVersion(image string) string {
// Strip registry/path prefix so that a port like "registry:5000" isn't
// mistaken for a tag separator.
name := image
if i := strings.LastIndex(image, "/"); i >= 0 {
name = image[i+1:]
}
if i := strings.LastIndex(name, ":"); i >= 0 {
return name[i+1:]
}
return ""
}