Skip to content

Commit

Permalink
WIP: Refactor HelmChartReconciler
Browse files Browse the repository at this point in the history
Signed-off-by: Hidde Beydals <[email protected]>
  • Loading branch information
hiddeco committed Jun 14, 2021
1 parent afbaaca commit 2b2cef8
Show file tree
Hide file tree
Showing 18 changed files with 1,630 additions and 2,061 deletions.
2 changes: 2 additions & 0 deletions api/v1beta1/condition_types.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,8 @@ const (
SourceVerifiedCondition string = "SourceVerified"

ArtifactAvailableCondition string = "ArtifactAvailable"

SourceRefReadyCondition string = "SourceRefReady"
)

const (
Expand Down
56 changes: 15 additions & 41 deletions api/v1beta1/helmchart_types.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,13 +18,19 @@ package v1beta1

import (
"github.com/fluxcd/pkg/apis/meta"
apimeta "k8s.io/apimachinery/pkg/api/meta"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
)

// HelmChartKind is the string representation of a HelmChart.
const HelmChartKind = "HelmChart"

const (
DependenciesBuildCondition string = "DependenciesBuild"
ValuesFilesMergedCondition string = "ValuesFilesMerged"
ChartPackagedCondition string = "ChartPackaged"
ChartReconciled string = "ChartReconciled"
)

// HelmChartSpec defines the desired state of a Helm chart.
type HelmChartSpec struct {
// The name or path the Helm chart is available at in the SourceRef.
Expand Down Expand Up @@ -122,52 +128,20 @@ const (
ChartPackageSucceededReason string = "ChartPackageSucceeded"
)

// HelmChartProgressing resets the conditions of the HelmChart to meta.Condition
// of type meta.ReadyCondition with status 'Unknown' and meta.ProgressingReason
// reason and message. It returns the modified HelmChart.
func HelmChartProgressing(chart HelmChart) HelmChart {
chart.Status.ObservedGeneration = chart.Generation
chart.Status.URL = ""
chart.Status.Conditions = []metav1.Condition{}
meta.SetResourceCondition(&chart, meta.ReadyCondition, metav1.ConditionUnknown, meta.ProgressingReason, "reconciliation in progress")
return chart
}

// HelmChartReady sets the given Artifact and URL on the HelmChart and sets the
// meta.ReadyCondition to 'True', with the given reason and message. It returns
// the modified HelmChart.
func HelmChartReady(chart HelmChart, artifact Artifact, url, reason, message string) HelmChart {
chart.Status.Artifact = &artifact
chart.Status.URL = url
meta.SetResourceCondition(&chart, meta.ReadyCondition, metav1.ConditionTrue, reason, message)
return chart
}

// HelmChartNotReady sets the meta.ReadyCondition on the given HelmChart to
// 'False', with the given reason and message. It returns the modified
// HelmChart.
func HelmChartNotReady(chart HelmChart, reason, message string) HelmChart {
meta.SetResourceCondition(&chart, meta.ReadyCondition, metav1.ConditionFalse, reason, message)
return chart
}

// HelmChartReadyMessage returns the message of the meta.ReadyCondition with
// status 'True', or an empty string.
func HelmChartReadyMessage(chart HelmChart) string {
if c := apimeta.FindStatusCondition(chart.Status.Conditions, meta.ReadyCondition); c != nil {
if c.Status == metav1.ConditionTrue {
return c.Message
}
}
return ""
}

// GetArtifact returns the latest artifact from the source if present in the
// status sub-resource.
func (in *HelmChart) GetArtifact() *Artifact {
return in.Status.Artifact
}

func (in HelmChart) GetConditions() []metav1.Condition {
return in.Status.Conditions
}

func (in *HelmChart) SetConditions(conditions []metav1.Condition) {
in.Status.Conditions = conditions
}

// GetStatusConditions returns a pointer to the Status.Conditions slice
func (in *HelmChart) GetStatusConditions() *[]metav1.Condition {
return &in.Status.Conditions
Expand Down
7 changes: 7 additions & 0 deletions api/v1beta1/source.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ package v1beta1

import (
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime"
)

const (
Expand All @@ -29,9 +30,15 @@ const (
// Source interface must be supported by all API types.
// +k8s:deepcopy-gen=false
type Source interface {
metav1.Object
runtime.Object

// GetArtifact returns the latest artifact from the source if present in the
// status sub-resource.
GetArtifact() *Artifact
// GetConditions returns the conditions of the source if present in the
// status sub-resource.
GetConditions() []metav1.Condition
// GetInterval returns the interval at which the source is updated.
GetInterval() metav1.Duration
}
2 changes: 1 addition & 1 deletion api/v1beta1/zz_generated.deepcopy.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

15 changes: 10 additions & 5 deletions controllers/bucket_controller.go
Original file line number Diff line number Diff line change
Expand Up @@ -147,10 +147,16 @@ func (r *BucketReconciler) Reconcile(ctx context.Context, req ctrl.Request) (res
// We have now observed this generation
patchOpts = append(patchOpts, patch.WithStatusObservedGeneration{})

if readyCondition := conditions.Get(obj, meta.ReadyCondition); readyCondition.Status == metav1.ConditionFalse {
// As we are no longer reconciling, and the end-state
readyCondition := conditions.Get(obj, meta.ReadyCondition)
switch readyCondition.Status {
case metav1.ConditionFalse:
// As we are no longer reconciling and the end-state
// is not ready, the reconciliation has stalled
conditions.MarkTrue(obj, meta.StalledCondition, readyCondition.Reason, readyCondition.Message)
case metav1.ConditionTrue:
// As we are no longer reconciling and the end-state
// is ready, the reconciliation is no longer stalled
conditions.Delete(obj, meta.StalledCondition)
}
}

Expand Down Expand Up @@ -357,7 +363,7 @@ func (r *BucketReconciler) reconcileArtifact(ctx context.Context, obj *sourcev1.
// The artifact is up-to-date
if obj.GetArtifact().HasRevision(artifact.Revision) {
logr.FromContext(ctx).Info("Artifact is up-to-date")
conditions.MarkTrue(obj, sourcev1.ArtifactAvailableCondition, "ArchivedArtifact", "Artifact revision %s", artifact.Revision)
conditions.MarkTrue(obj, sourcev1.ArtifactAvailableCondition, "ArchivedArtifact", "Compressed source to artifact with revision %s", artifact.Revision)
return ctrl.Result{RequeueAfter: obj.GetInterval().Duration}, nil
}

Expand All @@ -372,7 +378,6 @@ func (r *BucketReconciler) reconcileArtifact(ctx context.Context, obj *sourcev1.
}

// Ensure artifact directory exists and acquire lock
err := r.Storage.MkdirAll(artifact)
if err := r.Storage.MkdirAll(artifact); err != nil {
conditions.MarkFalse(obj, sourcev1.ArtifactAvailableCondition, sourcev1.StorageOperationFailedReason, "Failed to create directory: %s", err)
return ctrl.Result{}, err
Expand All @@ -392,7 +397,7 @@ func (r *BucketReconciler) reconcileArtifact(ctx context.Context, obj *sourcev1.

// Record it on the object
obj.Status.Artifact = artifact.DeepCopy()
conditions.MarkTrue(obj, sourcev1.ArtifactAvailableCondition, "ArchivedArtifact", "Artifact revision %s", artifact.Revision)
conditions.MarkTrue(obj, sourcev1.ArtifactAvailableCondition, "ArchivedArtifact", "Compressed source to artifact with revision %s", artifact.Revision)
r.Events.Eventf(ctx, obj, map[string]string{
"revision": obj.GetArtifact().Revision,
}, events.EventSeverityInfo, sourcev1.GitOperationSucceedReason, conditions.Get(obj, sourcev1.ArtifactAvailableCondition).Message)
Expand Down
2 changes: 1 addition & 1 deletion controllers/bucket_controller_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,7 @@ import (

const (
bucketInterval = 1 * time.Second
bucketTimeout = 30 * time.Second
bucketTimeout = 30 * time.Second
)

func TestBucketReconciler_Reconcile(t *testing.T) {
Expand Down
15 changes: 10 additions & 5 deletions controllers/gitrepository_controller.go
Original file line number Diff line number Diff line change
Expand Up @@ -154,10 +154,16 @@ func (r *GitRepositoryReconciler) Reconcile(ctx context.Context, req ctrl.Reques
// We have now observed this generation
patchOpts = append(patchOpts, patch.WithStatusObservedGeneration{})

if readyCondition := conditions.Get(obj, meta.ReadyCondition); readyCondition.Status == metav1.ConditionFalse {
// As we are no longer reconciling, and the end-state
readyCondition := conditions.Get(obj, meta.ReadyCondition)
switch readyCondition.Status {
case metav1.ConditionFalse:
// As we are no longer reconciling and the end-state
// is not ready, the reconciliation has stalled
conditions.MarkTrue(obj, meta.StalledCondition, readyCondition.Reason, readyCondition.Message)
case metav1.ConditionTrue:
// As we are no longer reconciling and the end-state
// is ready, the reconciliation is no longer stalled
conditions.Delete(obj, meta.StalledCondition)
}
}

Expand Down Expand Up @@ -331,7 +337,7 @@ func (r *GitRepositoryReconciler) reconcileSource(ctx context.Context, obj *sour

// Create potential new artifact
*artifact = r.Storage.NewArtifactFor(obj.Kind, obj, revision, fmt.Sprintf("%s.tar.gz", commit.Hash()))
conditions.MarkTrue(obj, sourcev1.SourceAvailableCondition, sourcev1.GitOperationSucceedReason, "Checked out revision %s from %s", revision, obj.Spec.URL)
conditions.MarkTrue(obj, sourcev1.SourceAvailableCondition, sourcev1.GitOperationSucceedReason, "Checked out revision %s", revision)

return ctrl.Result{RequeueAfter: obj.Spec.Interval.Duration}, nil
}
Expand Down Expand Up @@ -362,7 +368,6 @@ func (r *GitRepositoryReconciler) reconcileArtifact(ctx context.Context, obj *so
}

// Ensure artifact directory exists and acquire lock
err := r.Storage.MkdirAll(artifact)
if err := r.Storage.MkdirAll(artifact); err != nil {
conditions.MarkFalse(obj, sourcev1.ArtifactAvailableCondition, sourcev1.StorageOperationFailedReason, "Failed to create directory: %s", err)
return ctrl.Result{}, err
Expand Down Expand Up @@ -393,7 +398,7 @@ func (r *GitRepositoryReconciler) reconcileArtifact(ctx context.Context, obj *so
// Record it on the object
obj.Status.Artifact = artifact.DeepCopy()
obj.Status.IncludedArtifacts = includes
conditions.MarkTrue(obj, sourcev1.ArtifactAvailableCondition, "ArchivedArtifact", "Artifact revision %s", artifact.Revision)
conditions.MarkTrue(obj, sourcev1.ArtifactAvailableCondition, "ArchivedArtifact", "Compressed source to artifact with revision %s", artifact.Revision)
r.Events.Eventf(ctx, obj, map[string]string{
"revision": obj.GetArtifact().Revision,
}, events.EventSeverityInfo, sourcev1.GitOperationSucceedReason, conditions.Get(obj, sourcev1.ArtifactAvailableCondition).Message)
Expand Down
39 changes: 25 additions & 14 deletions controllers/gitrepository_controller_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -66,6 +66,12 @@ var (
timeout = 10 * time.Second
mockInterval = 1 * time.Second
testGitImplementations = []string{sourcev1.GoGitImplementation, sourcev1.LibGit2Implementation}
testGetters = getter.Providers{
getter.Provider{
Schemes: []string{"http", "https"},
New: getter.NewHTTPGetter,
},
}
)

var (
Expand Down Expand Up @@ -160,7 +166,7 @@ func TestGitRepositoryReconciler_reconcileSource_authStrategy(t *testing.T) {
protocol: "http",
want: ctrl.Result{RequeueAfter: mockInterval},
assertConditions: []metav1.Condition{
*conditions.TrueCondition(sourcev1.SourceAvailableCondition, "SuccessfulCheckout", "Checked out revision master/<commit> from <url>"),
*conditions.TrueCondition(sourcev1.SourceAvailableCondition, sourcev1.GitOperationSucceedReason, "Checked out revision master/<commit>"),
},
},
{
Expand All @@ -184,7 +190,7 @@ func TestGitRepositoryReconciler_reconcileSource_authStrategy(t *testing.T) {
},
want: ctrl.Result{RequeueAfter: mockInterval},
assertConditions: []metav1.Condition{
*conditions.TrueCondition(sourcev1.SourceAvailableCondition, "SuccessfulCheckout", "Checked out revision master/<commit> from <url>"),
*conditions.TrueCondition(sourcev1.SourceAvailableCondition, sourcev1.GitOperationSucceedReason, "Checked out revision master/<commit>"),
},
},
{
Expand All @@ -208,7 +214,7 @@ func TestGitRepositoryReconciler_reconcileSource_authStrategy(t *testing.T) {
},
want: ctrl.Result{RequeueAfter: mockInterval},
assertConditions: []metav1.Condition{
*conditions.TrueCondition(sourcev1.SourceAvailableCondition, "SuccessfulCheckout", "Checked out revision master/<commit> from <url>"),
*conditions.TrueCondition(sourcev1.SourceAvailableCondition, sourcev1.GitOperationSucceedReason, "Checked out revision master/<commit>"),
},
},
{
Expand Down Expand Up @@ -281,7 +287,7 @@ func TestGitRepositoryReconciler_reconcileSource_authStrategy(t *testing.T) {
},
want: ctrl.Result{RequeueAfter: mockInterval},
assertConditions: []metav1.Condition{
*conditions.TrueCondition(sourcev1.SourceAvailableCondition, "SuccessfulCheckout", "Checked out revision master/<commit> from <url>"),
*conditions.TrueCondition(sourcev1.SourceAvailableCondition, sourcev1.GitOperationSucceedReason, "Checked out revision master/<commit>"),
},
},
{
Expand All @@ -305,7 +311,7 @@ func TestGitRepositoryReconciler_reconcileSource_authStrategy(t *testing.T) {
},
want: ctrl.Result{RequeueAfter: mockInterval},
assertConditions: []metav1.Condition{
*conditions.TrueCondition(sourcev1.SourceAvailableCondition, "SuccessfulCheckout", "Checked out revision master/<commit> from <url>"),
*conditions.TrueCondition(sourcev1.SourceAvailableCondition, sourcev1.GitOperationSucceedReason, "Checked out revision master/<commit>"),
},
},
{
Expand All @@ -319,7 +325,7 @@ func TestGitRepositoryReconciler_reconcileSource_authStrategy(t *testing.T) {
},
want: ctrl.Result{RequeueAfter: mockInterval},
assertConditions: []metav1.Condition{
*conditions.FalseCondition(sourcev1.SourceAvailableCondition, "AuthenticationFailed", "Failed to get auth secret /non-existing: secrets \"non-existing\" not found"),
*conditions.FalseCondition(sourcev1.SourceAvailableCondition, "AuthenticationFailed", "Failed to get secret /non-existing: secrets \"non-existing\" not found"),
},
},
}
Expand Down Expand Up @@ -595,7 +601,7 @@ func TestGitRepositoryReconciler_reconcileArtifact(t *testing.T) {
},
want: ctrl.Result{RequeueAfter: mockInterval},
assertConditions: []metav1.Condition{
*conditions.TrueCondition(sourcev1.ArtifactAvailableCondition, "ArchivedArtifact", "Artifact revision main/revision"),
*conditions.TrueCondition(sourcev1.ArtifactAvailableCondition, "ArchivedArtifact", "Compressed source to artifact with revision main/revision"),
},
},
{
Expand Down Expand Up @@ -627,7 +633,7 @@ func TestGitRepositoryReconciler_reconcileArtifact(t *testing.T) {
},
want: ctrl.Result{RequeueAfter: mockInterval},
assertConditions: []metav1.Condition{
*conditions.TrueCondition(sourcev1.ArtifactAvailableCondition, "ArchivedArtifact", "Archived artifact revision main/revision"),
*conditions.TrueCondition(sourcev1.ArtifactAvailableCondition, "ArchivedArtifact", "Compressed source to artifact with revision main/revision"),
},
},
}
Expand Down Expand Up @@ -1070,12 +1076,7 @@ func TestMain(m *testing.M) {
Client: newTestEnv,
Events: eventsHelper,
Metrics: metricsHelper,
Getters: getter.Providers{
getter.Provider{
Schemes: []string{"http", "https"},
New: getter.NewHTTPGetter,
},
},
Getters: testGetters,
Storage: storage,
}).SetupWithManager(newTestEnv); err != nil {
panic(fmt.Sprintf("Failed to start HelmRepositoryReconciler: %v", err))
Expand All @@ -1090,6 +1091,16 @@ func TestMain(m *testing.M) {
panic(fmt.Sprintf("Failed to start BucketReconciler: %v", err))
}

if err := (&HelmChartReconciler{
Client: newTestEnv,
Events: eventsHelper,
Metrics: metricsHelper,
Getters: testGetters,
Storage: storage,
}).SetupWithManager(newTestEnv); err != nil {
panic(fmt.Sprintf("Failed to start HelmChartReconciler: %v", err))
}

go func() {
fmt.Println("Starting the test environment manager")
if err := newTestEnv.StartManager(ctx); err != nil {
Expand Down
Loading

0 comments on commit 2b2cef8

Please sign in to comment.