| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414 | /*   Copyright 2020 Docker, Inc.   Licensed under the Apache License, Version 2.0 (the "License");   you may not use this file except in compliance with the License.   You may obtain a copy of the License at       http://www.apache.org/licenses/LICENSE-2.0   Unless required by applicable law or agreed to in writing, software   distributed under the License is distributed on an "AS IS" BASIS,   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.   See the License for the specific language governing permissions and   limitations under the License.*/package aciimport (	"context"	"fmt"	"net/http"	"strconv"	"strings"	"github.com/Azure/azure-sdk-for-go/services/containerinstance/mgmt/2018-10-01/containerinstance"	"github.com/Azure/go-autorest/autorest/to"	"github.com/compose-spec/compose-go/cli"	"github.com/compose-spec/compose-go/types"	"github.com/pkg/errors"	"github.com/sirupsen/logrus"	ecstypes "github.com/docker/ecs-plugin/pkg/compose"	"github.com/docker/api/aci/convert"	"github.com/docker/api/aci/login"	"github.com/docker/api/backend"	"github.com/docker/api/compose"	"github.com/docker/api/containers"	apicontext "github.com/docker/api/context"	"github.com/docker/api/context/cloud"	"github.com/docker/api/context/store"	"github.com/docker/api/errdefs")const (	backendType               = store.AciContextType	singleContainerTag        = "docker-single-container"	composeContainerTag       = "docker-compose-application"	composeContainerSeparator = "_"	statusUnknown             = "Unknown")// ErrNoSuchContainer is returned when the mentioned container does not existvar ErrNoSuchContainer = errors.New("no such container")// ContextParams options for creating ACI contexttype ContextParams struct {	Description    string	Location       string	SubscriptionID string	ResourceGroup  string}// LoginParams azure login optionstype LoginParams struct {	TenantID     string	ClientID     string	ClientSecret string}// Validate returns an error if options are not used properlyfunc (opts LoginParams) Validate() error {	if opts.ClientID != "" || opts.ClientSecret != "" {		if opts.ClientID == "" || opts.ClientSecret == "" || opts.TenantID == "" {			return errors.New("for Service Principal login, 3 options must be specified: --client-id, --client-secret and --tenant-id")		}	}	return nil}func init() {	backend.Register(backendType, backendType, service, getCloudService)}func service(ctx context.Context) (backend.Service, error) {	contextStore := store.ContextStore(ctx)	currentContext := apicontext.CurrentContext(ctx)	var aciContext store.AciContext	if err := contextStore.GetEndpoint(currentContext, &aciContext); err != nil {		return nil, err	}	return getAciAPIService(aciContext), nil}func getCloudService() (cloud.Service, error) {	service, err := login.NewAzureLoginService()	if err != nil {		return nil, err	}	return &aciCloudService{		loginService: service,	}, nil}func getAciAPIService(aciCtx store.AciContext) *aciAPIService {	return &aciAPIService{		aciContainerService: &aciContainerService{			ctx: aciCtx,		},		aciComposeService: &aciComposeService{			ctx: aciCtx,		},	}}type aciAPIService struct {	*aciContainerService	*aciComposeService}func (a *aciAPIService) ContainerService() containers.Service {	return a.aciContainerService}func (a *aciAPIService) ComposeService() compose.Service {	return a.aciComposeService}type aciContainerService struct {	ctx store.AciContext}func (cs *aciContainerService) List(ctx context.Context, _ bool) ([]containers.Container, error) {	groupsClient, err := getContainerGroupsClient(cs.ctx.SubscriptionID)	if err != nil {		return nil, err	}	var containerGroups []containerinstance.ContainerGroup	result, err := groupsClient.ListByResourceGroup(ctx, cs.ctx.ResourceGroup)	if err != nil {		return []containers.Container{}, err	}	for result.NotDone() {		containerGroups = append(containerGroups, result.Values()...)		if err := result.NextWithContext(ctx); err != nil {			return []containers.Container{}, err		}	}	var res []containers.Container	for _, containerGroup := range containerGroups {		group, err := groupsClient.Get(ctx, cs.ctx.ResourceGroup, *containerGroup.Name)		if err != nil {			return []containers.Container{}, err		}		if _, ok := group.Tags[singleContainerTag]; ok {			if group.Containers == nil || len(*group.Containers) < 1 {				return []containers.Container{}, fmt.Errorf("no containers found in ACI container group %s", *containerGroup.Name)			}			container := (*group.Containers)[0]			c := getContainer(*containerGroup.Name, group.IPAddress, container)			res = append(res, c)			continue		}		for _, container := range *group.Containers {			var containerID string			// don't list sidecar container			if *container.Name == convert.ComposeDNSSidecarName {				continue			}			containerID = *containerGroup.Name + composeContainerSeparator + *container.Name			c := getContainer(containerID, group.IPAddress, container)			res = append(res, c)		}	}	return res, nil}func getContainer(containerID string, ipAddress *containerinstance.IPAddress, container containerinstance.Container) containers.Container {	status := statusUnknown	if container.InstanceView != nil && container.InstanceView.CurrentState != nil {		status = *container.InstanceView.CurrentState.State	}	return containers.Container{		ID:     containerID,		Image:  *container.Image,		Status: status,		Ports:  convert.ToPorts(ipAddress, *container.Ports),	}}func (cs *aciContainerService) Run(ctx context.Context, r containers.ContainerConfig) error {	if strings.Contains(r.ID, composeContainerSeparator) {		return errors.New(fmt.Sprintf("invalid container name. ACI container name cannot include %q", composeContainerSeparator))	}	project, err := convert.ContainerToComposeProject(r)	if err != nil {		return err	}	logrus.Debugf("Running container %q with name %q\n", r.Image, r.ID)	groupDefinition, err := convert.ToContainerGroup(cs.ctx, project)	if err != nil {		return err	}	addTag(&groupDefinition, singleContainerTag)	return createACIContainers(ctx, cs.ctx, groupDefinition)}func addTag(groupDefinition *containerinstance.ContainerGroup, tagName string) {	if groupDefinition.Tags == nil {		groupDefinition.Tags = make(map[string]*string, 1)	}	groupDefinition.Tags[tagName] = to.StringPtr(tagName)}func (cs *aciContainerService) Stop(ctx context.Context, containerName string, timeout *uint32) error {	return errdefs.ErrNotImplemented}func getGroupAndContainerName(containerID string) (string, string) {	tokens := strings.Split(containerID, composeContainerSeparator)	groupName := tokens[0]	containerName := groupName	if len(tokens) > 1 {		containerName = tokens[len(tokens)-1]		groupName = containerID[:len(containerID)-(len(containerName)+1)]	}	return groupName, containerName}func (cs *aciContainerService) Exec(ctx context.Context, name string, request containers.ExecRequest) error {	err := verifyExecCommand(request.Command)	if err != nil {		return err	}	groupName, containerAciName := getGroupAndContainerName(name)	containerExecResponse, err := execACIContainer(ctx, cs.ctx, request.Command, groupName, containerAciName)	if err != nil {		return err	}	return exec(		context.Background(),		*containerExecResponse.WebSocketURI,		*containerExecResponse.Password,		request,	)}func verifyExecCommand(command string) error {	tokens := strings.Split(command, " ")	if len(tokens) > 1 {		return errors.New("ACI exec command does not accept arguments to the command. " +			"Only the binary should be specified")	}	return nil}func (cs *aciContainerService) Logs(ctx context.Context, containerName string, req containers.LogsRequest) error {	groupName, containerAciName := getGroupAndContainerName(containerName)	var tail *int32	if req.Follow {		return streamLogs(ctx, cs.ctx, groupName, containerAciName, req)	}	if req.Tail != "all" {		reqTail, err := strconv.Atoi(req.Tail)		if err != nil {			return err		}		i32 := int32(reqTail)		tail = &i32	}	logs, err := getACIContainerLogs(ctx, cs.ctx, groupName, containerAciName, tail)	if err != nil {		return err	}	_, err = fmt.Fprint(req.Writer, logs)	return err}func (cs *aciContainerService) Delete(ctx context.Context, containerID string, _ bool) error {	groupName, containerName := getGroupAndContainerName(containerID)	if groupName != containerID {		msg := "cannot delete service %q from compose application %q, you can delete the entire compose app with docker compose down --project-name %s"		return errors.New(fmt.Sprintf(msg, containerName, groupName, groupName))	}	cg, err := deleteACIContainerGroup(ctx, cs.ctx, groupName)	if err != nil {		return err	}	if cg.StatusCode == http.StatusNoContent {		return ErrNoSuchContainer	}	return err}func (cs *aciContainerService) Inspect(ctx context.Context, containerID string) (containers.Container, error) {	groupName, containerName := getGroupAndContainerName(containerID)	cg, err := getACIContainerGroup(ctx, cs.ctx, groupName)	if err != nil {		return containers.Container{}, err	}	if cg.StatusCode == http.StatusNoContent {		return containers.Container{}, ErrNoSuchContainer	}	var cc containerinstance.Container	var found = false	for _, c := range *cg.Containers {		if to.String(c.Name) == containerName {			cc = c			found = true			break		}	}	if !found {		return containers.Container{}, ErrNoSuchContainer	}	return convert.ContainerGroupToContainer(containerID, cg, cc)}type aciComposeService struct {	ctx store.AciContext}func (cs *aciComposeService) Up(ctx context.Context, opts cli.ProjectOptions) error {	project, err := cli.ProjectFromOptions(&opts)	if err != nil {		return err	}	logrus.Debugf("Up on project with name %q\n", project.Name)	groupDefinition, err := convert.ToContainerGroup(cs.ctx, *project)	addTag(&groupDefinition, composeContainerTag)	if err != nil {		return err	}	return createOrUpdateACIContainers(ctx, cs.ctx, groupDefinition)}func (cs *aciComposeService) Down(ctx context.Context, opts cli.ProjectOptions) error {	var project types.Project	if opts.Name != "" {		project = types.Project{Name: opts.Name}	} else {		fullProject, err := cli.ProjectFromOptions(&opts)		if err != nil {			return err		}		project = *fullProject	}	logrus.Debugf("Down on project with name %q\n", project.Name)	cg, err := deleteACIContainerGroup(ctx, cs.ctx, project.Name)	if err != nil {		return err	}	if cg.StatusCode == http.StatusNoContent {		return ErrNoSuchContainer	}	return err}func (cs *aciComposeService) Ps(ctx context.Context, opts cli.ProjectOptions) ([]ecstypes.ServiceStatus, error) {	return nil, errdefs.ErrNotImplemented}func (cs *aciComposeService) Logs(ctx context.Context, opts cli.ProjectOptions) error {	return errdefs.ErrNotImplemented}type aciCloudService struct {	loginService login.AzureLoginServiceAPI}func (cs *aciCloudService) Login(ctx context.Context, params interface{}) error {	opts, ok := params.(LoginParams)	if !ok {		return errors.New("Could not read azure LoginParams struct from generic parameter")	}	if opts.ClientID != "" {		return cs.loginService.LoginServicePrincipal(opts.ClientID, opts.ClientSecret, opts.TenantID)	}	return cs.loginService.Login(ctx, opts.TenantID)}func (cs *aciCloudService) Logout(ctx context.Context) error {	return cs.loginService.Logout(ctx)}func (cs *aciCloudService) CreateContextData(ctx context.Context, params interface{}) (interface{}, string, error) {	contextHelper := newContextCreateHelper()	createOpts := params.(ContextParams)	return contextHelper.createContextData(ctx, createOpts)}
 |