package preview import ( "context" "fmt" "os" "strings" "github.com/cli/cli/git" "github.com/fatih/color" "github.com/mitchellh/mapstructure" api "github.com/porter-dev/porter/api/client" "github.com/porter-dev/porter/api/types" "github.com/porter-dev/porter/cli/cmd/config" "github.com/porter-dev/porter/cli/cmd/deploy" "github.com/porter-dev/porter/cli/cmd/deploy/wait" "github.com/porter-dev/porter/internal/integrations/preview" "github.com/porter-dev/porter/internal/templater/utils" "github.com/porter-dev/switchboard/pkg/drivers" "github.com/porter-dev/switchboard/pkg/models" ) type UpdateConfigDriver struct { source *preview.Source target *preview.Target config *preview.UpdateConfigDriverConfig lookupTable *map[string]drivers.Driver output map[string]interface{} } func NewUpdateConfigDriver(resource *models.Resource, opts *drivers.SharedDriverOpts) (drivers.Driver, error) { driver := &UpdateConfigDriver{ lookupTable: opts.DriverLookupTable, output: make(map[string]interface{}), } target, err := GetTarget(resource.Name, resource.Target) if err != nil { return nil, err } driver.target = target source, err := GetSource(driver.target.Project, resource.Name, resource.Source) if err != nil { return nil, err } driver.source = source return driver, nil } func (d *UpdateConfigDriver) ShouldApply(resource *models.Resource) bool { return true } func (d *UpdateConfigDriver) Apply(resource *models.Resource) (*models.Resource, error) { updateConfigDriverConfig, err := d.getConfig(resource) if err != nil { return nil, err } d.config = updateConfigDriverConfig client := config.GetAPIClient() _, err = client.GetRelease( context.Background(), d.target.Project, d.target.Cluster, d.target.Namespace, d.target.AppName, ) shouldCreate := err != nil color.New(color.FgBlue).Println("checking for the existence of PORTER_TAG environment variable for the image tag") tag := os.Getenv("PORTER_TAG") if tag == "" { color.New(color.FgBlue).Println("PORTER_TAG environment variable not found, checking for update_config.tag in porter.yaml for the image tag") tag = d.config.UpdateConfig.Tag } if tag == "" { color.New(color.FgBlue).Println("update_config.tag not found in porter.yaml, falling back to the latest git commit SHA as the image tag") commit, err := git.LastCommit() if err != nil { return nil, err } tag = commit.Sha[:7] } regList, err := client.ListRegistries(context.Background(), d.target.Project) if err != nil { return nil, err } var registryURL string if len(*regList) == 0 { return nil, fmt.Errorf("no registry found") } else { registryURL = (*regList)[0].URL } var repoSuffix string if repoName := os.Getenv("PORTER_REPO_NAME"); repoName != "" { if repoOwner := os.Getenv("PORTER_REPO_OWNER"); repoOwner != "" { repoSuffix = strings.ToLower(strings.ReplaceAll(fmt.Sprintf("%s-%s", repoOwner, repoName), "_", "-")) } } sharedOpts := &deploy.SharedOpts{ ProjectID: d.target.Project, ClusterID: d.target.Cluster, OverrideTag: tag, Namespace: d.target.Namespace, Method: "registry", EnvGroups: d.config.EnvGroups, } if shouldCreate { color.New(color.FgYellow).Printf("Could not read release %s/%s: attempting creation\n", d.target.Namespace, d.target.AppName) createAgent := &deploy.CreateAgent{ Client: client, CreateOpts: &deploy.CreateOpts{ SharedOpts: sharedOpts, Kind: d.source.Name, ReleaseName: d.target.AppName, RegistryURL: registryURL, RepoSuffix: repoSuffix, }, } regID, imageURL, err := createAgent.GetImageRepoURL(d.target.AppName, sharedOpts.Namespace) if err != nil { return nil, err } err = client.CreateRepository( context.Background(), sharedOpts.ProjectID, regID, &types.CreateRegistryRepositoryRequest{ ImageRepoURI: imageURL, }, ) if err != nil { return nil, err } image := fmt.Sprintf("%s:%s", strings.Split(d.config.UpdateConfig.Image, ":")[0], tag) _, err = createAgent.CreateFromRegistry(image, d.config.Values) if err != nil { return nil, err } } else if !updateConfigDriverConfig.OnlyCreate { updateAgent, err := deploy.NewDeployAgent(client, d.target.AppName, &deploy.DeployOpts{ SharedOpts: sharedOpts, Local: false, }) if err != nil { return nil, err } err = updateAgent.UpdateImageAndValues(d.config.Values) if err != nil { return nil, err } } if d.source.Name == "job" && updateConfigDriverConfig.WaitForJob && (shouldCreate || !updateConfigDriverConfig.OnlyCreate) { color.New(color.FgYellow).Printf("Waiting for job '%s' to finish\n", resource.Name) err = wait.WaitForJob(client, &wait.WaitOpts{ ProjectID: d.target.Project, ClusterID: d.target.Cluster, Namespace: d.target.Namespace, Name: d.target.AppName, }) if err != nil { return nil, err } } err = d.assignOutput(resource, client) if err != nil { return nil, err } return resource, nil } func (d *UpdateConfigDriver) Output() (map[string]interface{}, error) { return d.output, nil } func (d *UpdateConfigDriver) getConfig(resource *models.Resource) (*preview.UpdateConfigDriverConfig, error) { populatedConf, err := drivers.ConstructConfig(&drivers.ConstructConfigOpts{ RawConf: resource.Config, LookupTable: *d.lookupTable, Dependencies: resource.Dependencies, }) if err != nil { return nil, err } config := &preview.UpdateConfigDriverConfig{} err = mapstructure.Decode(populatedConf, config) if err != nil { return nil, err } return config, nil } func (d *UpdateConfigDriver) assignOutput(resource *models.Resource, client *api.Client) error { release, err := client.GetRelease( context.Background(), d.target.Project, d.target.Cluster, d.target.Namespace, d.target.AppName, ) if err != nil { return err } d.output = utils.CoalesceValues(d.source.SourceValues, release.Config) return nil }