2020-06-18 16:13:24 +02:00
/ *
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 .
* /
2020-07-29 15:12:45 +02:00
package aci
2020-04-29 19:57:53 +02:00
import (
"context"
2020-05-03 13:41:45 +02:00
"fmt"
2020-05-10 22:37:28 +02:00
"net/http"
2020-05-04 16:38:02 +02:00
"strconv"
"strings"
2020-04-29 19:57:53 +02:00
"github.com/Azure/azure-sdk-for-go/services/containerinstance/mgmt/2018-10-01/containerinstance"
2020-08-12 10:35:15 +02:00
"github.com/Azure/go-autorest/autorest"
2020-06-30 12:23:22 +02:00
"github.com/Azure/go-autorest/autorest/to"
2020-07-02 16:05:45 +02:00
"github.com/compose-spec/compose-go/cli"
2020-05-01 15:28:44 +02:00
"github.com/compose-spec/compose-go/types"
2020-04-29 19:57:53 +02:00
"github.com/pkg/errors"
2020-05-01 15:28:44 +02:00
"github.com/sirupsen/logrus"
2020-08-10 14:36:05 +02:00
ecstypes "github.com/docker/ecs-plugin/pkg/compose"
2020-07-29 15:12:45 +02:00
"github.com/docker/api/aci/convert"
"github.com/docker/api/aci/login"
2020-04-29 19:57:53 +02:00
"github.com/docker/api/backend"
2020-05-01 15:28:44 +02:00
"github.com/docker/api/compose"
2020-04-29 19:57:53 +02:00
"github.com/docker/api/containers"
apicontext "github.com/docker/api/context"
2020-06-30 12:23:22 +02:00
"github.com/docker/api/context/cloud"
2020-05-01 15:48:20 +02:00
"github.com/docker/api/context/store"
2020-06-30 12:23:22 +02:00
"github.com/docker/api/errdefs"
2020-04-29 19:57:53 +02:00
)
2020-07-03 10:53:50 +02:00
const (
2020-07-29 16:05:13 +02:00
backendType = store . AciContextType
2020-07-10 14:27:55 +02:00
singleContainerTag = "docker-single-container"
composeContainerTag = "docker-compose-application"
2020-07-03 10:53:50 +02:00
composeContainerSeparator = "_"
2020-07-08 23:31:01 +02:00
statusUnknown = "Unknown"
2020-08-11 15:46:29 +02:00
statusRunning = "Running"
2020-07-03 10:53:50 +02:00
)
2020-05-06 17:14:53 +02:00
2020-07-13 10:38:29 +02:00
// ContextParams options for creating ACI context
type ContextParams struct {
Description string
Location string
SubscriptionID string
ResourceGroup string
}
// LoginParams azure login options
type LoginParams struct {
2020-08-10 16:38:59 +02:00
TenantID string
ClientID string
ClientSecret string
}
// Validate returns an error if options are not used properly
func ( 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
2020-07-13 10:38:29 +02:00
}
2020-04-29 19:57:53 +02:00
func init ( ) {
2020-07-29 15:12:45 +02:00
backend . Register ( backendType , backendType , service , getCloudService )
2020-04-29 19:57:53 +02:00
}
2020-05-28 17:37:59 +02:00
func service ( ctx context . Context ) ( backend . Service , error ) {
2020-05-20 17:15:56 +02:00
contextStore := store . ContextStore ( ctx )
2020-05-28 17:37:59 +02:00
currentContext := apicontext . CurrentContext ( ctx )
Change the way a context is stored
Initially we stored the context data in the `Metadata` of the context
but in hindsight this data would be better of in the `Endpoints` because
that's what it is used for.
Before:
```json
{
"Name": "aci",
"Metadata": {
"Type": "aci",
"Data": {
"key": "value"
}
},
"Endpoints": {
"docker": {}
}
}
```
After:
```json
{
"Name": "aci",
"Type": "aci",
"Metadata": {},
"Endpoints": {
"aci": {
"key": "value"
},
"docker": {}
}
}
```
With this change the contexts that we create are more in line with the contexts the docker cli creates.
It also makes the code less complicated since we don't need to marsal twice any more. The API is nicer too:
```go
// Get a context:
c, err := store.Get(contextName)
// Get the stored endpoint:
var aciContext store.AciContext
if err := contextStore.GetEndpoint(currentContext, &aciContext); err != nil {
return nil, err
}
```
2020-05-22 11:16:01 +02:00
var aciContext store . AciContext
2020-05-28 17:37:59 +02:00
Change the way a context is stored
Initially we stored the context data in the `Metadata` of the context
but in hindsight this data would be better of in the `Endpoints` because
that's what it is used for.
Before:
```json
{
"Name": "aci",
"Metadata": {
"Type": "aci",
"Data": {
"key": "value"
}
},
"Endpoints": {
"docker": {}
}
}
```
After:
```json
{
"Name": "aci",
"Type": "aci",
"Metadata": {},
"Endpoints": {
"aci": {
"key": "value"
},
"docker": {}
}
}
```
With this change the contexts that we create are more in line with the contexts the docker cli creates.
It also makes the code less complicated since we don't need to marsal twice any more. The API is nicer too:
```go
// Get a context:
c, err := store.Get(contextName)
// Get the stored endpoint:
var aciContext store.AciContext
if err := contextStore.GetEndpoint(currentContext, &aciContext); err != nil {
return nil, err
}
```
2020-05-22 11:16:01 +02:00
if err := contextStore . GetEndpoint ( currentContext , & aciContext ) ; err != nil {
return nil , err
2020-04-29 19:57:53 +02:00
}
2020-05-28 17:37:59 +02:00
return getAciAPIService ( aciContext ) , nil
2020-05-05 15:37:12 +02:00
}
2020-05-28 17:37:59 +02:00
func getCloudService ( ) ( cloud . Service , error ) {
2020-05-13 23:33:16 +02:00
service , err := login . NewAzureLoginService ( )
if err != nil {
return nil , err
}
2020-05-28 17:37:59 +02:00
return & aciCloudService {
loginService : service ,
} , nil
}
func getAciAPIService ( aciCtx store . AciContext ) * aciAPIService {
2020-05-05 16:27:22 +02:00
return & aciAPIService {
2020-05-28 17:37:59 +02:00
aciContainerService : & aciContainerService {
2020-05-28 10:16:42 +02:00
ctx : aciCtx ,
2020-05-05 15:37:12 +02:00
} ,
2020-05-28 17:37:59 +02:00
aciComposeService : & aciComposeService {
2020-05-12 23:00:58 +02:00
ctx : aciCtx ,
2020-05-05 15:37:12 +02:00
} ,
2020-05-28 17:37:59 +02:00
}
2020-05-05 15:37:12 +02:00
}
2020-05-05 16:27:22 +02:00
type aciAPIService struct {
2020-05-28 17:37:59 +02:00
* aciContainerService
* aciComposeService
2020-04-29 19:57:53 +02:00
}
2020-05-05 16:27:22 +02:00
func ( a * aciAPIService ) ContainerService ( ) containers . Service {
2020-05-28 17:37:59 +02:00
return a . aciContainerService
2020-05-05 15:37:12 +02:00
}
2020-05-05 16:27:22 +02:00
func ( a * aciAPIService ) ComposeService ( ) compose . Service {
2020-05-28 17:37:59 +02:00
return a . aciComposeService
2020-05-12 13:37:28 +02:00
}
2020-05-05 15:37:12 +02:00
type aciContainerService struct {
2020-05-28 10:16:42 +02:00
ctx store . AciContext
2020-05-05 15:37:12 +02:00
}
2020-08-12 11:59:36 +02:00
func ( cs * aciContainerService ) List ( ctx context . Context , all bool ) ( [ ] containers . Container , error ) {
2020-05-28 10:16:42 +02:00
groupsClient , err := getContainerGroupsClient ( cs . ctx . SubscriptionID )
if err != nil {
return nil , err
}
2020-05-04 09:57:52 +02:00
var containerGroups [ ] containerinstance . ContainerGroup
2020-05-28 10:16:42 +02:00
result , err := groupsClient . ListByResourceGroup ( ctx , cs . ctx . ResourceGroup )
2020-04-29 19:57:53 +02:00
if err != nil {
return [ ] containers . Container { } , err
}
for result . NotDone ( ) {
2020-05-04 09:57:52 +02:00
containerGroups = append ( containerGroups , result . Values ( ) ... )
2020-04-29 19:57:53 +02:00
if err := result . NextWithContext ( ctx ) ; err != nil {
return [ ] containers . Container { } , err
}
}
2020-05-05 10:58:24 +02:00
var res [ ] containers . Container
2020-05-04 09:57:52 +02:00
for _ , containerGroup := range containerGroups {
2020-05-28 10:16:42 +02:00
group , err := groupsClient . Get ( ctx , cs . ctx . ResourceGroup , * containerGroup . Name )
2020-05-03 14:51:03 +02:00
if err != nil {
return [ ] containers . Container { } , err
}
2020-08-12 11:59:36 +02:00
if group . Containers == nil || len ( * group . Containers ) < 1 {
return [ ] containers . Container { } , fmt . Errorf ( "no containers found in ACI container group %s" , * containerGroup . Name )
2020-07-08 23:31:01 +02:00
}
2020-05-04 09:57:52 +02:00
for _ , container := range * group . Containers {
2020-06-18 15:51:13 +02:00
// don't list sidecar container
if * container . Name == convert . ComposeDNSSidecarName {
continue
}
2020-08-12 11:59:36 +02:00
if ! all && getStatus ( container ) != statusRunning {
continue
}
containerID := * containerGroup . Name + composeContainerSeparator + * container . Name
if _ , ok := group . Tags [ singleContainerTag ] ; ok {
containerID = * containerGroup . Name
}
2020-07-08 23:31:01 +02:00
c := getContainer ( containerID , group . IPAddress , container )
res = append ( res , c )
2020-04-29 19:57:53 +02:00
}
}
return res , nil
}
2020-05-01 15:28:44 +02:00
2020-07-08 23:31:01 +02:00
func getContainer ( containerID string , ipAddress * containerinstance . IPAddress , container containerinstance . Container ) containers . Container {
2020-08-12 11:59:36 +02:00
status := getStatus ( container )
2020-07-08 23:31:01 +02:00
return containers . Container {
ID : containerID ,
Image : * container . Image ,
Status : status ,
Ports : convert . ToPorts ( ipAddress , * container . Ports ) ,
}
}
2020-08-12 11:59:36 +02:00
func getStatus ( container containerinstance . Container ) string {
status := statusUnknown
if container . InstanceView != nil && container . InstanceView . CurrentState != nil {
status = * container . InstanceView . CurrentState . State
}
return status
}
2020-05-05 15:37:12 +02:00
func ( cs * aciContainerService ) Run ( ctx context . Context , r containers . ContainerConfig ) error {
2020-07-03 10:53:50 +02:00
if strings . Contains ( r . ID , composeContainerSeparator ) {
2020-07-03 15:39:44 +02:00
return errors . New ( fmt . Sprintf ( "invalid container name. ACI container name cannot include %q" , composeContainerSeparator ) )
2020-07-03 10:53:50 +02:00
}
2020-07-08 23:31:01 +02:00
project , err := convert . ContainerToComposeProject ( r )
2020-05-07 04:58:04 +02:00
if err != nil {
return err
}
2020-05-01 15:28:44 +02:00
logrus . Debugf ( "Running container %q with name %q\n" , r . Image , r . ID )
2020-05-01 15:48:20 +02:00
groupDefinition , err := convert . ToContainerGroup ( cs . ctx , project )
if err != nil {
return err
}
2020-07-16 10:01:49 +02:00
addTag ( & groupDefinition , singleContainerTag )
2020-05-01 15:48:20 +02:00
2020-07-10 14:27:55 +02:00
return createACIContainers ( ctx , cs . ctx , groupDefinition )
}
2020-07-16 10:01:49 +02:00
func addTag ( groupDefinition * containerinstance . ContainerGroup , tagName string ) {
2020-07-08 23:31:01 +02:00
if groupDefinition . Tags == nil {
groupDefinition . Tags = make ( map [ string ] * string , 1 )
}
2020-07-16 10:01:49 +02:00
groupDefinition . Tags [ tagName ] = to . StringPtr ( tagName )
2020-05-01 15:28:44 +02:00
}
2020-05-03 13:35:25 +02:00
2020-08-12 10:35:15 +02:00
func ( cs * aciContainerService ) Start ( ctx context . Context , containerID string ) error {
groupName , containerName := getGroupAndContainerName ( containerID )
if groupName != containerID {
2020-08-12 15:10:54 +02:00
msg := "cannot start specified service %q from compose application %q, you can update and restart the entire compose app with docker compose up --project-name %s"
2020-08-12 10:35:15 +02:00
return errors . New ( fmt . Sprintf ( msg , containerName , groupName , groupName ) )
}
containerGroupsClient , err := getContainerGroupsClient ( cs . ctx . SubscriptionID )
if err != nil {
return err
}
future , err := containerGroupsClient . Start ( ctx , cs . ctx . ResourceGroup , containerName )
if err != nil {
var aerr autorest . DetailedError
if ok := errors . As ( err , & aerr ) ; ok {
if aerr . StatusCode == http . StatusNotFound {
return errdefs . ErrNotFound
}
}
return err
}
return future . WaitForCompletionRef ( ctx , containerGroupsClient . Client )
}
2020-08-11 17:49:02 +02:00
func ( cs * aciContainerService ) Stop ( ctx context . Context , containerID string , timeout * uint32 ) error {
if timeout != nil && * timeout != uint32 ( 0 ) {
return errors . Errorf ( "ACI integration does not support setting a timeout to stop a container before killing it." )
}
groupName , containerName := getGroupAndContainerName ( containerID )
if groupName != containerID {
msg := "cannot stop service %q from compose application %q, you can stop the entire compose app with docker stop %s"
return errors . New ( fmt . Sprintf ( msg , containerName , groupName , groupName ) )
}
return stopACIContainerGroup ( ctx , cs . ctx , groupName )
2020-05-16 12:13:51 +02:00
}
2020-07-08 23:31:01 +02:00
func getGroupAndContainerName ( containerID string ) ( string , string ) {
2020-07-03 10:53:50 +02:00
tokens := strings . Split ( containerID , composeContainerSeparator )
2020-07-08 23:31:01 +02:00
groupName := tokens [ 0 ]
containerName := groupName
2020-05-06 17:14:53 +02:00
if len ( tokens ) > 1 {
containerName = tokens [ len ( tokens ) - 1 ]
groupName = containerID [ : len ( containerID ) - ( len ( containerName ) + 1 ) ]
}
return groupName , containerName
}
2020-07-08 14:31:27 +02:00
func ( cs * aciContainerService ) Exec ( ctx context . Context , name string , request containers . ExecRequest ) error {
err := verifyExecCommand ( request . Command )
2020-07-06 03:29:48 +02:00
if err != nil {
return err
}
2020-05-18 14:56:32 +02:00
groupName , containerAciName := getGroupAndContainerName ( name )
2020-07-08 14:31:27 +02:00
containerExecResponse , err := execACIContainer ( ctx , cs . ctx , request . Command , groupName , containerAciName )
2020-05-03 13:35:25 +02:00
if err != nil {
return err
}
return exec (
context . Background ( ) ,
* containerExecResponse . WebSocketURI ,
* containerExecResponse . Password ,
2020-07-08 14:31:27 +02:00
request ,
2020-05-03 13:35:25 +02:00
)
}
2020-05-03 13:41:45 +02:00
2020-07-06 03:29:48 +02:00
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
}
2020-05-05 15:37:12 +02:00
func ( cs * aciContainerService ) Logs ( ctx context . Context , containerName string , req containers . LogsRequest ) error {
2020-05-18 14:56:32 +02:00
groupName , containerAciName := getGroupAndContainerName ( containerName )
2020-06-29 15:40:58 +02:00
var tail * int32
2020-06-30 12:23:22 +02:00
if req . Follow {
2020-07-09 10:50:48 +02:00
return streamLogs ( ctx , cs . ctx , groupName , containerAciName , req )
2020-06-30 12:23:22 +02:00
}
2020-05-04 16:38:02 +02:00
if req . Tail != "all" {
2020-06-29 15:40:58 +02:00
reqTail , err := strconv . Atoi ( req . Tail )
2020-05-04 16:38:02 +02:00
if err != nil {
return err
}
2020-06-29 15:40:58 +02:00
i32 := int32 ( reqTail )
tail = & i32
}
2020-05-04 16:38:02 +02:00
2020-06-29 15:40:58 +02:00
logs , err := getACIContainerLogs ( ctx , cs . ctx , groupName , containerAciName , tail )
if err != nil {
return err
2020-05-04 16:38:02 +02:00
}
_ , err = fmt . Fprint ( req . Writer , logs )
2020-05-03 13:41:45 +02:00
return err
}
2020-05-05 10:58:24 +02:00
2020-08-11 15:46:29 +02:00
func ( cs * aciContainerService ) Delete ( ctx context . Context , containerID string , request containers . DeleteRequest ) error {
2020-07-03 10:03:46 +02:00
groupName , containerName := getGroupAndContainerName ( containerID )
if groupName != containerID {
2020-07-03 15:55:51 +02:00
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 ) )
2020-07-03 10:03:46 +02:00
}
2020-08-11 15:46:29 +02:00
if ! request . Force {
2020-08-12 10:35:15 +02:00
containerGroupsClient , err := getContainerGroupsClient ( cs . ctx . SubscriptionID )
if err != nil {
return err
}
2020-08-11 15:46:29 +02:00
cg , err := containerGroupsClient . Get ( ctx , cs . ctx . ResourceGroup , groupName )
if err != nil {
if cg . StatusCode == http . StatusNotFound {
return errdefs . ErrNotFound
}
return err
}
for _ , container := range * cg . Containers {
status := statusUnknown
if container . InstanceView != nil && container . InstanceView . CurrentState != nil {
status = * container . InstanceView . CurrentState . State
}
if status == statusRunning {
return errdefs . ErrForbidden
}
}
}
cg , err := deleteACIContainerGroup ( ctx , cs . ctx , groupName )
// Delete returns `StatusNoContent` if the group is not found
2020-05-10 22:37:28 +02:00
if cg . StatusCode == http . StatusNoContent {
2020-08-11 15:46:29 +02:00
return errdefs . ErrNotFound
}
if err != nil {
return err
2020-05-10 22:37:28 +02:00
}
return err
}
2020-06-12 10:53:06 +02:00
func ( cs * aciContainerService ) Inspect ( ctx context . Context , containerID string ) ( containers . Container , error ) {
2020-06-15 10:38:37 +02:00
groupName , containerName := getGroupAndContainerName ( containerID )
cg , err := getACIContainerGroup ( ctx , cs . ctx , groupName )
2020-06-12 10:53:06 +02:00
if err != nil {
return containers . Container { } , err
}
if cg . StatusCode == http . StatusNoContent {
2020-08-11 15:46:29 +02:00
return containers . Container { } , errdefs . ErrNotFound
2020-06-12 10:53:06 +02:00
}
2020-06-15 10:38:37 +02:00
var cc containerinstance . Container
var found = false
for _ , c := range * cg . Containers {
if to . String ( c . Name ) == containerName {
cc = c
found = true
break
}
2020-06-12 10:53:06 +02:00
}
2020-06-15 10:38:37 +02:00
if ! found {
2020-08-11 15:46:29 +02:00
return containers . Container { } , errdefs . ErrNotFound
2020-06-12 10:53:06 +02:00
}
2020-06-15 10:38:37 +02:00
return convert . ContainerGroupToContainer ( containerID , cg , cc )
2020-06-12 10:53:06 +02:00
}
2020-05-05 15:37:12 +02:00
type aciComposeService struct {
2020-05-12 23:00:58 +02:00
ctx store . AciContext
2020-05-05 15:37:12 +02:00
}
2020-07-02 16:05:45 +02:00
func ( cs * aciComposeService ) Up ( ctx context . Context , opts cli . ProjectOptions ) error {
project , err := cli . ProjectFromOptions ( & opts )
2020-05-05 10:58:24 +02:00
if err != nil {
return err
}
logrus . Debugf ( "Up on project with name %q\n" , project . Name )
groupDefinition , err := convert . ToContainerGroup ( cs . ctx , * project )
2020-07-16 10:01:49 +02:00
addTag ( & groupDefinition , composeContainerTag )
2020-05-06 17:14:53 +02:00
2020-05-05 10:58:24 +02:00
if err != nil {
return err
}
2020-06-18 10:03:28 +02:00
return createOrUpdateACIContainers ( ctx , cs . ctx , groupDefinition )
2020-05-05 10:58:24 +02:00
}
2020-07-02 16:05:45 +02:00
func ( cs * aciComposeService ) Down ( ctx context . Context , opts cli . ProjectOptions ) error {
2020-07-03 10:24:06 +02:00
var project types . Project
if opts . Name != "" {
2020-07-03 10:53:50 +02:00
project = types . Project { Name : opts . Name }
2020-07-03 10:24:06 +02:00
} else {
fullProject , err := cli . ProjectFromOptions ( & opts )
if err != nil {
return err
}
project = * fullProject
2020-05-05 10:58:24 +02:00
}
logrus . Debugf ( "Down on project with name %q\n" , project . Name )
2020-05-10 22:37:28 +02:00
cg , err := deleteACIContainerGroup ( ctx , cs . ctx , project . Name )
if err != nil {
return err
}
if cg . StatusCode == http . StatusNoContent {
2020-08-11 15:46:29 +02:00
return errdefs . ErrNotFound
2020-05-10 22:37:28 +02:00
}
2020-05-05 10:58:24 +02:00
return err
}
2020-05-12 13:37:28 +02:00
2020-08-05 16:32:51 +02:00
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
}
2020-05-12 13:37:28 +02:00
type aciCloudService struct {
2020-08-10 16:38:59 +02:00
loginService login . AzureLoginServiceAPI
2020-05-12 13:37:28 +02:00
}
2020-07-10 16:55:44 +02:00
func ( cs * aciCloudService ) Login ( ctx context . Context , params interface { } ) error {
2020-08-10 16:38:59 +02:00
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 )
2020-05-12 13:37:28 +02:00
}
2020-06-02 09:49:30 +02:00
2020-07-08 16:01:54 +02:00
func ( cs * aciCloudService ) Logout ( ctx context . Context ) error {
return cs . loginService . Logout ( ctx )
}
2020-07-10 16:39:09 +02:00
func ( cs * aciCloudService ) CreateContextData ( ctx context . Context , params interface { } ) ( interface { } , string , error ) {
2020-06-02 23:33:41 +02:00
contextHelper := newContextCreateHelper ( )
2020-07-13 10:38:29 +02:00
createOpts := params . ( ContextParams )
2020-07-10 16:39:09 +02:00
return contextHelper . createContextData ( ctx , createOpts )
2020-06-02 09:49:30 +02:00
}