123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124 |
- package container
- import (
- "fmt"
- "strconv"
- "golang.org/x/net/context"
- "github.com/Sirupsen/logrus"
- "github.com/docker/docker/api/types"
- "github.com/docker/docker/api/types/events"
- "github.com/docker/docker/api/types/filters"
- "github.com/docker/docker/cli/command"
- "github.com/docker/docker/cli/command/system"
- clientapi "github.com/docker/docker/client"
- )
- func waitExitOrRemoved(dockerCli *command.DockerCli, ctx context.Context, containerID string, waitRemove bool) (chan int, error) {
- if len(containerID) == 0 {
- // containerID can never be empty
- panic("Internal Error: waitExitOrRemoved needs a containerID as parameter")
- }
- statusChan := make(chan int)
- exitCode := 125
- eventProcessor := func(e events.Message, err error) error {
- if err != nil {
- statusChan <- exitCode
- return fmt.Errorf("failed to decode event: %v", err)
- }
- stopProcessing := false
- switch e.Status {
- case "die":
- if v, ok := e.Actor.Attributes["exitCode"]; ok {
- code, cerr := strconv.Atoi(v)
- if cerr != nil {
- logrus.Errorf("failed to convert exitcode '%q' to int: %v", v, cerr)
- } else {
- exitCode = code
- }
- }
- if !waitRemove {
- stopProcessing = true
- }
- case "detach":
- exitCode = 0
- stopProcessing = true
- case "destroy":
- stopProcessing = true
- }
- if stopProcessing {
- statusChan <- exitCode
- // stop the loop processing
- return fmt.Errorf("done")
- }
- return nil
- }
- // Get events via Events API
- f := filters.NewArgs()
- f.Add("type", "container")
- f.Add("container", containerID)
- options := types.EventsOptions{
- Filters: f,
- }
- resBody, err := dockerCli.Client().Events(ctx, options)
- if err != nil {
- return nil, fmt.Errorf("can't get events from daemon: %v", err)
- }
- go system.DecodeEvents(resBody, eventProcessor)
- return statusChan, nil
- }
- // getExitCode performs an inspect on the container. It returns
- // the running state and the exit code.
- func getExitCode(dockerCli *command.DockerCli, ctx context.Context, containerID string) (bool, int, error) {
- c, err := dockerCli.Client().ContainerInspect(ctx, containerID)
- if err != nil {
- // If we can't connect, then the daemon probably died.
- if err != clientapi.ErrConnectionFailed {
- return false, -1, err
- }
- return false, -1, nil
- }
- return c.State.Running, c.State.ExitCode, nil
- }
- func parallelOperation(ctx context.Context, cids []string, op func(ctx context.Context, id string) error) chan error {
- if len(cids) == 0 {
- return nil
- }
- const defaultParallel int = 50
- sem := make(chan struct{}, defaultParallel)
- errChan := make(chan error)
- // make sure result is printed in correct order
- output := map[string]chan error{}
- for _, c := range cids {
- output[c] = make(chan error, 1)
- }
- go func() {
- for _, c := range cids {
- err := <-output[c]
- errChan <- err
- }
- }()
- go func() {
- for _, c := range cids {
- sem <- struct{}{} // Wait for active queue sem to drain.
- go func(container string) {
- output[container] <- op(ctx, container)
- <-sem
- }(c)
- }
- }()
- return errChan
- }
|