|
@@ -13,64 +13,77 @@ import (
|
|
|
|
|
|
var (
|
|
var (
|
|
serviceCommands = []command{
|
|
serviceCommands = []command{
|
|
- {"create", "Create a service endpoint"},
|
|
|
|
- {"rm", "Remove a service endpoint"},
|
|
|
|
- {"join", "Join a container to a service endpoint"},
|
|
|
|
- {"leave", "Leave a container from a service endpoint"},
|
|
|
|
- {"ls", "Lists all service endpoints on a network"},
|
|
|
|
- {"info", "Display information of a service endpoint"},
|
|
|
|
|
|
+ {"publish", "Publish a service"},
|
|
|
|
+ {"unpublish", "Remove a service"},
|
|
|
|
+ {"attach", "Attach a provider (container) to the service"},
|
|
|
|
+ {"detach", "Detach the provider from the service"},
|
|
|
|
+ {"ls", "Lists all services"},
|
|
|
|
+ {"info", "Display information about a service"},
|
|
}
|
|
}
|
|
)
|
|
)
|
|
|
|
|
|
-func lookupServiceID(cli *NetworkCli, networkID string, nameID string) (string, error) {
|
|
|
|
- obj, statusCode, err := readBody(cli.call("GET", fmt.Sprintf("/networks/%s/endpoints?name=%s", networkID, nameID), nil, nil))
|
|
|
|
|
|
+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)
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // Query service by name
|
|
|
|
+ obj, statusCode, err := readBody(cli.call("GET", fmt.Sprintf("/services?name=%s", svNameID), nil, nil))
|
|
if err != nil {
|
|
if err != nil {
|
|
return "", err
|
|
return "", err
|
|
}
|
|
}
|
|
|
|
|
|
if statusCode != http.StatusOK {
|
|
if statusCode != http.StatusOK {
|
|
- return "", fmt.Errorf("name query failed for %s due to : statuscode(%d) %v", nameID, statusCode, string(obj))
|
|
|
|
|
|
+ return "", fmt.Errorf("name query failed for %s due to: (%d) %s", svNameID, statusCode, string(obj))
|
|
}
|
|
}
|
|
|
|
|
|
- var list []*networkResource
|
|
|
|
- err = json.Unmarshal(obj, &list)
|
|
|
|
- if err != nil {
|
|
|
|
|
|
+ var list []*serviceResource
|
|
|
|
+ if err = json.Unmarshal(obj, &list); err != nil {
|
|
return "", err
|
|
return "", err
|
|
}
|
|
}
|
|
- if len(list) > 0 {
|
|
|
|
- // name query filter will always return a single-element collection
|
|
|
|
- return list[0].ID, nil
|
|
|
|
|
|
+ for _, sr := range list {
|
|
|
|
+ if sr.Network == nwName {
|
|
|
|
+ return sr.ID, nil
|
|
|
|
+ }
|
|
}
|
|
}
|
|
|
|
|
|
- // Check for Partial-id
|
|
|
|
- obj, statusCode, err = readBody(cli.call("GET", fmt.Sprintf("/networks/%s/endpoints?partial-id=%s", networkID, nameID), nil, 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 {
|
|
if err != nil {
|
|
return "", err
|
|
return "", err
|
|
}
|
|
}
|
|
|
|
|
|
if statusCode != http.StatusOK {
|
|
if statusCode != http.StatusOK {
|
|
- return "", fmt.Errorf("partial-id match query failed for %s due to : statuscode(%d) %v", nameID, statusCode, string(obj))
|
|
|
|
|
|
+ return "", fmt.Errorf("partial-id match query failed for %s due to: (%d) %s", svNameID, statusCode, string(obj))
|
|
}
|
|
}
|
|
|
|
|
|
- err = json.Unmarshal(obj, &list)
|
|
|
|
- if err != nil {
|
|
|
|
|
|
+ if err = json.Unmarshal(obj, &list); err != nil {
|
|
return "", err
|
|
return "", err
|
|
}
|
|
}
|
|
- if len(list) == 0 {
|
|
|
|
- return "", fmt.Errorf("resource not found %s", nameID)
|
|
|
|
- }
|
|
|
|
- if len(list) > 1 {
|
|
|
|
- return "", fmt.Errorf("multiple services matching the partial identifier (%s). Please use full identifier", nameID)
|
|
|
|
|
|
+ for _, sr := range list {
|
|
|
|
+ if sr.Network == nwName {
|
|
|
|
+ return sr.ID, nil
|
|
|
|
+ }
|
|
}
|
|
}
|
|
- return list[0].ID, nil
|
|
|
|
|
|
+
|
|
|
|
+ return "", fmt.Errorf("Service %s not found on network %s", svNameID, nwName)
|
|
}
|
|
}
|
|
|
|
|
|
-func lookupContainerID(cli *NetworkCli, nameID string) (string, error) {
|
|
|
|
|
|
+func lookupContainerID(cli *NetworkCli, cnNameID string) (string, error) {
|
|
// TODO : containerID to sandbox-key ?
|
|
// TODO : containerID to sandbox-key ?
|
|
- return nameID, nil
|
|
|
|
|
|
+ return cnNameID, nil
|
|
}
|
|
}
|
|
|
|
|
|
-// CmdService handles the network service UI
|
|
|
|
|
|
+// CmdService handles the service UI
|
|
func (cli *NetworkCli) CmdService(chain string, args ...string) error {
|
|
func (cli *NetworkCli) CmdService(chain string, args ...string) error {
|
|
cmd := cli.Subcmd(chain, "service", "COMMAND [OPTIONS] [arg...]", serviceUsage(chain), false)
|
|
cmd := cli.Subcmd(chain, "service", "COMMAND [OPTIONS] [arg...]", serviceUsage(chain), false)
|
|
cmd.Require(flag.Min, 1)
|
|
cmd.Require(flag.Min, 1)
|
|
@@ -82,23 +95,25 @@ func (cli *NetworkCli) CmdService(chain string, args ...string) error {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
|
|
-// CmdServiceCreate handles service create UI
|
|
|
|
-func (cli *NetworkCli) CmdServiceCreate(chain string, args ...string) error {
|
|
|
|
- cmd := cli.Subcmd(chain, "create", "SERVICE NETWORK", "Creates a new service on a network", false)
|
|
|
|
- cmd.Require(flag.Min, 2)
|
|
|
|
|
|
+// CmdServicePublish handles service create UI
|
|
|
|
+func (cli *NetworkCli) CmdServicePublish(chain string, args ...string) error {
|
|
|
|
+ cmd := cli.Subcmd(chain, "publish", "SERVICE", "Publish a new service on a network", false)
|
|
|
|
+ flNetwork := cmd.String([]string{"net", "-network"}, "", "Network where to publish the service")
|
|
|
|
+ cmd.Require(flag.Min, 1)
|
|
|
|
+
|
|
err := cmd.ParseFlags(args, true)
|
|
err := cmd.ParseFlags(args, true)
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
|
|
- networkID, err := lookupNetworkID(cli, cmd.Arg(1))
|
|
|
|
- if err != nil {
|
|
|
|
- return err
|
|
|
|
|
|
+ // Default network changes will come later
|
|
|
|
+ nw := "docker0"
|
|
|
|
+ if *flNetwork != "" {
|
|
|
|
+ nw = *flNetwork
|
|
}
|
|
}
|
|
|
|
|
|
- ec := endpointCreate{Name: cmd.Arg(0), NetworkID: networkID}
|
|
|
|
-
|
|
|
|
- obj, _, err := readBody(cli.call("POST", "/networks/"+networkID+"/endpoints", ec, nil))
|
|
|
|
|
|
+ sc := serviceCreate{Name: cmd.Arg(0), Network: nw}
|
|
|
|
+ obj, _, err := readBody(cli.call("POST", "/services", sc, nil))
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
@@ -113,39 +128,40 @@ func (cli *NetworkCli) CmdServiceCreate(chain string, args ...string) error {
|
|
return nil
|
|
return nil
|
|
}
|
|
}
|
|
|
|
|
|
-// CmdServiceRm handles service delete UI
|
|
|
|
-func (cli *NetworkCli) CmdServiceRm(chain string, args ...string) error {
|
|
|
|
- cmd := cli.Subcmd(chain, "rm", "SERVICE NETWORK", "Deletes a service", false)
|
|
|
|
- cmd.Require(flag.Min, 2)
|
|
|
|
|
|
+// CmdServiceUnpublish handles service delete UI
|
|
|
|
+func (cli *NetworkCli) CmdServiceUnpublish(chain string, args ...string) error {
|
|
|
|
+ cmd := cli.Subcmd(chain, "unpublish", "SERVICE", "Removes a service", false)
|
|
|
|
+ flNetwork := cmd.String([]string{"net", "-network"}, "", "Network where to publish the service")
|
|
|
|
+ cmd.Require(flag.Min, 1)
|
|
|
|
+
|
|
err := cmd.ParseFlags(args, true)
|
|
err := cmd.ParseFlags(args, true)
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
|
|
- networkID, err := lookupNetworkID(cli, cmd.Arg(1))
|
|
|
|
- if err != nil {
|
|
|
|
- return err
|
|
|
|
|
|
+ // Default network changes will come later
|
|
|
|
+ nw := "docker0"
|
|
|
|
+ if *flNetwork != "" {
|
|
|
|
+ nw = *flNetwork
|
|
}
|
|
}
|
|
|
|
|
|
- serviceID, err := lookupServiceID(cli, networkID, cmd.Arg(0))
|
|
|
|
|
|
+ serviceID, err := lookupServiceID(cli, nw, cmd.Arg(0))
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
|
|
- _, _, err = readBody(cli.call("DELETE", "/networks/"+networkID+"/endpoints/"+serviceID, nil, nil))
|
|
|
|
- if err != nil {
|
|
|
|
- return err
|
|
|
|
- }
|
|
|
|
- return nil
|
|
|
|
|
|
+ _, _, err = readBody(cli.call("DELETE", "/services/"+serviceID, nil, nil))
|
|
|
|
+
|
|
|
|
+ return err
|
|
}
|
|
}
|
|
|
|
|
|
// CmdServiceLs handles service list UI
|
|
// CmdServiceLs handles service list UI
|
|
func (cli *NetworkCli) CmdServiceLs(chain string, args ...string) error {
|
|
func (cli *NetworkCli) CmdServiceLs(chain string, args ...string) error {
|
|
- cmd := cli.Subcmd(chain, "ls", "NETWORK", "Lists all the services on a network", false)
|
|
|
|
|
|
+ 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")
|
|
quiet := cmd.Bool([]string{"q", "-quiet"}, false, "Only display numeric IDs")
|
|
noTrunc := cmd.Bool([]string{"#notrunc", "-no-trunc"}, false, "Do not truncate the output")
|
|
noTrunc := cmd.Bool([]string{"#notrunc", "-no-trunc"}, false, "Do not truncate the output")
|
|
- nLatest := cmd.Bool([]string{"l", "-latest"}, false, "Show the latest network created")
|
|
|
|
- last := cmd.Int([]string{"n"}, -1, "Show n last created networks")
|
|
|
|
|
|
+
|
|
err := cmd.ParseFlags(args, true)
|
|
err := cmd.ParseFlags(args, true)
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
@@ -153,151 +169,174 @@ func (cli *NetworkCli) CmdServiceLs(chain string, args ...string) error {
|
|
|
|
|
|
cmd.Require(flag.Min, 1)
|
|
cmd.Require(flag.Min, 1)
|
|
|
|
|
|
- networkID, err := lookupNetworkID(cli, cmd.Arg(0))
|
|
|
|
- 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))
|
|
}
|
|
}
|
|
-
|
|
|
|
- obj, _, err := readBody(cli.call("GET", "/networks/"+networkID+"/endpoints", nil, nil))
|
|
|
|
if err != nil {
|
|
if err != nil {
|
|
- fmt.Fprintf(cli.err, "%s", err.Error())
|
|
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
- if *last == -1 && *nLatest {
|
|
|
|
- *last = 1
|
|
|
|
- }
|
|
|
|
|
|
|
|
- var endpointResources []endpointResource
|
|
|
|
- err = json.Unmarshal(obj, &endpointResources)
|
|
|
|
|
|
+ var serviceResources []serviceResource
|
|
|
|
+ err = json.Unmarshal(obj, &serviceResources)
|
|
if err != nil {
|
|
if err != nil {
|
|
|
|
+ fmt.Println(err)
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
|
|
wr := tabwriter.NewWriter(cli.out, 20, 1, 3, ' ', 0)
|
|
wr := tabwriter.NewWriter(cli.out, 20, 1, 3, ' ', 0)
|
|
// unless quiet (-q) is specified, print field titles
|
|
// unless quiet (-q) is specified, print field titles
|
|
if !*quiet {
|
|
if !*quiet {
|
|
- fmt.Fprintln(wr, "NETWORK SERVICE ID\tNAME\tNETWORK")
|
|
|
|
|
|
+ fmt.Fprintln(wr, "SERVICE ID\tNAME\tNETWORK\tPROVIDER")
|
|
}
|
|
}
|
|
|
|
|
|
- for _, networkResource := range endpointResources {
|
|
|
|
- ID := networkResource.ID
|
|
|
|
- netName := networkResource.Name
|
|
|
|
|
|
+ for _, sr := range serviceResources {
|
|
|
|
+ ID := sr.ID
|
|
|
|
+ bkID, err := getBackendID(cli, ID)
|
|
|
|
+ if err != nil {
|
|
|
|
+ return err
|
|
|
|
+ }
|
|
if !*noTrunc {
|
|
if !*noTrunc {
|
|
ID = stringid.TruncateID(ID)
|
|
ID = stringid.TruncateID(ID)
|
|
|
|
+ bkID = stringid.TruncateID(bkID)
|
|
}
|
|
}
|
|
- if *quiet {
|
|
|
|
|
|
+ if !*quiet {
|
|
|
|
+ fmt.Fprintf(wr, "%s\t%s\t%s\t%s\n", ID, sr.Name, sr.Network, bkID)
|
|
|
|
+ } else {
|
|
fmt.Fprintln(wr, ID)
|
|
fmt.Fprintln(wr, ID)
|
|
- continue
|
|
|
|
}
|
|
}
|
|
- network := networkResource.Network
|
|
|
|
- fmt.Fprintf(wr, "%s\t%s\t%s",
|
|
|
|
- ID,
|
|
|
|
- netName,
|
|
|
|
- network)
|
|
|
|
- fmt.Fprint(wr, "\n")
|
|
|
|
}
|
|
}
|
|
wr.Flush()
|
|
wr.Flush()
|
|
|
|
|
|
return nil
|
|
return nil
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+func getBackendID(cli *NetworkCli, servID string) (string, error) {
|
|
|
|
+ var (
|
|
|
|
+ obj []byte
|
|
|
|
+ err error
|
|
|
|
+ bk string
|
|
|
|
+ )
|
|
|
|
+
|
|
|
|
+ if obj, _, err = readBody(cli.call("GET", "/services/"+servID+"/backend", nil, nil)); err == nil {
|
|
|
|
+ var bkl []backendResource
|
|
|
|
+ if err := json.NewDecoder(bytes.NewReader(obj)).Decode(&bkl); err == nil {
|
|
|
|
+ if len(bkl) > 0 {
|
|
|
|
+ bk = bkl[0].ID
|
|
|
|
+ }
|
|
|
|
+ } else {
|
|
|
|
+ // Only print a message, don't make the caller cli fail for this
|
|
|
|
+ fmt.Fprintf(cli.out, "Failed to retrieve provider list for service %s (%v)", servID, err)
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return bk, err
|
|
|
|
+}
|
|
|
|
+
|
|
// CmdServiceInfo handles service info UI
|
|
// CmdServiceInfo handles service info UI
|
|
func (cli *NetworkCli) CmdServiceInfo(chain string, args ...string) error {
|
|
func (cli *NetworkCli) CmdServiceInfo(chain string, args ...string) error {
|
|
- cmd := cli.Subcmd(chain, "info", "SERVICE NETWORK", "Displays detailed information on a service", false)
|
|
|
|
- cmd.Require(flag.Min, 2)
|
|
|
|
|
|
+ cmd := cli.Subcmd(chain, "info", "SERVICE", "Displays detailed information about a service", false)
|
|
|
|
+ flNetwork := cmd.String([]string{"net", "-network"}, "", "Network where to publish the service")
|
|
|
|
+ cmd.Require(flag.Min, 1)
|
|
|
|
+
|
|
err := cmd.ParseFlags(args, true)
|
|
err := cmd.ParseFlags(args, true)
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
|
|
- networkID, err := lookupNetworkID(cli, cmd.Arg(1))
|
|
|
|
- if err != nil {
|
|
|
|
- return err
|
|
|
|
|
|
+ // Default network changes will come later
|
|
|
|
+ nw := "docker0"
|
|
|
|
+ if *flNetwork != "" {
|
|
|
|
+ nw = *flNetwork
|
|
}
|
|
}
|
|
|
|
|
|
- serviceID, err := lookupServiceID(cli, networkID, cmd.Arg(0))
|
|
|
|
|
|
+ serviceID, err := lookupServiceID(cli, nw, cmd.Arg(0))
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
|
|
- obj, _, err := readBody(cli.call("GET", "/networks/"+networkID+"/endpoints/"+serviceID, nil, nil))
|
|
|
|
|
|
+ obj, _, err := readBody(cli.call("GET", "/services/"+serviceID, nil, nil))
|
|
if err != nil {
|
|
if err != nil {
|
|
- fmt.Fprintf(cli.err, "%s", err.Error())
|
|
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
|
|
- endpointResource := &endpointResource{}
|
|
|
|
- if err := json.NewDecoder(bytes.NewReader(obj)).Decode(endpointResource); err != nil {
|
|
|
|
|
|
+ sr := &serviceResource{}
|
|
|
|
+ if err := json.NewDecoder(bytes.NewReader(obj)).Decode(sr); err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
- fmt.Fprintf(cli.out, "Service Id: %s\n", endpointResource.ID)
|
|
|
|
- fmt.Fprintf(cli.out, "\tName: %s\n", endpointResource.Name)
|
|
|
|
- fmt.Fprintf(cli.out, "\tNetwork: %s\n", endpointResource.Network)
|
|
|
|
|
|
+
|
|
|
|
+ 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
|
|
return nil
|
|
}
|
|
}
|
|
|
|
|
|
-// CmdServiceJoin handles service join UI
|
|
|
|
-func (cli *NetworkCli) CmdServiceJoin(chain string, args ...string) error {
|
|
|
|
- cmd := cli.Subcmd(chain, "join", "CONTAINER SERVICE NETWORK", "Sets a container as a service backend", false)
|
|
|
|
- cmd.Require(flag.Min, 3)
|
|
|
|
|
|
+// CmdServiceAttach handles service attach UI
|
|
|
|
+func (cli *NetworkCli) CmdServiceAttach(chain string, args ...string) error {
|
|
|
|
+ cmd := cli.Subcmd(chain, "attach", "CONTAINER SERVICE", "Sets a container as a service backend", false)
|
|
|
|
+ flNetwork := cmd.String([]string{"net", "-network"}, "", "Network where to publish the service")
|
|
|
|
+ cmd.Require(flag.Min, 2)
|
|
|
|
+
|
|
err := cmd.ParseFlags(args, true)
|
|
err := cmd.ParseFlags(args, true)
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
|
|
- containerID, err := lookupContainerID(cli, cmd.Arg(0))
|
|
|
|
- if err != nil {
|
|
|
|
- return err
|
|
|
|
|
|
+ // Default network changes will come later
|
|
|
|
+ nw := "docker0"
|
|
|
|
+ if *flNetwork != "" {
|
|
|
|
+ nw = *flNetwork
|
|
}
|
|
}
|
|
|
|
|
|
- networkID, err := lookupNetworkID(cli, cmd.Arg(2))
|
|
|
|
|
|
+ containerID, err := lookupContainerID(cli, cmd.Arg(0))
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
|
|
- serviceID, err := lookupServiceID(cli, networkID, cmd.Arg(1))
|
|
|
|
|
|
+ serviceID, err := lookupServiceID(cli, nw, cmd.Arg(1))
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
|
|
- nc := endpointJoin{ContainerID: containerID}
|
|
|
|
|
|
+ nc := serviceAttach{ContainerID: containerID}
|
|
|
|
|
|
- _, _, err = readBody(cli.call("POST", "/networks/"+networkID+"/endpoints/"+serviceID+"/containers", nc, nil))
|
|
|
|
- if err != nil {
|
|
|
|
- fmt.Fprintf(cli.err, "%s", err.Error())
|
|
|
|
- return err
|
|
|
|
- }
|
|
|
|
- return nil
|
|
|
|
|
|
+ _, _, err = readBody(cli.call("POST", "/services/"+serviceID+"/backend", nc, nil))
|
|
|
|
+
|
|
|
|
+ return err
|
|
}
|
|
}
|
|
|
|
|
|
-// CmdServiceLeave handles service leave UI
|
|
|
|
-func (cli *NetworkCli) CmdServiceLeave(chain string, args ...string) error {
|
|
|
|
- cmd := cli.Subcmd(chain, "leave", "CONTAINER SERVICE NETWORK", "Removes a container from service backend", false)
|
|
|
|
- cmd.Require(flag.Min, 3)
|
|
|
|
|
|
+// 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)
|
|
|
|
+ flNetwork := cmd.String([]string{"net", "-network"}, "", "Network where to publish the service")
|
|
|
|
+ cmd.Require(flag.Min, 2)
|
|
|
|
+
|
|
err := cmd.ParseFlags(args, true)
|
|
err := cmd.ParseFlags(args, true)
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
|
|
- containerID, err := lookupContainerID(cli, cmd.Arg(0))
|
|
|
|
- if err != nil {
|
|
|
|
- return err
|
|
|
|
|
|
+ // Default network changes will come later
|
|
|
|
+ nw := "docker0"
|
|
|
|
+ if *flNetwork != "" {
|
|
|
|
+ nw = *flNetwork
|
|
}
|
|
}
|
|
|
|
|
|
- networkID, err := lookupNetworkID(cli, cmd.Arg(2))
|
|
|
|
|
|
+ containerID, err := lookupContainerID(cli, cmd.Arg(0))
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
|
|
- serviceID, err := lookupServiceID(cli, networkID, cmd.Arg(1))
|
|
|
|
|
|
+ serviceID, err := lookupServiceID(cli, nw, cmd.Arg(1))
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
|
|
- _, _, err = readBody(cli.call("DELETE", "/networks/"+networkID+"/endpoints/"+serviceID+"/containers/"+containerID, nil, nil))
|
|
|
|
|
|
+ _, _, err = readBody(cli.call("DELETE", "/services/"+serviceID+"/backend/"+containerID, nil, nil))
|
|
if err != nil {
|
|
if err != nil {
|
|
- fmt.Fprintf(cli.err, "%s", err.Error())
|
|
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
return nil
|