package sources import ( "slices" portainer "github.com/portainer/portainer/api" "github.com/portainer/portainer/api/dataservices" ce "github.com/portainer/portainer/api/gitops/workflows" "github.com/portainer/portainer/api/set" "github.com/portainer/portainer/api/slicesx" ) // FetchSourceWorkflows returns the workflows and stats for a single source. func FetchSourceWorkflows(tx dataservices.DataStoreTx, src *portainer.Source) ([]ce.Workflow, ce.SourceStats, error) { wfs, err := tx.Workflow().ReadAll(func(wf portainer.Workflow) bool { return slices.ContainsFunc(wf.Artifacts, func(artifact portainer.ArtifactSources) bool { return slices.Contains(artifact.SourceIDs, src.ID) }) }) if err != nil { return nil, ce.SourceStats{}, err } if len(wfs) == 0 { return nil, ce.SourceStats{}, nil } wfIDSet := set.ToSet(slicesx.Map(wfs, func(wf portainer.Workflow) portainer.WorkflowID { return wf.ID })) stacks, err := tx.Stack().ReadAll(func(s portainer.Stack) bool { _, ok := wfIDSet[s.WorkflowID] return ok }) if err != nil { return nil, ce.SourceStats{}, err } unknown := ce.WorkflowPhaseStatus{Status: ce.StatusUnknown} items := make([]ce.Workflow, 0, len(stacks)) stats := ce.SourceStats{EndpointIDs: set.Set[portainer.EndpointID]{}} for _, stacks := range stacks { items = append(items, ce.MapStackToWorkflow(stacks, src.GitConfig, unknown, unknown)) stats.WorkflowCount++ if stacks.EndpointID != 0 { stats.EndpointIDs.Add(stacks.EndpointID) } if lastSync := ce.StackLastSyncDate(stacks); lastSync > stats.LastSync { stats.LastSync = lastSync } } return items, stats, nil }