package container import ( "github.com/bcicen/ctop/connector/collector" "github.com/bcicen/ctop/connector/manager" "github.com/bcicen/ctop/cwidgets" "github.com/bcicen/ctop/cwidgets/compact" "github.com/bcicen/ctop/logging" "github.com/bcicen/ctop/models" ) var ( log = logging.Init() ) const ( running = "running" ) // Metrics and metadata representing a container type Container struct { models.Metrics Id string Meta map[string]string Widgets *compact.Compact Display bool // display this container in compact view updater cwidgets.WidgetUpdater collector collector.Collector manager manager.Manager } func New(id string, collector collector.Collector, manager manager.Manager) *Container { widgets := compact.NewCompact(id) return &Container{ Metrics: models.NewMetrics(), Id: id, Meta: make(map[string]string), Widgets: widgets, updater: widgets, collector: collector, manager: manager, } } func (c *Container) SetUpdater(u cwidgets.WidgetUpdater) { c.updater = u for k, v := range c.Meta { c.updater.SetMeta(k, v) } } func (c *Container) SetMeta(k, v string) { c.Meta[k] = v c.updater.SetMeta(k, v) } func (c *Container) GetMeta(k string) string { if v, ok := c.Meta[k]; ok { return v } return "" } func (c *Container) SetState(s string) { c.SetMeta("state", s) // start collector, if needed if s == running && !c.collector.Running() { c.collector.Start() c.Read(c.collector.Stream()) } // stop collector, if needed if s != running && c.collector.Running() { c.collector.Stop() } } // Logs returns container log collector func (c *Container) Logs() collector.LogCollector { return c.collector.Logs() } // Read metric stream, updating widgets func (c *Container) Read(stream chan models.Metrics) { go func() { for metrics := range stream { c.Metrics = metrics c.updater.SetMetrics(metrics) } log.Infof("reader stopped for container: %s", c.Id) c.Metrics = models.NewMetrics() c.Widgets.Reset() }() log.Infof("reader started for container: %s", c.Id) } func (c *Container) Start() { if c.Meta["state"] != running { if err := c.manager.Start(); err != nil { log.Warningf("container %s: %v", c.Id, err) log.StatusErr(err) return } c.SetState(running) } } func (c *Container) Stop() { if c.Meta["state"] == running { if err := c.manager.Stop(); err != nil { log.Warningf("container %s: %v", c.Id, err) log.StatusErr(err) return } c.SetState("exited") } } func (c *Container) Remove() { if err := c.manager.Remove(); err != nil { log.Warningf("container %s: %v", c.Id, err) log.StatusErr(err) } } func (c *Container) Pause() { if c.Meta["state"] == running { if err := c.manager.Pause(); err != nil { log.Warningf("container %s: %v", c.Id, err) log.StatusErr(err) return } c.SetState("paused") } } func (c *Container) Unpause() { if c.Meta["state"] == "paused" { if err := c.manager.Unpause(); err != nil { log.Warningf("container %s: %v", c.Id, err) log.StatusErr(err) return } c.SetState(running) } } func (c *Container) Restart() { if c.Meta["state"] == running { if err := c.manager.Restart(); err != nil { log.Warningf("container %s: %v", c.Id, err) log.StatusErr(err) return } } } func (c *Container) Exec(cmd []string) error { return c.manager.Exec(cmd) }