2017-06-12 14:12:03 +00:00
|
|
|
package collector
|
2017-01-20 12:41:26 +00:00
|
|
|
|
|
|
|
import (
|
2017-06-27 16:21:16 +00:00
|
|
|
"bufio"
|
|
|
|
"io"
|
|
|
|
|
2017-06-27 15:46:03 +00:00
|
|
|
"github.com/bcicen/ctop/models"
|
2017-01-20 12:41:26 +00:00
|
|
|
api "github.com/fsouza/go-dockerclient"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Docker collector
|
|
|
|
type Docker struct {
|
2017-06-27 15:46:03 +00:00
|
|
|
models.Metrics
|
2017-02-02 07:09:43 +00:00
|
|
|
id string
|
|
|
|
client *api.Client
|
|
|
|
running bool
|
2017-06-27 15:46:03 +00:00
|
|
|
stream chan models.Metrics
|
2017-01-20 12:41:26 +00:00
|
|
|
done chan bool
|
|
|
|
lastCpu float64
|
|
|
|
lastSysCpu float64
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewDocker(client *api.Client, id string) *Docker {
|
2017-02-02 07:09:43 +00:00
|
|
|
return &Docker{
|
2017-06-27 15:46:03 +00:00
|
|
|
Metrics: models.Metrics{},
|
2017-02-02 07:09:43 +00:00
|
|
|
id: id,
|
|
|
|
client: client,
|
2017-01-20 12:41:26 +00:00
|
|
|
}
|
2017-02-02 07:09:43 +00:00
|
|
|
}
|
2017-01-20 12:41:26 +00:00
|
|
|
|
2017-02-02 07:09:43 +00:00
|
|
|
func (c *Docker) Start() {
|
2017-02-23 02:24:26 +00:00
|
|
|
c.done = make(chan bool)
|
2017-06-27 15:46:03 +00:00
|
|
|
c.stream = make(chan models.Metrics)
|
2017-01-20 12:41:26 +00:00
|
|
|
stats := make(chan *api.Stats)
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
opts := api.StatsOptions{
|
2017-02-02 07:09:43 +00:00
|
|
|
ID: c.id,
|
2017-01-20 12:41:26 +00:00
|
|
|
Stats: stats,
|
|
|
|
Stream: true,
|
|
|
|
Done: c.done,
|
|
|
|
}
|
2017-02-02 07:09:43 +00:00
|
|
|
c.client.Stats(opts)
|
|
|
|
c.running = false
|
2017-01-20 12:41:26 +00:00
|
|
|
}()
|
|
|
|
|
|
|
|
go func() {
|
2017-01-26 00:53:03 +00:00
|
|
|
defer close(c.stream)
|
2017-01-20 12:41:26 +00:00
|
|
|
for s := range stats {
|
|
|
|
c.ReadCPU(s)
|
|
|
|
c.ReadMem(s)
|
|
|
|
c.ReadNet(s)
|
2017-03-12 01:35:40 +00:00
|
|
|
c.ReadIO(s)
|
2017-01-20 12:41:26 +00:00
|
|
|
c.stream <- c.Metrics
|
|
|
|
}
|
2017-02-23 02:24:26 +00:00
|
|
|
log.Infof("collector stopped for container: %s", c.id)
|
2017-01-20 12:41:26 +00:00
|
|
|
}()
|
|
|
|
|
2017-02-02 07:09:43 +00:00
|
|
|
c.running = true
|
2017-02-23 02:24:26 +00:00
|
|
|
log.Infof("collector started for container: %s", c.id)
|
2017-02-02 07:09:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Docker) Running() bool {
|
|
|
|
return c.running
|
2017-01-20 12:41:26 +00:00
|
|
|
}
|
|
|
|
|
2017-06-27 15:46:03 +00:00
|
|
|
func (c *Docker) Stream() chan models.Metrics {
|
2017-01-20 12:41:26 +00:00
|
|
|
return c.stream
|
|
|
|
}
|
|
|
|
|
2017-06-27 16:21:16 +00:00
|
|
|
func (c *Docker) StreamLogs() (chan string, error) {
|
|
|
|
r, w := io.Pipe()
|
|
|
|
logCh := make(chan string)
|
|
|
|
|
|
|
|
opts := api.LogsOptions{
|
|
|
|
Container: c.id,
|
|
|
|
OutputStream: w,
|
|
|
|
ErrorStream: w,
|
|
|
|
Stdout: true,
|
|
|
|
Stderr: true,
|
|
|
|
Tail: "10",
|
|
|
|
Follow: true,
|
|
|
|
Timestamps: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
go tailLogs(r, logCh)
|
|
|
|
go func() {
|
|
|
|
err := c.client.Logs(opts)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("error reading container logs: %s", err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
return logCh, nil
|
|
|
|
}
|
|
|
|
|
2017-01-20 12:41:26 +00:00
|
|
|
// Stop collector
|
|
|
|
func (c *Docker) Stop() {
|
|
|
|
c.done <- true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Docker) ReadCPU(stats *api.Stats) {
|
|
|
|
ncpus := float64(len(stats.CPUStats.CPUUsage.PercpuUsage))
|
|
|
|
total := float64(stats.CPUStats.CPUUsage.TotalUsage)
|
|
|
|
system := float64(stats.CPUStats.SystemCPUUsage)
|
|
|
|
|
|
|
|
cpudiff := total - c.lastCpu
|
|
|
|
syscpudiff := system - c.lastSysCpu
|
|
|
|
|
|
|
|
c.CPUUtil = round((cpudiff / syscpudiff * 100) * ncpus)
|
|
|
|
c.lastCpu = total
|
|
|
|
c.lastSysCpu = system
|
2017-03-12 01:35:40 +00:00
|
|
|
c.Pids = int(stats.PidsStats.Current)
|
2017-01-20 12:41:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Docker) ReadMem(stats *api.Stats) {
|
|
|
|
c.MemUsage = int64(stats.MemoryStats.Usage)
|
|
|
|
c.MemLimit = int64(stats.MemoryStats.Limit)
|
2017-06-10 13:00:54 +00:00
|
|
|
c.MemPercent = percent(float64(c.MemUsage), float64(c.MemLimit))
|
2017-01-20 12:41:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Docker) ReadNet(stats *api.Stats) {
|
|
|
|
var rx, tx int64
|
|
|
|
for _, network := range stats.Networks {
|
|
|
|
rx += int64(network.RxBytes)
|
|
|
|
tx += int64(network.TxBytes)
|
|
|
|
}
|
|
|
|
c.NetRx, c.NetTx = rx, tx
|
|
|
|
}
|
2017-03-12 01:35:40 +00:00
|
|
|
|
|
|
|
func (c *Docker) ReadIO(stats *api.Stats) {
|
|
|
|
var read, write int64
|
|
|
|
for _, blk := range stats.BlkioStats.IOServiceBytesRecursive {
|
|
|
|
if blk.Op == "Read" {
|
|
|
|
read = int64(blk.Value)
|
|
|
|
}
|
|
|
|
if blk.Op == "Write" {
|
|
|
|
write = int64(blk.Value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
c.IOBytesRead, c.IOBytesWrite = read, write
|
|
|
|
}
|
2017-06-27 16:21:16 +00:00
|
|
|
|
|
|
|
func tailLogs(reader io.Reader, ch chan string) {
|
|
|
|
scanner := bufio.NewScanner(reader)
|
|
|
|
for scanner.Scan() {
|
|
|
|
ch <- scanner.Text()
|
|
|
|
}
|
|
|
|
}
|