123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400 |
- package client
- import (
- "bytes"
- "encoding/json"
- "errors"
- "fmt"
- "net/http"
- "strings"
- "text/tabwriter"
- flag "github.com/docker/docker/libnetwork/client/mflag"
- "github.com/docker/docker/libnetwork/netutils"
- "github.com/docker/docker/pkg/stringid"
- )
- var (
- serviceCommands = []command{
- {"publish", "Publish a service"},
- {"unpublish", "Remove a service"},
- {"attach", "Attach a backend (container) to the service"},
- {"detach", "Detach the backend from the service"},
- {"ls", "Lists all services"},
- {"info", "Display information about a service"},
- }
- )
- func lookupServiceID(cli *NetworkCli, nwName, svNameID string) (string, error) {
- // Sanity Check
- obj, _, err := readBody(cli.call("GET", fmt.Sprintf("/networks?name=%s", nwName), nil, nil))
- if err != nil {
- return "", err
- }
- var nwList []networkResource
- if err = json.Unmarshal(obj, &nwList); err != nil {
- return "", err
- }
- if len(nwList) == 0 {
- return "", fmt.Errorf("Network %s does not exist", nwName)
- }
- if nwName == "" {
- obj, _, err := readBody(cli.call("GET", "/networks/"+nwList[0].ID, nil, nil))
- if err != nil {
- return "", err
- }
- networkResource := &networkResource{}
- if err := json.NewDecoder(bytes.NewReader(obj)).Decode(networkResource); err != nil {
- return "", err
- }
- nwName = networkResource.Name
- }
- // Query service by name
- obj, statusCode, err := readBody(cli.call("GET", fmt.Sprintf("/services?name=%s", svNameID), nil, nil))
- if err != nil {
- return "", err
- }
- if statusCode != http.StatusOK {
- return "", fmt.Errorf("name query failed for %s due to: (%d) %s", svNameID, statusCode, string(obj))
- }
- var list []*serviceResource
- if err = json.Unmarshal(obj, &list); err != nil {
- return "", err
- }
- for _, sr := range list {
- if sr.Network == nwName {
- return sr.ID, nil
- }
- }
- // Query service by Partial-id (this covers full id as well)
- obj, statusCode, err = readBody(cli.call("GET", fmt.Sprintf("/services?partial-id=%s", svNameID), nil, nil))
- if err != nil {
- return "", err
- }
- if statusCode != http.StatusOK {
- return "", fmt.Errorf("partial-id match query failed for %s due to: (%d) %s", svNameID, statusCode, string(obj))
- }
- if err = json.Unmarshal(obj, &list); err != nil {
- return "", err
- }
- for _, sr := range list {
- if sr.Network == nwName {
- return sr.ID, nil
- }
- }
- return "", fmt.Errorf("Service %s not found on network %s", svNameID, nwName)
- }
- func lookupContainerID(cli *NetworkCli, cnNameID string) (string, error) {
- // Container is a Docker resource, ask docker about it.
- // In case of connection error, we assume we are running in dnet and return whatever was passed to us
- obj, _, err := readBody(cli.call("GET", fmt.Sprintf("/containers/%s/json", cnNameID), nil, nil))
- if err != nil {
- // We are probably running outside of docker
- return cnNameID, nil
- }
- var x map[string]interface{}
- err = json.Unmarshal(obj, &x)
- if err != nil {
- return "", err
- }
- if iid, ok := x["Id"]; ok {
- if id, ok := iid.(string); ok {
- return id, nil
- }
- return "", errors.New("Unexpected data type for container ID in json response")
- }
- return "", errors.New("Cannot find container ID in json response")
- }
- func lookupSandboxID(cli *NetworkCli, containerID string) (string, error) {
- obj, _, err := readBody(cli.call("GET", fmt.Sprintf("/sandboxes?partial-container-id=%s", containerID), nil, nil))
- if err != nil {
- return "", err
- }
- var sandboxList []SandboxResource
- err = json.Unmarshal(obj, &sandboxList)
- if err != nil {
- return "", err
- }
- if len(sandboxList) == 0 {
- return "", fmt.Errorf("cannot find sandbox for container: %s", containerID)
- }
- return sandboxList[0].ID, nil
- }
- // CmdService handles the service UI
- func (cli *NetworkCli) CmdService(chain string, args ...string) error {
- cmd := cli.Subcmd(chain, "service", "COMMAND [OPTIONS] [arg...]", serviceUsage(chain), false)
- cmd.Require(flag.Min, 1)
- err := cmd.ParseFlags(args, true)
- if err == nil {
- cmd.Usage()
- return fmt.Errorf("Invalid command : %v", args)
- }
- return err
- }
- // Parse service name for "SERVICE[.NETWORK]" format
- func parseServiceName(name string) (string, string) {
- s := strings.Split(name, ".")
- var sName, nName string
- if len(s) > 1 {
- nName = s[len(s)-1]
- sName = strings.Join(s[:len(s)-1], ".")
- } else {
- sName = s[0]
- }
- return sName, nName
- }
- // CmdServicePublish handles service create UI
- func (cli *NetworkCli) CmdServicePublish(chain string, args ...string) error {
- cmd := cli.Subcmd(chain, "publish", "SERVICE[.NETWORK]", "Publish a new service on a network", false)
- flAlias := flag.NewListOpts(netutils.ValidateAlias)
- cmd.Var(&flAlias, []string{"-alias"}, "Add alias to self")
- cmd.Require(flag.Exact, 1)
- err := cmd.ParseFlags(args, true)
- if err != nil {
- return err
- }
- sn, nn := parseServiceName(cmd.Arg(0))
- sc := serviceCreate{Name: sn, Network: nn, MyAliases: flAlias.GetAll()}
- obj, _, err := readBody(cli.call("POST", "/services", sc, nil))
- if err != nil {
- return err
- }
- var replyID string
- err = json.Unmarshal(obj, &replyID)
- if err != nil {
- return err
- }
- fmt.Fprintf(cli.out, "%s\n", replyID)
- return nil
- }
- // CmdServiceUnpublish handles service delete UI
- func (cli *NetworkCli) CmdServiceUnpublish(chain string, args ...string) error {
- cmd := cli.Subcmd(chain, "unpublish", "SERVICE[.NETWORK]", "Removes a service", false)
- force := cmd.Bool([]string{"f", "-force"}, false, "force unpublish service")
- cmd.Require(flag.Exact, 1)
- err := cmd.ParseFlags(args, true)
- if err != nil {
- return err
- }
- sn, nn := parseServiceName(cmd.Arg(0))
- serviceID, err := lookupServiceID(cli, nn, sn)
- if err != nil {
- return err
- }
- sd := serviceDelete{Name: sn, Force: *force}
- _, _, err = readBody(cli.call("DELETE", "/services/"+serviceID, sd, nil))
- return err
- }
- // CmdServiceLs handles service list UI
- func (cli *NetworkCli) CmdServiceLs(chain string, args ...string) error {
- cmd := cli.Subcmd(chain, "ls", "SERVICE", "Lists all the services on a network", false)
- flNetwork := cmd.String([]string{"net", "-network"}, "", "Only show the services that are published on the specified network")
- quiet := cmd.Bool([]string{"q", "-quiet"}, false, "Only display numeric IDs")
- noTrunc := cmd.Bool([]string{"#notrunc", "-no-trunc"}, false, "Do not truncate the output")
- err := cmd.ParseFlags(args, true)
- if err != nil {
- return err
- }
- var obj []byte
- if *flNetwork == "" {
- obj, _, err = readBody(cli.call("GET", "/services", nil, nil))
- } else {
- obj, _, err = readBody(cli.call("GET", "/services?network="+*flNetwork, nil, nil))
- }
- if err != nil {
- return err
- }
- var serviceResources []serviceResource
- err = json.Unmarshal(obj, &serviceResources)
- if err != nil {
- fmt.Println(err)
- return err
- }
- wr := tabwriter.NewWriter(cli.out, 20, 1, 3, ' ', 0)
- // unless quiet (-q) is specified, print field titles
- if !*quiet {
- fmt.Fprintln(wr, "SERVICE ID\tNAME\tNETWORK\tCONTAINER\tSANDBOX")
- }
- for _, sr := range serviceResources {
- ID := sr.ID
- bkID, sbID, err := getBackendID(cli, ID)
- if err != nil {
- return err
- }
- if !*noTrunc {
- ID = stringid.TruncateID(ID)
- bkID = stringid.TruncateID(bkID)
- sbID = stringid.TruncateID(sbID)
- }
- if !*quiet {
- fmt.Fprintf(wr, "%s\t%s\t%s\t%s\t%s\n", ID, sr.Name, sr.Network, bkID, sbID)
- } else {
- fmt.Fprintln(wr, ID)
- }
- }
- wr.Flush()
- return nil
- }
- func getBackendID(cli *NetworkCli, servID string) (string, string, error) {
- var (
- obj []byte
- err error
- bk string
- sb string
- )
- if obj, _, err = readBody(cli.call("GET", "/services/"+servID+"/backend", nil, nil)); err == nil {
- var sr SandboxResource
- if err := json.NewDecoder(bytes.NewReader(obj)).Decode(&sr); err == nil {
- bk = sr.ContainerID
- sb = sr.ID
- } else {
- // Only print a message, don't make the caller cli fail for this
- fmt.Fprintf(cli.out, "Failed to retrieve backend list for service %s (%v)\n", servID, err)
- }
- }
- return bk, sb, err
- }
- // CmdServiceInfo handles service info UI
- func (cli *NetworkCli) CmdServiceInfo(chain string, args ...string) error {
- cmd := cli.Subcmd(chain, "info", "SERVICE[.NETWORK]", "Displays detailed information about a service", false)
- cmd.Require(flag.Min, 1)
- err := cmd.ParseFlags(args, true)
- if err != nil {
- return err
- }
- sn, nn := parseServiceName(cmd.Arg(0))
- serviceID, err := lookupServiceID(cli, nn, sn)
- if err != nil {
- return err
- }
- obj, _, err := readBody(cli.call("GET", "/services/"+serviceID, nil, nil))
- if err != nil {
- return err
- }
- sr := &serviceResource{}
- if err := json.NewDecoder(bytes.NewReader(obj)).Decode(sr); err != nil {
- return err
- }
- fmt.Fprintf(cli.out, "Service Id: %s\n", sr.ID)
- fmt.Fprintf(cli.out, "\tName: %s\n", sr.Name)
- fmt.Fprintf(cli.out, "\tNetwork: %s\n", sr.Network)
- return nil
- }
- // CmdServiceAttach handles service attach UI
- func (cli *NetworkCli) CmdServiceAttach(chain string, args ...string) error {
- cmd := cli.Subcmd(chain, "attach", "CONTAINER SERVICE[.NETWORK]", "Sets a container as a service backend", false)
- flAlias := flag.NewListOpts(netutils.ValidateAlias)
- cmd.Var(&flAlias, []string{"-alias"}, "Add alias for another container")
- cmd.Require(flag.Min, 2)
- err := cmd.ParseFlags(args, true)
- if err != nil {
- return err
- }
- containerID, err := lookupContainerID(cli, cmd.Arg(0))
- if err != nil {
- return err
- }
- sandboxID, err := lookupSandboxID(cli, containerID)
- if err != nil {
- return err
- }
- sn, nn := parseServiceName(cmd.Arg(1))
- serviceID, err := lookupServiceID(cli, nn, sn)
- if err != nil {
- return err
- }
- nc := serviceAttach{SandboxID: sandboxID, Aliases: flAlias.GetAll()}
- _, _, err = readBody(cli.call("POST", "/services/"+serviceID+"/backend", nc, nil))
- return err
- }
- // CmdServiceDetach handles service detach UI
- func (cli *NetworkCli) CmdServiceDetach(chain string, args ...string) error {
- cmd := cli.Subcmd(chain, "detach", "CONTAINER SERVICE", "Removes a container from service backend", false)
- cmd.Require(flag.Min, 2)
- err := cmd.ParseFlags(args, true)
- if err != nil {
- return err
- }
- sn, nn := parseServiceName(cmd.Arg(1))
- containerID, err := lookupContainerID(cli, cmd.Arg(0))
- if err != nil {
- return err
- }
- sandboxID, err := lookupSandboxID(cli, containerID)
- if err != nil {
- return err
- }
- serviceID, err := lookupServiceID(cli, nn, sn)
- if err != nil {
- return err
- }
- _, _, err = readBody(cli.call("DELETE", "/services/"+serviceID+"/backend/"+sandboxID, nil, nil))
- if err != nil {
- return err
- }
- return nil
- }
- func serviceUsage(chain string) string {
- help := "Commands:\n"
- for _, cmd := range serviceCommands {
- help += fmt.Sprintf(" %-10.10s%s\n", cmd.name, cmd.description)
- }
- help += fmt.Sprintf("\nRun '%s service COMMAND --help' for more information on a command.", chain)
- return help
- }
|