| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119 |
- /*
- Copyright (c) 2019 Docker Inc.
- Permission is hereby granted, free of charge, to any person
- obtaining a copy of this software and associated documentation
- files (the "Software"), to deal in the Software without
- restriction, including without limitation the rights to use, copy,
- modify, merge, publish, distribute, sublicense, and/or sell copies
- of the Software, and to permit persons to whom the Software is
- furnished to do so, subject to the following conditions:
- The above copyright notice and this permission notice shall be
- included in all copies or substantial portions of the Software.
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- EXPRESS OR IMPLIED,
- INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
- IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
- HOLDERS BE LIABLE FOR ANY CLAIM,
- DAMAGES OR OTHER LIABILITY,
- WHETHER IN AN ACTION OF CONTRACT,
- TORT OR OTHERWISE,
- ARISING FROM, OUT OF OR IN CONNECTION WITH
- THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
- */
- package main
- import (
- "context"
- "encoding/json"
- "fmt"
- "os"
- "os/exec"
- "path/filepath"
- "time"
- "github.com/docker/api/client"
- "github.com/gogo/protobuf/types"
- "github.com/pkg/errors"
- "github.com/urfave/cli"
- )
- func init() {
- // initial hack to get the path of the project's bin dir
- // into the env of this cli for development
- path := filepath.Join(os.Getenv("GOPATH"), "src/github.com/docker/api/bin")
- if err := os.Setenv("PATH", fmt.Sprintf("$PATH:%s", path)); err != nil {
- panic(err)
- }
- }
- var contextCommand = cli.Command{
- Name: "context",
- Usage: "manage contexts",
- Action: func(clix *cli.Context) error {
- // return information for the current context
- ctx, cancel := client.NewContext()
- defer cancel()
- // get our current context
- ctx = current(ctx)
- client, err := connect(ctx)
- if err != nil {
- return errors.Wrap(err, "cannot connect to backend")
- }
- defer client.Close()
- info, err := client.BackendInformation(ctx, &types.Empty{})
- if err != nil {
- return errors.Wrap(err, "fetch backend information")
- }
- enc := json.NewEncoder(os.Stdout)
- enc.SetIndent("", " ")
- return enc.Encode(info)
- },
- }
- // mock information for getting context
- // factor out this into a context store package
- func current(ctx context.Context) context.Context {
- // test backend address
- return context.WithValue(ctx, backendAddressKey{}, "127.0.0.1:7654")
- }
- func connect(ctx context.Context) (*client.Client, error) {
- address, err := BackendAddress(ctx)
- if err != nil {
- return nil, errors.Wrap(err, "no backend address")
- }
- c, err := client.New(address, 500*time.Millisecond)
- if err != nil {
- if err != context.DeadlineExceeded {
- return nil, errors.Wrap(err, "connect to backend")
- }
- // the backend is not running so start it
- cmd := exec.Command("backend-example", "--address", address)
- go cmd.Wait()
- if err := cmd.Start(); err != nil {
- return nil, errors.Wrap(err, "start backend")
- }
- return client.New(address, 2*time.Second)
- }
- return c, nil
- }
- type backendAddressKey struct{}
- func BackendAddress(ctx context.Context) (string, error) {
- v, ok := ctx.Value(backendAddressKey{}).(string)
- if !ok {
- return "", errors.New("no backend address key")
- }
- return v, nil
- }
|