psu/cmd/stackDeploy.go

213 lines
7.1 KiB
Go
Raw Normal View History

2019-07-21 02:00:04 +00:00
package cmd
import (
"io/ioutil"
"github.com/greenled/portainer-stack-utils/client"
"github.com/sirupsen/logrus"
2019-07-21 02:00:04 +00:00
"github.com/greenled/portainer-stack-utils/common"
"github.com/joho/godotenv"
"github.com/spf13/cobra"
"github.com/spf13/viper"
)
// stackDeployCmd represents the undeploy command
var stackDeployCmd = &cobra.Command{
Use: "deploy STACK_NAME",
Short: "Deploy a new stack or update an existing one",
Aliases: []string{"up", "create"},
Example: "psu stack deploy mystack --stack-file mystack.yml",
Args: cobra.ExactArgs(1),
Run: func(cmd *cobra.Command, args []string) {
var loadedEnvironmentVariables []client.StackEnv
2019-07-21 02:00:04 +00:00
if viper.GetString("stack.deploy.env-file") != "" {
var loadingErr error
loadedEnvironmentVariables, loadingErr = loadEnvironmentVariablesFile(viper.GetString("stack.deploy.env-file"))
common.CheckError(loadingErr)
2019-07-21 02:00:04 +00:00
}
portainerClient, clientRetrievalErr := common.GetClient()
common.CheckError(clientRetrievalErr)
2019-07-21 02:00:04 +00:00
stackName := args[0]
logrus.WithFields(logrus.Fields{
"stack": stackName,
}).Debug("Getting stack")
2019-07-21 02:00:04 +00:00
retrievedStack, stackRetrievalErr := common.GetStackByName(stackName)
switch stackRetrievalErr.(type) {
case nil:
// We are updating an existing stack
logrus.WithFields(logrus.Fields{
"stack": retrievedStack.Name,
}).Debug("Stack found")
2019-07-21 02:00:04 +00:00
var stackFileContent string
if viper.GetString("stack.deploy.stack-file") != "" {
var loadingErr error
stackFileContent, loadingErr = loadStackFile(viper.GetString("stack.deploy.stack-file"))
common.CheckError(loadingErr)
2019-07-21 02:00:04 +00:00
} else {
var stackFileContentRetrievalErr error
logrus.WithFields(logrus.Fields{
"stack": retrievedStack.Name,
}).Debug("Getting stack file content")
stackFileContent, stackFileContentRetrievalErr = portainerClient.GetStackFileContent(retrievedStack.Id)
common.CheckError(stackFileContentRetrievalErr)
2019-07-21 02:00:04 +00:00
}
var newEnvironmentVariables []client.StackEnv
2019-07-21 02:00:04 +00:00
if viper.GetBool("stack.deploy.replace-env") {
newEnvironmentVariables = loadedEnvironmentVariables
} else {
// Merge stack environment variables with the loaded ones
newEnvironmentVariables = retrievedStack.Env
LoadedVariablesLoop:
for _, loadedEnvironmentVariable := range loadedEnvironmentVariables {
for _, newEnvironmentVariable := range newEnvironmentVariables {
if loadedEnvironmentVariable.Name == newEnvironmentVariable.Name {
newEnvironmentVariable.Value = loadedEnvironmentVariable.Value
continue LoadedVariablesLoop
}
}
newEnvironmentVariables = append(newEnvironmentVariables, client.StackEnv{
2019-07-21 02:00:04 +00:00
Name: loadedEnvironmentVariable.Name,
Value: loadedEnvironmentVariable.Value,
})
}
}
logrus.WithFields(logrus.Fields{
"stack": retrievedStack.Name,
}).Info("Updating stack")
err := portainerClient.UpdateStack(retrievedStack, newEnvironmentVariables, stackFileContent, viper.GetBool("stack.deploy.prune"), viper.GetString("stack.deploy.endpoint"))
common.CheckError(err)
2019-07-21 02:00:04 +00:00
case *common.StackNotFoundError:
// We are deploying a new stack
logrus.WithFields(logrus.Fields{
"stack": stackName,
}).Debug("Stack not found")
2019-07-21 02:00:04 +00:00
if viper.GetString("stack.deploy.stack-file") == "" {
logrus.WithFields(logrus.Fields{
"flag": "--stack-file",
}).Fatal("Provide required flag")
2019-07-21 02:00:04 +00:00
}
stackFileContent, loadingErr := loadStackFile(viper.GetString("stack.deploy.stack-file"))
common.CheckError(loadingErr)
2019-07-21 02:00:04 +00:00
swarmClusterId, selectionErr := getSwarmClusterId()
endpointId := viper.GetString("stack.deploy.endpoint")
2019-07-21 02:00:04 +00:00
switch selectionErr.(type) {
case nil:
// It's a swarm cluster
logrus.WithFields(logrus.Fields{
"stack": stackName,
"endpoint": endpointId,
"cluster": swarmClusterId,
}).Info("Creating stack")
deploymentErr := portainerClient.CreateSwarmStack(stackName, loadedEnvironmentVariables, stackFileContent, swarmClusterId, endpointId)
common.CheckError(deploymentErr)
2019-07-21 02:00:04 +00:00
case *valueNotFoundError:
// It's not a swarm cluster
logrus.WithFields(logrus.Fields{
"stack": stackName,
"endpoint": endpointId,
}).Info("Creating stack")
deploymentErr := portainerClient.CreateComposeStack(stackName, loadedEnvironmentVariables, stackFileContent, endpointId)
common.CheckError(deploymentErr)
2019-07-21 02:00:04 +00:00
default:
// Something else happened
common.CheckError(stackRetrievalErr)
2019-07-21 02:00:04 +00:00
}
default:
// Something else happened
common.CheckError(stackRetrievalErr)
2019-07-21 02:00:04 +00:00
}
},
}
func init() {
stackCmd.AddCommand(stackDeployCmd)
stackDeployCmd.Flags().StringP("stack-file", "c", "", "path to a file with the content of the stack")
stackDeployCmd.Flags().String("endpoint", "1", "endpoint ID")
stackDeployCmd.Flags().StringP("env-file", "e", "", "path to a file with environment variables used during stack deployment")
stackDeployCmd.Flags().Bool("replace-env", false, "replace environment variables instead of merging them")
2019-07-24 00:40:51 +00:00
stackDeployCmd.Flags().BoolP("prune", "r", false, "prune services that are no longer referenced (only available for Swarm stacks)")
2019-07-21 02:00:04 +00:00
viper.BindPFlag("stack.deploy.stack-file", stackDeployCmd.Flags().Lookup("stack-file"))
viper.BindPFlag("stack.deploy.endpoint", stackDeployCmd.Flags().Lookup("endpoint"))
viper.BindPFlag("stack.deploy.env-file", stackDeployCmd.Flags().Lookup("env-file"))
viper.BindPFlag("stack.deploy.replace-env", stackDeployCmd.Flags().Lookup("replace-env"))
viper.BindPFlag("stack.deploy.prune", stackDeployCmd.Flags().Lookup("prune"))
}
func getSwarmClusterId() (id string, err error) {
2019-07-21 02:00:04 +00:00
// Get docker information for endpoint
client, err := common.GetClient()
if err != nil {
return
2019-07-21 02:00:04 +00:00
}
endpointId := viper.GetString("stack.deploy.endpoint")
logrus.WithFields(logrus.Fields{
"endpoint": endpointId,
}).Debug("Getting endpoint's Docker info")
result, err := client.GetEndpointDockerInfo(endpointId)
if err != nil {
return
2019-07-21 02:00:04 +00:00
}
// Get swarm (if any) information for endpoint
swarmClusterId, err := selectValue(result, []string{"Swarm", "Cluster", "ID"})
if err != nil {
return
2019-07-21 02:00:04 +00:00
}
id = swarmClusterId.(string)
2019-07-21 02:00:04 +00:00
return
2019-07-21 02:00:04 +00:00
}
func selectValue(jsonMap map[string]interface{}, jsonPath []string) (interface{}, error) {
value := jsonMap[jsonPath[0]]
if value == nil {
return nil, &valueNotFoundError{}
} else if len(jsonPath) > 1 {
return selectValue(value.(map[string]interface{}), jsonPath[1:])
} else {
return value, nil
}
}
func loadStackFile(path string) (string, error) {
loadedStackFileContentBytes, readingErr := ioutil.ReadFile(path)
if readingErr != nil {
return "", readingErr
}
return string(loadedStackFileContentBytes), nil
}
// Load environment variables
func loadEnvironmentVariablesFile(path string) ([]client.StackEnv, error) {
var variables []client.StackEnv
2019-07-21 02:00:04 +00:00
variablesMap, readingErr := godotenv.Read(path)
if readingErr != nil {
return []client.StackEnv{}, readingErr
2019-07-21 02:00:04 +00:00
}
for key, value := range variablesMap {
variables = append(variables, client.StackEnv{
2019-07-21 02:00:04 +00:00
Name: key,
Value: value,
})
}
return variables, nil
}
type valueNotFoundError struct{}
func (e *valueNotFoundError) Error() string {
return "Value not found"
}