123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303 |
- package app
- import (
- "fmt"
- "os"
- "os/signal"
- "strconv"
- "strings"
- "syscall"
- "golang.org/x/net/context"
- "github.com/Sirupsen/logrus"
- "github.com/codegangsta/cli"
- "github.com/docker/libcompose/project"
- "github.com/docker/libcompose/project/options"
- )
- // ProjectAction is an adapter to allow the use of ordinary functions as libcompose actions.
- // Any function that has the appropriate signature can be register as an action on a codegansta/cli command.
- //
- // cli.Command{
- // Name: "ps",
- // Usage: "List containers",
- // Action: app.WithProject(factory, app.ProjectPs),
- // }
- type ProjectAction func(project project.APIProject, c *cli.Context) error
- // BeforeApp is an action that is executed before any cli command.
- func BeforeApp(c *cli.Context) error {
- if c.GlobalBool("verbose") {
- logrus.SetLevel(logrus.DebugLevel)
- }
- logrus.Warning("Note: This is an experimental alternate implementation of the Compose CLI (https://github.com/docker/compose)")
- return nil
- }
- // WithProject is a helper function to create a cli.Command action with a ProjectFactory.
- func WithProject(factory ProjectFactory, action ProjectAction) func(context *cli.Context) error {
- return func(context *cli.Context) error {
- p, err := factory.Create(context)
- if err != nil {
- logrus.Fatalf("Failed to read project: %v", err)
- }
- return action(p, context)
- }
- }
- // ProjectPs lists the containers.
- func ProjectPs(p project.APIProject, c *cli.Context) error {
- qFlag := c.Bool("q")
- allInfo, err := p.Ps(context.Background(), qFlag, c.Args()...)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- os.Stdout.WriteString(allInfo.String(!qFlag))
- return nil
- }
- // ProjectPort prints the public port for a port binding.
- func ProjectPort(p project.APIProject, c *cli.Context) error {
- if len(c.Args()) != 2 {
- return cli.NewExitError("Please pass arguments in the form: SERVICE PORT", 1)
- }
- index := c.Int("index")
- protocol := c.String("protocol")
- serviceName := c.Args()[0]
- privatePort := c.Args()[1]
- port, err := p.Port(context.Background(), index, protocol, serviceName, privatePort)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- fmt.Println(port)
- return nil
- }
- // ProjectStop stops all services.
- func ProjectStop(p project.APIProject, c *cli.Context) error {
- err := p.Stop(context.Background(), c.Int("timeout"), c.Args()...)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- return nil
- }
- // ProjectDown brings all services down (stops and clean containers).
- func ProjectDown(p project.APIProject, c *cli.Context) error {
- options := options.Down{
- RemoveVolume: c.Bool("volumes"),
- RemoveImages: options.ImageType(c.String("rmi")),
- RemoveOrphans: c.Bool("remove-orphans"),
- }
- err := p.Down(context.Background(), options, c.Args()...)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- return nil
- }
- // ProjectBuild builds or rebuilds services.
- func ProjectBuild(p project.APIProject, c *cli.Context) error {
- config := options.Build{
- NoCache: c.Bool("no-cache"),
- ForceRemove: c.Bool("force-rm"),
- Pull: c.Bool("pull"),
- }
- err := p.Build(context.Background(), config, c.Args()...)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- return nil
- }
- // ProjectCreate creates all services but do not start them.
- func ProjectCreate(p project.APIProject, c *cli.Context) error {
- options := options.Create{
- NoRecreate: c.Bool("no-recreate"),
- ForceRecreate: c.Bool("force-recreate"),
- NoBuild: c.Bool("no-build"),
- }
- err := p.Create(context.Background(), options, c.Args()...)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- return nil
- }
- // ProjectUp brings all services up.
- func ProjectUp(p project.APIProject, c *cli.Context) error {
- options := options.Up{
- Create: options.Create{
- NoRecreate: c.Bool("no-recreate"),
- ForceRecreate: c.Bool("force-recreate"),
- NoBuild: c.Bool("no-build"),
- },
- }
- ctx, cancelFun := context.WithCancel(context.Background())
- err := p.Up(ctx, options, c.Args()...)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- if !c.Bool("d") {
- signalChan := make(chan os.Signal, 1)
- cleanupDone := make(chan bool)
- signal.Notify(signalChan, syscall.SIGINT, syscall.SIGTERM)
- errChan := make(chan error)
- go func() {
- errChan <- p.Log(ctx, true, c.Args()...)
- }()
- go func() {
- select {
- case <-signalChan:
- fmt.Printf("\nGracefully stopping...\n")
- cancelFun()
- ProjectStop(p, c)
- cleanupDone <- true
- case err := <-errChan:
- if err != nil {
- logrus.Fatal(err)
- }
- cleanupDone <- true
- }
- }()
- <-cleanupDone
- return nil
- }
- return nil
- }
- // ProjectRun runs a given command within a service's container.
- func ProjectRun(p project.APIProject, c *cli.Context) error {
- if len(c.Args()) == 1 {
- logrus.Fatal("No service specified")
- }
- serviceName := c.Args()[0]
- commandParts := c.Args()[1:]
- exitCode, err := p.Run(context.Background(), serviceName, commandParts)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- return cli.NewExitError("", exitCode)
- }
- // ProjectStart starts services.
- func ProjectStart(p project.APIProject, c *cli.Context) error {
- err := p.Start(context.Background(), c.Args()...)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- return nil
- }
- // ProjectRestart restarts services.
- func ProjectRestart(p project.APIProject, c *cli.Context) error {
- err := p.Restart(context.Background(), c.Int("timeout"), c.Args()...)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- return nil
- }
- // ProjectLog gets services logs.
- func ProjectLog(p project.APIProject, c *cli.Context) error {
- err := p.Log(context.Background(), c.Bool("follow"), c.Args()...)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- return nil
- }
- // ProjectPull pulls images for services.
- func ProjectPull(p project.APIProject, c *cli.Context) error {
- err := p.Pull(context.Background(), c.Args()...)
- if err != nil && !c.Bool("ignore-pull-failures") {
- return cli.NewExitError(err.Error(), 1)
- }
- return nil
- }
- // ProjectDelete deletes services.
- func ProjectDelete(p project.APIProject, c *cli.Context) error {
- options := options.Delete{
- RemoveVolume: c.Bool("v"),
- }
- if !c.Bool("force") {
- options.BeforeDeleteCallback = func(stoppedContainers []string) bool {
- fmt.Printf("Going to remove %v\nAre you sure? [yN]\n", strings.Join(stoppedContainers, ", "))
- var answer string
- _, err := fmt.Scanln(&answer)
- if err != nil {
- logrus.Error(err)
- return false
- }
- if answer != "y" && answer != "Y" {
- return false
- }
- return true
- }
- }
- err := p.Delete(context.Background(), options, c.Args()...)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- return nil
- }
- // ProjectKill forces stop service containers.
- func ProjectKill(p project.APIProject, c *cli.Context) error {
- err := p.Kill(context.Background(), c.String("signal"), c.Args()...)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- return nil
- }
- // ProjectPause pauses service containers.
- func ProjectPause(p project.APIProject, c *cli.Context) error {
- err := p.Pause(context.Background(), c.Args()...)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- return nil
- }
- // ProjectUnpause unpauses service containers.
- func ProjectUnpause(p project.APIProject, c *cli.Context) error {
- err := p.Unpause(context.Background(), c.Args()...)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- return nil
- }
- // ProjectScale scales services.
- func ProjectScale(p project.APIProject, c *cli.Context) error {
- servicesScale := map[string]int{}
- for _, arg := range c.Args() {
- kv := strings.SplitN(arg, "=", 2)
- if len(kv) != 2 {
- return cli.NewExitError(fmt.Sprintf("Invalid scale parameter: %s", arg), 2)
- }
- name := kv[0]
- count, err := strconv.Atoi(kv[1])
- if err != nil {
- return cli.NewExitError(fmt.Sprintf("Invalid scale parameter: %v", err), 2)
- }
- servicesScale[name] = count
- }
- err := p.Scale(context.Background(), c.Int("timeout"), servicesScale)
- if err != nil {
- return cli.NewExitError(err.Error(), 1)
- }
- return nil
- }
|