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-05 17:55:53 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2020-05-19 11:29:48 +02:00
|
|
|
"fmt"
|
|
|
|
"net/url"
|
2020-06-29 17:57:06 +02:00
|
|
|
"os"
|
2020-05-18 11:00:09 +02:00
|
|
|
"strings"
|
2020-05-20 15:57:10 +02:00
|
|
|
"testing"
|
2020-05-05 17:55:53 +02:00
|
|
|
|
2020-06-02 23:33:41 +02:00
|
|
|
"github.com/docker/api/azure"
|
2020-06-29 17:57:06 +02:00
|
|
|
"github.com/docker/api/azure/login"
|
2020-06-02 23:33:41 +02:00
|
|
|
|
2020-05-05 17:55:53 +02:00
|
|
|
"github.com/Azure/azure-sdk-for-go/profiles/2019-03-01/resources/mgmt/resources"
|
2020-05-19 11:29:48 +02:00
|
|
|
azure_storage "github.com/Azure/azure-sdk-for-go/profiles/2019-03-01/storage/mgmt/storage"
|
|
|
|
"github.com/Azure/azure-storage-file-go/azfile"
|
2020-05-20 18:05:32 +02:00
|
|
|
"github.com/Azure/go-autorest/autorest/to"
|
2020-05-13 07:52:43 +02:00
|
|
|
. "github.com/onsi/gomega"
|
2020-05-20 18:05:32 +02:00
|
|
|
log "github.com/sirupsen/logrus"
|
2020-05-20 15:57:10 +02:00
|
|
|
"github.com/stretchr/testify/suite"
|
2020-05-13 07:52:43 +02:00
|
|
|
|
2020-05-19 11:29:48 +02:00
|
|
|
"github.com/docker/api/context/store"
|
2020-05-19 15:26:59 +02:00
|
|
|
"github.com/docker/api/tests/aci-e2e/storage"
|
2020-05-05 17:55:53 +02:00
|
|
|
. "github.com/docker/api/tests/framework"
|
|
|
|
)
|
|
|
|
|
2020-05-13 08:54:48 +02:00
|
|
|
const (
|
|
|
|
resourceGroupName = "resourceGroupTest"
|
|
|
|
location = "westeurope"
|
|
|
|
contextName = "acitest"
|
2020-05-13 17:09:18 +02:00
|
|
|
|
|
|
|
testContainerName = "testcontainername"
|
2020-05-13 08:54:48 +02:00
|
|
|
)
|
2020-05-05 17:55:53 +02:00
|
|
|
|
2020-05-20 15:57:10 +02:00
|
|
|
var (
|
|
|
|
subscriptionID string
|
|
|
|
)
|
|
|
|
|
|
|
|
type E2eACISuite struct {
|
|
|
|
Suite
|
|
|
|
}
|
2020-05-05 17:55:53 +02:00
|
|
|
|
2020-05-20 15:57:10 +02:00
|
|
|
func (s *E2eACISuite) TestContextDefault() {
|
2020-05-05 17:55:53 +02:00
|
|
|
It("should be initialized with default context", func() {
|
2020-05-20 15:57:10 +02:00
|
|
|
_, err := s.NewCommand("docker", "context", "rm", "-f", contextName).Exec()
|
2020-05-06 15:28:03 +02:00
|
|
|
if err == nil {
|
|
|
|
log.Println("Cleaning existing test context")
|
|
|
|
}
|
|
|
|
|
2020-05-20 15:57:10 +02:00
|
|
|
s.NewCommand("docker", "context", "use", "default").ExecOrDie()
|
|
|
|
output := s.NewCommand("docker", "context", "ls").ExecOrDie()
|
2020-05-05 17:55:53 +02:00
|
|
|
Expect(output).To(Not(ContainSubstring(contextName)))
|
|
|
|
Expect(output).To(ContainSubstring("default *"))
|
|
|
|
})
|
2020-05-20 15:57:10 +02:00
|
|
|
}
|
2020-05-05 17:55:53 +02:00
|
|
|
|
2020-05-20 15:57:10 +02:00
|
|
|
func (s *E2eACISuite) TestACIBackend() {
|
2020-06-29 17:57:06 +02:00
|
|
|
|
|
|
|
It("Logs in azure using service principal credentials", func() {
|
|
|
|
login, err := login.NewAzureLoginService()
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
// in order to create new service principal and get these 3 values : `az ad sp create-for-rbac --name 'TestServicePrincipal' --sdk-auth`
|
|
|
|
clientID := os.Getenv("AZURE_CLIENT_ID")
|
|
|
|
clientSecret := os.Getenv("AZURE_CLIENT_SECRET")
|
|
|
|
tenantID := os.Getenv("AZURE_TENANT_ID")
|
|
|
|
err = login.LoginFromServicePrincipal(clientID, clientSecret, tenantID)
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
})
|
|
|
|
|
2020-05-05 17:55:53 +02:00
|
|
|
It("creates a new aci context for tests", func() {
|
2020-05-20 06:45:06 +02:00
|
|
|
setupTestResourceGroup(resourceGroupName)
|
2020-06-02 23:33:41 +02:00
|
|
|
helper := azure.NewACIResourceGroupHelper()
|
|
|
|
models, err := helper.GetSubscriptionIDs(context.TODO())
|
2020-05-05 17:55:53 +02:00
|
|
|
Expect(err).To(BeNil())
|
2020-06-02 09:49:30 +02:00
|
|
|
subscriptionID = *models[0].SubscriptionID
|
2020-05-05 17:55:53 +02:00
|
|
|
|
2020-06-11 10:12:41 +02:00
|
|
|
s.NewDockerCommand("context", "create", "aci", contextName, "--subscription-id", subscriptionID, "--resource-group", resourceGroupName, "--location", location).ExecOrDie()
|
2020-05-13 08:54:48 +02:00
|
|
|
// Expect(output).To(ContainSubstring("ACI context acitest created"))
|
2020-05-05 17:55:53 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
defer deleteResourceGroup(resourceGroupName)
|
|
|
|
|
|
|
|
It("uses the aci context", func() {
|
2020-05-20 15:57:10 +02:00
|
|
|
currentContext := s.NewCommand("docker", "context", "use", contextName).ExecOrDie()
|
2020-05-05 17:55:53 +02:00
|
|
|
Expect(currentContext).To(ContainSubstring(contextName))
|
2020-05-20 15:57:10 +02:00
|
|
|
output := s.NewCommand("docker", "context", "ls").ExecOrDie()
|
2020-05-05 17:55:53 +02:00
|
|
|
Expect(output).To(ContainSubstring("acitest *"))
|
|
|
|
})
|
|
|
|
|
|
|
|
It("ensures no container is running initially", func() {
|
2020-05-20 15:57:10 +02:00
|
|
|
output := s.NewDockerCommand("ps").ExecOrDie()
|
2020-05-05 17:55:53 +02:00
|
|
|
Expect(len(Lines(output))).To(Equal(1))
|
|
|
|
})
|
|
|
|
|
2020-05-18 11:00:09 +02:00
|
|
|
It("runs nginx on port 80", func() {
|
2020-05-19 19:51:22 +02:00
|
|
|
aciContext := store.AciContext{
|
|
|
|
SubscriptionID: subscriptionID,
|
|
|
|
Location: location,
|
|
|
|
ResourceGroup: resourceGroupName,
|
|
|
|
}
|
|
|
|
createStorageAccount(aciContext, testStorageAccountName)
|
|
|
|
defer deleteStorageAccount(aciContext)
|
|
|
|
keys := getStorageKeys(aciContext, testStorageAccountName)
|
|
|
|
firstKey := *keys[0].Value
|
|
|
|
credential, u := createFileShare(firstKey, testShareName)
|
|
|
|
uploadFile(credential, u.String(), testFileName, testFileContent)
|
|
|
|
|
|
|
|
mountTarget := "/usr/share/nginx/html"
|
2020-05-20 15:57:10 +02:00
|
|
|
output := s.NewDockerCommand("run", "nginx",
|
2020-05-19 19:51:22 +02:00
|
|
|
"-v", fmt.Sprintf("%s:%s@%s:%s",
|
|
|
|
testStorageAccountName, firstKey, testShareName, mountTarget),
|
|
|
|
"-p", "80:80",
|
|
|
|
"--name", testContainerName).ExecOrDie()
|
2020-06-18 17:45:01 +02:00
|
|
|
Expect(output).To(ContainSubstring(testContainerName))
|
2020-05-20 15:57:10 +02:00
|
|
|
output = s.NewDockerCommand("ps").ExecOrDie()
|
2020-05-19 19:51:22 +02:00
|
|
|
lines := Lines(output)
|
|
|
|
Expect(len(lines)).To(Equal(2))
|
|
|
|
|
|
|
|
containerFields := Columns(lines[1])
|
|
|
|
Expect(containerFields[1]).To(Equal("nginx"))
|
|
|
|
Expect(containerFields[2]).To(Equal("Running"))
|
|
|
|
exposedIP := containerFields[3]
|
2020-06-08 12:07:20 +02:00
|
|
|
containerID := containerFields[0]
|
2020-05-19 19:51:22 +02:00
|
|
|
Expect(exposedIP).To(ContainSubstring(":80->80/tcp"))
|
|
|
|
|
|
|
|
publishedURL := strings.ReplaceAll(exposedIP, "->80/tcp", "")
|
2020-05-20 15:57:10 +02:00
|
|
|
output = s.NewCommand("curl", publishedURL).ExecOrDie()
|
2020-05-19 19:51:22 +02:00
|
|
|
Expect(output).To(ContainSubstring(testFileContent))
|
2020-06-08 12:07:20 +02:00
|
|
|
|
|
|
|
output = s.NewDockerCommand("logs", containerID).ExecOrDie()
|
|
|
|
Expect(output).To(ContainSubstring("GET"))
|
2020-05-13 09:00:48 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
It("removes container nginx", func() {
|
2020-05-20 15:57:10 +02:00
|
|
|
output := s.NewDockerCommand("rm", testContainerName).ExecOrDie()
|
2020-05-18 11:00:09 +02:00
|
|
|
Expect(Lines(output)[0]).To(Equal(testContainerName))
|
2020-05-13 09:00:48 +02:00
|
|
|
})
|
2020-05-05 17:55:53 +02:00
|
|
|
|
2020-06-18 10:03:28 +02:00
|
|
|
var exposedURL string
|
|
|
|
const composeFile = "../composefiles/aci-demo/aci_demo_port.yaml"
|
2020-06-24 18:20:27 +02:00
|
|
|
const composeFileMultiplePorts = "../composefiles/aci-demo/aci_demo_multi_port.yaml"
|
2020-06-18 10:03:28 +02:00
|
|
|
const serverContainer = "acidemo_web"
|
2020-06-24 18:20:27 +02:00
|
|
|
const wordsContainer = "acidemo_words"
|
2020-05-06 15:28:03 +02:00
|
|
|
It("deploys a compose app", func() {
|
2020-06-18 10:03:28 +02:00
|
|
|
s.NewDockerCommand("compose", "up", "-f", composeFile, "--project-name", "acidemo").ExecOrDie()
|
2020-05-13 08:54:48 +02:00
|
|
|
// Expect(output).To(ContainSubstring("Successfully deployed"))
|
2020-05-20 15:57:10 +02:00
|
|
|
output := s.NewDockerCommand("ps").ExecOrDie()
|
2020-05-06 15:28:03 +02:00
|
|
|
Lines := Lines(output)
|
|
|
|
Expect(len(Lines)).To(Equal(4))
|
2020-05-18 11:00:09 +02:00
|
|
|
webChecked := false
|
|
|
|
|
2020-05-06 15:28:03 +02:00
|
|
|
for _, line := range Lines[1:] {
|
|
|
|
Expect(line).To(ContainSubstring("Running"))
|
2020-06-18 10:03:28 +02:00
|
|
|
if strings.Contains(line, serverContainer) {
|
2020-05-18 11:00:09 +02:00
|
|
|
webChecked = true
|
|
|
|
containerFields := Columns(line)
|
|
|
|
exposedIP := containerFields[3]
|
|
|
|
Expect(exposedIP).To(ContainSubstring(":80->80/tcp"))
|
|
|
|
|
2020-06-18 10:03:28 +02:00
|
|
|
exposedURL = strings.ReplaceAll(exposedIP, "->80/tcp", "")
|
|
|
|
output = s.NewCommand("curl", exposedURL).ExecOrDie()
|
2020-05-18 11:00:09 +02:00
|
|
|
Expect(output).To(ContainSubstring("Docker Compose demo"))
|
2020-06-18 10:03:28 +02:00
|
|
|
output = s.NewCommand("curl", exposedURL+"/words/noun").ExecOrDie()
|
2020-05-18 11:00:09 +02:00
|
|
|
Expect(output).To(ContainSubstring("\"word\":"))
|
2020-05-05 17:55:53 +02:00
|
|
|
}
|
2020-05-18 11:00:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Expect(webChecked).To(BeTrue())
|
2020-05-06 15:28:03 +02:00
|
|
|
})
|
2020-05-05 17:55:53 +02:00
|
|
|
|
2020-05-06 17:14:53 +02:00
|
|
|
It("get logs from web service", func() {
|
2020-06-18 10:03:28 +02:00
|
|
|
output := s.NewDockerCommand("logs", serverContainer).ExecOrDie()
|
2020-05-06 17:14:53 +02:00
|
|
|
Expect(output).To(ContainSubstring("Listening on port 80"))
|
|
|
|
})
|
2020-05-06 15:28:03 +02:00
|
|
|
|
2020-06-18 10:03:28 +02:00
|
|
|
It("updates a compose app", func() {
|
2020-06-24 18:20:27 +02:00
|
|
|
s.NewDockerCommand("compose", "up", "-f", composeFileMultiplePorts, "--project-name", "acidemo").ExecOrDie()
|
2020-06-18 10:03:28 +02:00
|
|
|
// Expect(output).To(ContainSubstring("Successfully deployed"))
|
|
|
|
output := s.NewDockerCommand("ps").ExecOrDie()
|
|
|
|
Lines := Lines(output)
|
|
|
|
Expect(len(Lines)).To(Equal(4))
|
|
|
|
webChecked := false
|
2020-06-24 18:20:27 +02:00
|
|
|
wordsChecked := false
|
2020-06-18 10:03:28 +02:00
|
|
|
|
|
|
|
for _, line := range Lines[1:] {
|
|
|
|
Expect(line).To(ContainSubstring("Running"))
|
2020-06-24 18:20:27 +02:00
|
|
|
if strings.Contains(line, serverContainer) {
|
2020-06-18 10:03:28 +02:00
|
|
|
webChecked = true
|
|
|
|
containerFields := Columns(line)
|
|
|
|
exposedIP := containerFields[3]
|
|
|
|
Expect(exposedIP).To(ContainSubstring(":80->80/tcp"))
|
|
|
|
|
|
|
|
url := strings.ReplaceAll(exposedIP, "->80/tcp", "")
|
|
|
|
Expect(exposedURL).To(Equal(url))
|
|
|
|
}
|
2020-06-24 18:20:27 +02:00
|
|
|
if strings.Contains(line, wordsContainer) {
|
|
|
|
wordsChecked = true
|
|
|
|
containerFields := Columns(line)
|
|
|
|
exposedIP := containerFields[3]
|
|
|
|
Expect(exposedIP).To(ContainSubstring(":8080->8080/tcp"))
|
|
|
|
|
|
|
|
url := strings.ReplaceAll(exposedIP, "->8080/tcp", "")
|
|
|
|
output = s.NewCommand("curl", url+"/noun").ExecOrDie()
|
|
|
|
Expect(output).To(ContainSubstring("\"word\":"))
|
|
|
|
}
|
2020-06-18 10:03:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Expect(webChecked).To(BeTrue())
|
2020-06-24 18:20:27 +02:00
|
|
|
Expect(wordsChecked).To(BeTrue())
|
2020-06-18 10:03:28 +02:00
|
|
|
})
|
|
|
|
|
2020-05-06 15:28:03 +02:00
|
|
|
It("shutdown compose app", func() {
|
2020-06-18 10:03:28 +02:00
|
|
|
s.NewDockerCommand("compose", "down", "-f", composeFile, "--project-name", "acidemo").ExecOrDie()
|
2020-05-06 15:28:03 +02:00
|
|
|
})
|
2020-05-05 17:55:53 +02:00
|
|
|
It("switches back to default context", func() {
|
2020-05-20 15:57:10 +02:00
|
|
|
output := s.NewCommand("docker", "context", "use", "default").ExecOrDie()
|
2020-05-05 17:55:53 +02:00
|
|
|
Expect(output).To(ContainSubstring("default"))
|
|
|
|
})
|
|
|
|
|
|
|
|
It("deletes test context", func() {
|
2020-05-20 15:57:10 +02:00
|
|
|
output := s.NewCommand("docker", "context", "rm", contextName).ExecOrDie()
|
2020-05-05 17:55:53 +02:00
|
|
|
Expect(output).To(ContainSubstring(contextName))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-05-19 11:29:48 +02:00
|
|
|
const (
|
2020-06-02 23:33:41 +02:00
|
|
|
testStorageAccountName = "dockertestaccount"
|
|
|
|
testShareName = "dockertestshare"
|
2020-05-19 11:29:48 +02:00
|
|
|
testFileContent = "Volume mounted with success!"
|
|
|
|
testFileName = "index.html"
|
|
|
|
)
|
|
|
|
|
|
|
|
func createStorageAccount(aciContext store.AciContext, accountName string) azure_storage.Account {
|
2020-06-02 23:33:41 +02:00
|
|
|
log.Println("Creating storage account " + accountName)
|
2020-05-19 11:29:48 +02:00
|
|
|
storageAccount, err := storage.CreateStorageAccount(context.TODO(), aciContext, accountName)
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
Expect(*storageAccount.Name).To(Equal(accountName))
|
|
|
|
return storageAccount
|
|
|
|
}
|
|
|
|
|
|
|
|
func getStorageKeys(aciContext store.AciContext, storageAccountName string) []azure_storage.AccountKey {
|
|
|
|
list, err := storage.ListKeys(context.TODO(), aciContext, storageAccountName)
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
Expect(list.Keys).ToNot(BeNil())
|
|
|
|
Expect(len(*list.Keys)).To(BeNumerically(">", 0))
|
|
|
|
|
|
|
|
return *list.Keys
|
|
|
|
}
|
|
|
|
|
|
|
|
func deleteStorageAccount(aciContext store.AciContext) {
|
2020-06-02 23:33:41 +02:00
|
|
|
log.Println("Deleting storage account " + testStorageAccountName)
|
2020-05-19 11:29:48 +02:00
|
|
|
_, err := storage.DeleteStorageAccount(context.TODO(), aciContext, testStorageAccountName)
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
}
|
|
|
|
|
|
|
|
func createFileShare(key, shareName string) (azfile.SharedKeyCredential, url.URL) {
|
|
|
|
// Create a ShareURL object that wraps a soon-to-be-created share's URL and a default pipeline.
|
|
|
|
u, _ := url.Parse(fmt.Sprintf("https://%s.file.core.windows.net/%s", testStorageAccountName, shareName))
|
|
|
|
credential, err := azfile.NewSharedKeyCredential(testStorageAccountName, key)
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
|
|
|
|
shareURL := azfile.NewShareURL(*u, azfile.NewPipeline(credential, azfile.PipelineOptions{}))
|
|
|
|
_, err = shareURL.Create(context.TODO(), azfile.Metadata{}, 0)
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
|
|
|
|
return *credential, *u
|
|
|
|
}
|
|
|
|
|
|
|
|
func uploadFile(credential azfile.SharedKeyCredential, baseURL, fileName, fileContent string) {
|
|
|
|
fURL, err := url.Parse(baseURL + "/" + fileName)
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
fileURL := azfile.NewFileURL(*fURL, azfile.NewPipeline(&credential, azfile.PipelineOptions{}))
|
2020-05-19 20:03:53 +02:00
|
|
|
err = azfile.UploadBufferToAzureFile(context.TODO(), []byte(fileContent), fileURL, azfile.UploadToAzureFileOptions{})
|
2020-05-19 11:29:48 +02:00
|
|
|
Expect(err).To(BeNil())
|
|
|
|
}
|
|
|
|
|
2020-05-20 15:57:10 +02:00
|
|
|
func TestE2eACI(t *testing.T) {
|
|
|
|
suite.Run(t, new(E2eACISuite))
|
|
|
|
}
|
|
|
|
|
2020-05-20 06:45:06 +02:00
|
|
|
func setupTestResourceGroup(groupName string) {
|
2020-05-05 17:55:53 +02:00
|
|
|
log.Println("Creating resource group " + resourceGroupName)
|
|
|
|
ctx := context.TODO()
|
2020-06-02 23:33:41 +02:00
|
|
|
helper := azure.NewACIResourceGroupHelper()
|
|
|
|
models, err := helper.GetSubscriptionIDs(ctx)
|
2020-05-05 17:55:53 +02:00
|
|
|
Expect(err).To(BeNil())
|
2020-06-02 23:33:41 +02:00
|
|
|
_, err = helper.CreateOrUpdate(ctx, *models[0].SubscriptionID, groupName, resources.Group{
|
2020-05-13 08:54:48 +02:00
|
|
|
Location: to.StringPtr(location),
|
2020-05-05 17:55:53 +02:00
|
|
|
})
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
}
|
|
|
|
|
|
|
|
func deleteResourceGroup(groupName string) {
|
|
|
|
log.Println("Deleting resource group " + resourceGroupName)
|
|
|
|
ctx := context.TODO()
|
2020-06-02 23:33:41 +02:00
|
|
|
helper := azure.NewACIResourceGroupHelper()
|
|
|
|
models, err := helper.GetSubscriptionIDs(ctx)
|
2020-05-05 17:55:53 +02:00
|
|
|
Expect(err).To(BeNil())
|
2020-06-02 23:33:41 +02:00
|
|
|
err = helper.Delete(ctx, *models[0].SubscriptionID, groupName)
|
2020-05-05 17:55:53 +02:00
|
|
|
Expect(err).To(BeNil())
|
|
|
|
}
|