added container menu closes #28

This commit is contained in:
Peter Reisinger 2017-11-20 11:09:36 +00:00
parent e1ec264345
commit 436266b1a4
8 changed files with 163 additions and 3 deletions

View File

@ -8,6 +8,7 @@ import (
"github.com/bcicen/ctop/connector/collector" "github.com/bcicen/ctop/connector/collector"
"github.com/bcicen/ctop/container" "github.com/bcicen/ctop/container"
api "github.com/fsouza/go-dockerclient" api "github.com/fsouza/go-dockerclient"
"github.com/bcicen/ctop/connector/manager"
) )
type Docker struct { type Docker struct {
@ -132,8 +133,10 @@ func (cm *Docker) MustGet(id string) *container.Container {
if !ok { if !ok {
// create collector // create collector
collector := collector.NewDocker(cm.client, id) collector := collector.NewDocker(cm.client, id)
// create manager
manager := manager.NewDocker(cm.client, id)
// create container // create container
c = container.New(id, collector) c = container.New(id, collector, manager)
cm.lock.Lock() cm.lock.Lock()
cm.containers[id] = c cm.containers[id] = c
cm.lock.Unlock() cm.lock.Unlock()

View File

@ -0,0 +1,44 @@
package manager
import (
api "github.com/fsouza/go-dockerclient"
"fmt"
)
type Docker struct {
id string
client *api.Client
}
func NewDocker(client *api.Client, id string) *Docker {
return &Docker{
id: id,
client: client,
}
}
func (dc *Docker) Start() error {
c, err := dc.client.InspectContainer(dc.id)
if err != nil {
return fmt.Errorf("cannot inspect container: %v", err)
}
if err := dc.client.StartContainer(c.ID, c.HostConfig); err != nil {
return fmt.Errorf("cannot start container: %v", err)
}
return nil
}
func (dc *Docker) Stop() error {
if err := dc.client.StopContainer(dc.id, 3); err != nil {
return fmt.Errorf("cannot stop container: %v", err)
}
return nil
}
func (dc *Docker) Remove() error {
if err := dc.client.RemoveContainer(api.RemoveContainerOptions{ID: dc.id}); err != nil {
return fmt.Errorf("cannot remove container: %v", err)
}
return nil
}

View File

@ -0,0 +1,7 @@
package manager
type Manager interface {
Start() error
Stop() error
Remove() error
}

19
connector/manager/mock.go Normal file
View File

@ -0,0 +1,19 @@
package manager
type Mock struct {}
func NewMock() *Mock {
return &Mock{}
}
func (m *Mock) Start() error {
return nil
}
func (m *Mock) Stop() error {
return nil
}
func (m *Mock) Remove() error {
return nil
}

View File

@ -11,6 +11,7 @@ import (
"github.com/bcicen/ctop/container" "github.com/bcicen/ctop/container"
"github.com/jgautheron/codename-generator" "github.com/jgautheron/codename-generator"
"github.com/nu7hatch/gouuid" "github.com/nu7hatch/gouuid"
"github.com/bcicen/ctop/connector/manager"
) )
type Mock struct { type Mock struct {
@ -40,7 +41,8 @@ func (cs *Mock) Init() {
func (cs *Mock) makeContainer(aggression int64) { func (cs *Mock) makeContainer(aggression int64) {
collector := collector.NewMock(aggression) collector := collector.NewMock(aggression)
c := container.New(makeID(), collector) manager := manager.NewMock()
c := container.New(makeID(), collector, manager)
c.SetMeta("name", makeName()) c.SetMeta("name", makeName())
c.SetState(makeState()) c.SetState(makeState())
cs.containers = append(cs.containers, c) cs.containers = append(cs.containers, c)

View File

@ -6,6 +6,7 @@ import (
"github.com/bcicen/ctop/cwidgets/compact" "github.com/bcicen/ctop/cwidgets/compact"
"github.com/bcicen/ctop/logging" "github.com/bcicen/ctop/logging"
"github.com/bcicen/ctop/models" "github.com/bcicen/ctop/models"
"github.com/bcicen/ctop/connector/manager"
) )
var ( var (
@ -21,9 +22,10 @@ type Container struct {
Display bool // display this container in compact view Display bool // display this container in compact view
updater cwidgets.WidgetUpdater updater cwidgets.WidgetUpdater
collector collector.Collector collector collector.Collector
manager manager.Manager
} }
func New(id string, collector collector.Collector) *Container { func New(id string, collector collector.Collector, manager manager.Manager) *Container {
widgets := compact.NewCompact(id) widgets := compact.NewCompact(id)
return &Container{ return &Container{
Metrics: models.NewMetrics(), Metrics: models.NewMetrics(),
@ -32,6 +34,7 @@ func New(id string, collector collector.Collector) *Container {
Widgets: widgets, Widgets: widgets,
updater: widgets, updater: widgets,
collector: collector, collector: collector,
manager: manager,
} }
} }
@ -85,3 +88,31 @@ func (c *Container) Read(stream chan models.Metrics) {
}() }()
log.Infof("reader started for container: %s", c.Id) 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)
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)
return
}
c.SetState("exited")
}
}
func (c *Container) Remove() {
if c.Meta["state"] == "exited" {
if err := c.manager.Remove(); err != nil {
log.Warningf("container %s: %v", c.Id, err)
}
}
}

View File

@ -93,6 +93,10 @@ func Display() bool {
ui.StopLoop() ui.StopLoop()
}) })
ui.Handle("/sys/kbd/m", func(ui.Event) {
menu = ContainerMenu
ui.StopLoop()
})
ui.Handle("/sys/kbd/<enter>", func(ui.Event) { ui.Handle("/sys/kbd/<enter>", func(ui.Event) {
single = true single = true
ui.StopLoop() ui.StopLoop()

View File

@ -94,3 +94,53 @@ func SortMenu() {
ui.Render(m) ui.Render(m)
ui.Loop() ui.Loop()
} }
func ContainerMenu() {
c := cursor.Selected()
if c == nil {
return
}
ui.DefaultEvtStream.ResetHandlers()
defer ui.DefaultEvtStream.ResetHandlers()
m := menu.NewMenu()
m.Selectable = true
m.BorderLabel = "Menu"
var items []menu.Item
if c.Meta["state"] == "running" {
items = append(items, menu.Item{Val: "stop", Label: "stop"})
}
if c.Meta["state"] == "exited" {
items = append(items, menu.Item{Val: "start", Label: "start"})
items = append(items, menu.Item{Val: "remove", Label: "remove"})
}
items = append(items, menu.Item{Val: "cancel", Label: "cancel"})
m.AddItems(items...)
ui.Render(m)
HandleKeys("up", m.Up)
HandleKeys("down", m.Down)
ui.Handle("/sys/kbd/<enter>", func(ui.Event) {
switch m.SelectedItem().Val {
case "start":
c.Start()
ui.StopLoop()
case "stop":
c.Stop()
ui.StopLoop()
case "remove":
c.Remove()
ui.StopLoop()
case "cancel":
ui.StopLoop()
}
})
ui.Handle("/sys/kbd/", func(ui.Event) {
ui.StopLoop()
})
ui.Loop()
}