mirror of
https://github.com/portainer/portainer.git
synced 2026-06-23 06:40:13 +00:00
fcdd6b4510
Co-authored-by: Claude Sonnet 4.6 <noreply@anthropic.com>
170 lines
4.5 KiB
Go
170 lines
4.5 KiB
Go
package stackbuilders
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strconv"
|
|
|
|
portainer "github.com/portainer/portainer/api"
|
|
"github.com/portainer/portainer/api/dataservices"
|
|
"github.com/portainer/portainer/api/filesystem"
|
|
gittypes "github.com/portainer/portainer/api/git/types"
|
|
"github.com/portainer/portainer/api/gitops/workflows"
|
|
"github.com/portainer/portainer/api/scheduler"
|
|
"github.com/portainer/portainer/api/stacks/deployments"
|
|
"github.com/portainer/portainer/api/stacks/stackutils"
|
|
)
|
|
|
|
type GitMethodStackBuilder struct {
|
|
StackBuilder
|
|
gitService portainer.GitService
|
|
scheduler *scheduler.Scheduler
|
|
}
|
|
|
|
func (b *GitMethodStackBuilder) prepare(ctx context.Context, payload *StackPayload, userID portainer.UserID) error {
|
|
b.stack.AdditionalFiles = payload.AdditionalFiles
|
|
b.stack.AutoUpdate = payload.AutoUpdate
|
|
|
|
if err := b.initCreatedBy(userID); err != nil {
|
|
return err
|
|
}
|
|
|
|
var repoConfig gittypes.RepoConfig
|
|
var sourceID portainer.SourceID
|
|
|
|
if payload.SourceID != 0 {
|
|
src, err := b.dataStore.Source().Read(payload.SourceID)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to read source: %w", err)
|
|
}
|
|
if src.Git == nil {
|
|
return fmt.Errorf("source %d has no git configuration", payload.SourceID)
|
|
}
|
|
|
|
repoConfig.URL = src.Git.URL
|
|
repoConfig.Authentication = src.Git.Authentication
|
|
repoConfig.TLSSkipVerify = src.Git.TLSSkipVerify
|
|
repoConfig.ReferenceName = payload.ReferenceName
|
|
sourceID = src.ID
|
|
} else {
|
|
if payload.Authentication {
|
|
repoConfig.Authentication = &gittypes.GitAuthentication{
|
|
Username: payload.Username,
|
|
Password: payload.Password,
|
|
}
|
|
}
|
|
|
|
repoConfig.URL = payload.URL
|
|
repoConfig.ReferenceName = payload.ReferenceName
|
|
repoConfig.TLSSkipVerify = payload.TLSSkipVerify
|
|
}
|
|
|
|
repoConfig.ConfigFilePath = payload.ComposeFile
|
|
if payload.ComposeFile == "" {
|
|
repoConfig.ConfigFilePath = filesystem.ComposeFileDefaultName
|
|
}
|
|
|
|
// If a manifest file is specified (for kube git apps), then use it instead of the default compose file name
|
|
if payload.ManifestFile != "" {
|
|
repoConfig.ConfigFilePath = payload.ManifestFile
|
|
}
|
|
|
|
stackFolder := strconv.Itoa(int(b.stack.ID))
|
|
// Set the project path on the disk
|
|
b.stack.ProjectPath = b.fileService.GetStackProjectPath(stackFolder)
|
|
|
|
getProjectPath := func() string {
|
|
stackFolder := fmt.Sprintf("%d", b.stack.ID)
|
|
return b.fileService.GetStackProjectPath(stackFolder)
|
|
}
|
|
|
|
commitHash, err := stackutils.DownloadGitRepository(ctx, repoConfig, b.gitService, getProjectPath)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to download git repository: %w", err)
|
|
}
|
|
|
|
// Update the latest commit id
|
|
repoConfig.ConfigHash = commitHash
|
|
|
|
var workflowID portainer.WorkflowID
|
|
|
|
if err := b.dataStore.UpdateTx(func(tx dataservices.DataStoreTx) error {
|
|
file := portainer.ArtifactFile{
|
|
Path: repoConfig.ConfigFilePath,
|
|
Ref: repoConfig.ReferenceName,
|
|
Hash: repoConfig.ConfigHash,
|
|
}
|
|
|
|
if sourceID != 0 {
|
|
file.SourceID = sourceID
|
|
} else {
|
|
repoConfig.URL = gittypes.SanitizeURL(repoConfig.URL)
|
|
|
|
src, err := workflows.FindOrCreateGitSource(tx, &portainer.Source{
|
|
Name: gittypes.RepoName(repoConfig.URL),
|
|
Type: portainer.SourceTypeGit,
|
|
Git: &repoConfig,
|
|
})
|
|
if err != nil {
|
|
return fmt.Errorf("failed to find or create source: %w", err)
|
|
}
|
|
|
|
file.SourceID = src.ID
|
|
}
|
|
|
|
wf := &portainer.Workflow{
|
|
Name: b.stack.Name,
|
|
Artifacts: []portainer.Artifact{{
|
|
StackID: b.stack.ID,
|
|
Files: []portainer.ArtifactFile{file},
|
|
}},
|
|
}
|
|
if err := tx.Workflow().Create(wf); err != nil {
|
|
return fmt.Errorf("failed to create workflow: %w", err)
|
|
}
|
|
|
|
workflowID = wf.ID
|
|
|
|
return nil
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
|
|
b.stack.WorkflowID = workflowID
|
|
|
|
return nil
|
|
}
|
|
|
|
// postDeploy enables the auto-update scheduler job for the stack if configured,
|
|
// and persists the resulting job ID back to the database.
|
|
func (b *GitMethodStackBuilder) postDeploy(ctx context.Context, stack *portainer.Stack) error {
|
|
if stack.AutoUpdate == nil || stack.AutoUpdate.Interval == "" {
|
|
return nil
|
|
}
|
|
|
|
jobID, err := deployments.StartAutoupdate(ctx, stack.ID,
|
|
stack.AutoUpdate.Interval,
|
|
b.scheduler,
|
|
b.stackDeployer,
|
|
b.dataStore,
|
|
b.gitService)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return b.dataStore.UpdateTx(func(tx dataservices.DataStoreTx) error {
|
|
s, err := tx.Stack().Read(stack.ID)
|
|
if err != nil {
|
|
return fmt.Errorf("Unable to retrieve the stack from the database: %w", err)
|
|
}
|
|
|
|
s.AutoUpdate.JobID = jobID
|
|
|
|
if err := tx.Stack().Update(s.ID, s); err != nil {
|
|
return fmt.Errorf("Unable to update the stack inside the database: %w", err)
|
|
}
|
|
|
|
return nil
|
|
})
|
|
}
|