mirror of
https://github.com/rancher/os.git
synced 2025-07-07 03:48:38 +00:00
303 lines
7.9 KiB
Go
303 lines
7.9 KiB
Go
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
|
|
}
|