compose/azure/aci.go

328 lines
8.8 KiB
Go
Raw Normal View History

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-05-01 15:28:44 +02:00
package azure
import (
"context"
"fmt"
"io"
2020-05-03 13:35:25 +02:00
"io/ioutil"
2020-05-01 15:28:44 +02:00
"net/http"
"strings"
"time"
2020-05-01 15:28:44 +02:00
"github.com/Azure/azure-sdk-for-go/services/containerinstance/mgmt/2018-10-01/containerinstance"
"github.com/Azure/go-autorest/autorest"
"github.com/Azure/go-autorest/autorest/to"
tm "github.com/buger/goterm"
"github.com/gobwas/ws"
"github.com/gobwas/ws/wsutil"
"github.com/pkg/errors"
"github.com/docker/api/azure/login"
"github.com/docker/api/context/store"
2020-06-17 22:19:08 +02:00
"github.com/docker/api/progress"
2020-05-01 15:28:44 +02:00
)
2020-06-15 20:39:09 +02:00
const aciDockerUserAgent = "docker-cli"
func createACIContainers(ctx context.Context, aciContext store.AciContext, groupDefinition containerinstance.ContainerGroup) error {
2020-05-04 23:50:00 +02:00
containerGroupsClient, err := getContainerGroupsClient(aciContext.SubscriptionID)
if err != nil {
return errors.Wrapf(err, "cannot get container group client")
2020-05-04 23:50:00 +02:00
}
2020-05-01 15:28:44 +02:00
// Check if the container group already exists
_, err = containerGroupsClient.Get(ctx, aciContext.ResourceGroup, *groupDefinition.Name)
if err != nil {
if err, ok := err.(autorest.DetailedError); ok {
if err.StatusCode != http.StatusNotFound {
return err
2020-05-01 15:28:44 +02:00
}
} else {
return err
2020-05-01 15:28:44 +02:00
}
} else {
return fmt.Errorf("container group %q already exists", *groupDefinition.Name)
2020-05-01 15:28:44 +02:00
}
return createOrUpdateACIContainers(ctx, aciContext, groupDefinition)
}
func createOrUpdateACIContainers(ctx context.Context, aciContext store.AciContext, groupDefinition containerinstance.ContainerGroup) error {
2020-06-17 22:19:08 +02:00
w := progress.ContextWriter(ctx)
containerGroupsClient, err := getContainerGroupsClient(aciContext.SubscriptionID)
if err != nil {
return errors.Wrapf(err, "cannot get container group client")
}
2020-06-17 22:19:08 +02:00
w.Event(progress.Event{
ID: *groupDefinition.Name,
Status: progress.Working,
StatusText: "Waiting",
})
2020-05-01 15:28:44 +02:00
future, err := containerGroupsClient.CreateOrUpdate(
ctx,
aciContext.ResourceGroup,
*groupDefinition.Name,
groupDefinition,
)
if err != nil {
return err
2020-05-01 15:28:44 +02:00
}
2020-06-17 22:19:08 +02:00
w.Event(progress.Event{
ID: *groupDefinition.Name,
Status: progress.Done,
StatusText: "Created",
})
for _, c := range *groupDefinition.Containers {
w.Event(progress.Event{
ID: *c.Name,
Status: progress.Working,
StatusText: "Waiting",
})
}
2020-05-01 15:28:44 +02:00
err = future.WaitForCompletionRef(ctx, containerGroupsClient.Client)
if err != nil {
return err
2020-05-01 15:28:44 +02:00
}
2020-06-17 22:19:08 +02:00
2020-05-01 15:28:44 +02:00
containerGroup, err := future.Result(containerGroupsClient)
if err != nil {
return err
2020-05-01 15:28:44 +02:00
}
2020-06-17 22:19:08 +02:00
for _, c := range *groupDefinition.Containers {
w.Event(progress.Event{
ID: *c.Name,
Status: progress.Done,
StatusText: "Done",
})
}
2020-05-01 15:28:44 +02:00
if len(*containerGroup.Containers) > 1 {
2020-05-01 15:28:44 +02:00
var commands []string
for _, container := range *containerGroup.Containers {
commands = append(commands, fmt.Sprintf("echo 127.0.0.1 %s >> /etc/hosts", *container.Name))
2020-05-01 15:28:44 +02:00
}
commands = append(commands, "exit")
containers := *containerGroup.Containers
container := containers[0]
2020-05-03 13:35:25 +02:00
response, err := execACIContainer(ctx, aciContext, "/bin/sh", *containerGroup.Name, *container.Name)
2020-05-01 15:28:44 +02:00
if err != nil {
return err
2020-05-01 15:28:44 +02:00
}
2020-05-03 13:35:25 +02:00
if err = execCommands(
2020-05-01 15:28:44 +02:00
ctx,
*response.WebSocketURI,
*response.Password,
commands,
2020-05-03 13:35:25 +02:00
); err != nil {
return err
2020-05-01 15:28:44 +02:00
}
}
return err
2020-05-01 15:28:44 +02:00
}
func getACIContainerGroup(ctx context.Context, aciContext store.AciContext, containerGroupName string) (containerinstance.ContainerGroup, error) {
containerGroupsClient, err := getContainerGroupsClient(aciContext.SubscriptionID)
if err != nil {
return containerinstance.ContainerGroup{}, fmt.Errorf("cannot get container group client: %v", err)
}
return containerGroupsClient.Get(ctx, aciContext.ResourceGroup, containerGroupName)
}
func deleteACIContainerGroup(ctx context.Context, aciContext store.AciContext, containerGroupName string) (containerinstance.ContainerGroup, error) {
containerGroupsClient, err := getContainerGroupsClient(aciContext.SubscriptionID)
if err != nil {
return containerinstance.ContainerGroup{}, fmt.Errorf("cannot get container group client: %v", err)
}
return containerGroupsClient.Delete(ctx, aciContext.ResourceGroup, containerGroupName)
}
2020-05-03 13:35:25 +02:00
func execACIContainer(ctx context.Context, aciContext store.AciContext, command, containerGroup string, containerName string) (c containerinstance.ContainerExecResponse, err error) {
2020-05-04 23:50:00 +02:00
containerClient, err := getContainerClient(aciContext.SubscriptionID)
if err != nil {
return c, errors.Wrapf(err, "cannot get container client")
}
2020-05-01 15:28:44 +02:00
rows, cols := getTermSize()
containerExecRequest := containerinstance.ContainerExecRequest{
Command: to.StringPtr(command),
TerminalSize: &containerinstance.ContainerExecRequestTerminalSize{
Rows: rows,
Cols: cols,
},
}
2020-05-03 13:35:25 +02:00
2020-05-01 15:28:44 +02:00
return containerClient.ExecuteCommand(
ctx,
aciContext.ResourceGroup,
containerGroup,
containerName,
containerExecRequest)
}
func getTermSize() (*int32, *int32) {
rows := tm.Height()
cols := tm.Width()
return to.Int32Ptr(int32(rows)), to.Int32Ptr(int32(cols))
}
2020-05-03 13:35:25 +02:00
type commandSender struct {
commands string
2020-05-01 15:28:44 +02:00
}
func (cs *commandSender) Read(p []byte) (int, error) {
2020-05-03 13:35:25 +02:00
if len(cs.commands) == 0 {
return 0, io.EOF
}
var command string
if len(p) >= len(cs.commands) {
command = cs.commands
cs.commands = ""
} else {
command = cs.commands[:len(p)]
cs.commands = cs.commands[len(p):]
}
2020-05-03 13:35:25 +02:00
copy(p, command)
2020-05-03 13:35:25 +02:00
return len(command), nil
}
func execCommands(ctx context.Context, address string, password string, commands []string) error {
writer := ioutil.Discard
reader := &commandSender{
commands: strings.Join(commands, "\n"),
2020-05-03 13:35:25 +02:00
}
return exec(ctx, address, password, reader, writer)
}
func exec(ctx context.Context, address string, password string, reader io.Reader, writer io.Writer) error {
conn, _, _, err := ws.DefaultDialer.Dial(ctx, address)
2020-05-01 15:28:44 +02:00
if err != nil {
return err
}
2020-05-03 13:35:25 +02:00
err = wsutil.WriteClientMessage(conn, ws.OpText, []byte(password))
2020-05-01 15:28:44 +02:00
if err != nil {
return err
}
2020-05-03 13:35:25 +02:00
downstreamChannel := make(chan error, 10)
upstreamChannel := make(chan error, 10)
2020-05-03 13:35:25 +02:00
2020-05-01 15:28:44 +02:00
go func() {
for {
msg, _, err := wsutil.ReadServerData(conn)
if err != nil {
if err == io.EOF {
downstreamChannel <- nil
return
}
downstreamChannel <- err
2020-05-01 15:28:44 +02:00
return
}
2020-05-03 13:35:25 +02:00
fmt.Fprint(writer, string(msg))
2020-05-01 15:28:44 +02:00
}
}()
2020-05-03 13:35:25 +02:00
2020-05-01 15:28:44 +02:00
go func() {
for {
2020-05-03 13:35:25 +02:00
// We send each byte, byte-per-byte over the
// websocket because the console is in raw mode
buffer := make([]byte, 1)
n, err := reader.Read(buffer)
if err != nil {
if err == io.EOF {
upstreamChannel <- nil
return
}
upstreamChannel <- err
return
2020-05-01 15:28:44 +02:00
}
2020-05-03 13:35:25 +02:00
if n > 0 {
err := wsutil.WriteClientMessage(conn, ws.OpText, buffer)
if err != nil {
upstreamChannel <- err
return
}
2020-05-03 13:35:25 +02:00
}
2020-05-01 15:28:44 +02:00
}
}()
2020-05-03 13:35:25 +02:00
2020-05-01 15:28:44 +02:00
for {
select {
case err := <-downstreamChannel:
return errors.Wrap(err, "failed to read input from container")
case err := <-upstreamChannel:
return errors.Wrap(err, "failed to send input to container")
2020-05-01 15:28:44 +02:00
}
}
}
2020-05-03 13:41:45 +02:00
func getACIContainerLogs(ctx context.Context, aciContext store.AciContext, containerGroupName, containerName string) (string, error) {
2020-05-04 23:50:00 +02:00
containerClient, err := getContainerClient(aciContext.SubscriptionID)
if err != nil {
return "", errors.Wrapf(err, "cannot get container client")
}
2020-05-03 13:41:45 +02:00
logs, err := containerClient.ListLogs(ctx, aciContext.ResourceGroup, containerGroupName, containerName, nil)
if err != nil {
return "", fmt.Errorf("cannot get container logs: %v", err)
}
return *logs.Content, err
}
2020-05-04 23:50:00 +02:00
func getContainerGroupsClient(subscriptionID string) (containerinstance.ContainerGroupsClient, error) {
containerGroupsClient := containerinstance.NewContainerGroupsClient(subscriptionID)
err := setupClient(&containerGroupsClient.Client)
2020-05-04 23:50:00 +02:00
if err != nil {
return containerinstance.ContainerGroupsClient{}, err
}
containerGroupsClient.PollingDelay = 5 * time.Second
containerGroupsClient.RetryAttempts = 30
containerGroupsClient.RetryDuration = 1 * time.Second
2020-05-04 23:50:00 +02:00
return containerGroupsClient, nil
2020-05-01 15:28:44 +02:00
}
func setupClient(aciClient *autorest.Client) error {
aciClient.UserAgent = aciDockerUserAgent
auth, err := login.NewAuthorizerFromLogin()
2020-05-04 23:50:00 +02:00
if err != nil {
return err
2020-05-04 23:50:00 +02:00
}
aciClient.Authorizer = auth
return nil
}
func getContainerClient(subscriptionID string) (containerinstance.ContainerClient, error) {
2020-05-01 15:28:44 +02:00
containerClient := containerinstance.NewContainerClient(subscriptionID)
err := setupClient(&containerClient.Client)
if err != nil {
return containerinstance.ContainerClient{}, err
}
2020-05-04 23:50:00 +02:00
return containerClient, nil
2020-05-01 15:28:44 +02:00
}