From fd35fb4edd1780fad4572263b7fb18446c2ad3da Mon Sep 17 00:00:00 2001 From: Aditya Addepalli Date: Wed, 10 Jul 2024 13:34:09 -0400 Subject: [PATCH] Add limitations related to persisted information Signed-off-by: Aditya Addepalli --- .../stateful-failover-proposal.md | 215 ++++++++++++------ 1 file changed, 148 insertions(+), 67 deletions(-) diff --git a/docs/proposals/stateful-failover/stateful-failover-proposal.md b/docs/proposals/stateful-failover/stateful-failover-proposal.md index e56d95ab321c..96b90f69a1f9 100644 --- a/docs/proposals/stateful-failover/stateful-failover-proposal.md +++ b/docs/proposals/stateful-failover/stateful-failover-proposal.md @@ -16,54 +16,55 @@ create-date: 2024-06-28 # Stateful Failover Support in Karmada +Special thanks to `@RainbowMango` and `@XiShanYongYe-Chang` for their help with the API design! + ## Summary Karmada can be currently used to intelligently schedule all types of resources (both generic Kubernetes objects as well as user-applied CRDs). It is particularly useful for ensuring stateful application resilience in a multi-cluster environment in which applications may be rescheduled if a cluster becomes healthy. -However, Karmada’s scheduling logic runs on the assumption that resources that are scheduled and rescheduled are stateless. In some cases, users may desire to conserve a certain state so that applications can resume from where they left off in the previous cluster. - -For CRDs dealing with data-processing (such as Flink or Spark), it can be particularly useful to restart applications from a previous checkpoint. That way applications can seamlessly resume processing data while avoiding double processing. +However, Karmada’s scheduling logic runs on the assumption that resources that are scheduled and rescheduled are stateless. In some cases, users may desire to conserve a certain state so that applications can resume from where they left off in the previous cluster. -This proposal would like to introduce a more generalized way for users to define failover actions, which can be used in the case of stateful CRD failover. +For CRDs dealing with data-processing (such as Flink or Spark), it can be particularly useful to restart applications from a previous checkpoint. That way applications can seamlessly resume processing data while avoiding double processing. +This proposal would like to introduce a more generalized way for users to define application state preservation in the context of cluster to cluster failovers. ## Motivation This proposal aims to provide a framework for stateful CRDs to keep track of the state required during failover so that processing can be resumed from that state after failover has completed. -### Goals - -To add support for stateful application failover: -- Extend ResourceBinding API to include FailoverHistory field -- Extend PropagationPolicy API to include status items that should be persisted during failover - ## Proposal -Stateful applications need a way to read the last saved state to resume processing from that state after failover. +Karmada introduces a higher level of resiliency to kubernetes-deployed applications through the use of its failover feature. However, this failover feature does not currently account for state. Stateless applications have the benefit of being more fault-tolerant, as the loss of an application will not impact user sessions. For stateful applications however (such as Flink), the loss of the application can will result in loss of session data that can prevent the application from recovering correctly. -To enable this we split this proposal into two sections: -- A mechanism for when a failover happened so that stateful applications can load the previous state first before resuming processing. -- At the very minimum, a way of storing and reading state/metadata related to the job being failed over. +This proposal introduces a way for users to persist state required for their application in the event of a failover. That way, once rescheduled, users can configure their applications to resume using the metadata saved by Karmada. Since stateful applications have different implementations of how to retrieve the last state, it is important that this feature is configurable. -Since stateful applications have different implementations of how to retrieve the last state given this the job metadata we would then rely on those individual implementations to fetch all the details related to the last state. +To support this feature, we will need at minimum: +- A mechanism with which stateful applications can preserve state during failover and reload before resuming. +- A way users can configure which state will be preserved by the Karmada +- Flags that will denote when a resource has been failed over by Karmada -**NOTE: To be discussed with Karmada community** +`Note`: One important detail is that if all the replicas of the stateful application are not migrated together, it is not clear when the state needs to be restored. In this proposal we focus on the use case where all the replicas of a stateful application are migrated together. One way to ensure this is to make all the replicas scheduled together using spreadConstraints. -One important detail is that if all the replicas of the stateful application are not migrated together, it is not clear when the state needs to be restored. In this proposal we focus on the use case where all the replicas of a stateful application are migrated together. One way to ensure this is to make all the replicas scheduled together using spreadConstraints. - -``` +```yaml spreadConstraints: - spreadByField: cluster maxGroups: 1 minGroups: 1 ``` +### High Level Goals + +The proposal can be logically separated into three different efforts: +1. Extension of the ResourceBindingStatus API to include FailoverHistory field, FailoverHistory implementation +2. Addition of flags which mark resources when they failover +3. Extension of the PropagationPolicy API to include state preservation rules, which define which state will be conserved during failover + ### Use-case with Flink: In Flink for example, checkpoints are snapshots that store the state of the application till a particular moment in time. As a result they contain information about all the records processed till that moment. This information is collected and persisted continuously to some persistent storage at specific intervals. -In Flink, to retrieve the last state from the persistent store, we would require metadata about the job being restored (Job ID) along with the path where the state is being persisted. +In Flink, to retrieve the last state from the persistent store, we would require metadata about the job being restored (Job ID) along with the path where the state is being persisted. The Flink Operator can then use this information to retrieve the last state by providing a few modifications to the Flink CRD spec. @@ -71,7 +72,7 @@ The Flink Operator can then use this information to retrieve the last state by p Spec: Flink Configuration: initialSavepointPath: “desired checkpoint path to be resumed from (s3p://)” -upgradeMode: savepoint +upgradeMode: savepoint state: running ``` @@ -81,18 +82,16 @@ We believe that Karmada would benefit from having a generic way to store the job To summarize, we currently resume from the last state with some custom changes and third party services which: 1. Append a flag to the ResourceBinding of the CRD to indicate a failover. When the application is rescheduled, a label is appended to the CRD to mark it as failed-over. - 2. A custom Kyverno policy then reads the failover label and grabs the latest checkpoint for the application based on its jobID. - 3. The application can then resume from the fetched checkpoint and recommence processing. + 2. A custom Kyverno policy then reads the failover label and grabs the latest checkpoint for the application based on its jobID. + 3. The application can then resume from the fetched checkpoint and recommence processing. -## Design Details +## Stateful Failover API Design -### Support Stateful Failover Options +### ResourceBindingStatus API -#### ResourceBinding API Change +We can extend existing ResourceBindingStatus API with a new field "FailoverHistory", which would be added by the cluster + application failover controller to keep track when an application has been failed-over. -We can extend the ResourceBindingStatus with a new field "FailoverHistory", which would be added by the cluster + application failover controller to keep track when the replica has been failed-over. - -``` +```go // ResourceBindingStatus represents the overall status of the strategy as well as the referenced resources. type ResourceBindingStatus struct { // SchedulerObservedGeneration is the generation(.metadata.generation) observed by the scheduler. @@ -120,71 +119,153 @@ type ResourceBindingStatus struct { } ``` -The FailoverHistory is a list of FailoverHistoryItem objects and is updated every time a failover happens until a certain limit. The limit is set by another field persistedFields.maxHistory which is defined in the propogation policy. +### FailoverHistoryItem API -``` +The FailoverHistory is a list of FailoverHistoryItem objects and is updated every time a failover happens. The amount of items can be configured up to a certain limit, as to minimize excess status length. This limit is set by another field persistedFields.maxHistory which is defined in the propogation policy. + +```go +// FailoverHistoryItem represents either a failover event in the history. type FailoverHistoryItem struct { + // Reason denotes the type of failover. + // +required + Reason FailoverReason `json:"reason"` - // FailoverTime represents the timestamp when the workload failed over - // It is represented in RFC3339 form(like '2021-04-25T10:02:10Z') and is in UTC - FailoverTime *metav1.Time `json:"failoverTime,omitempty"` + // StartTime is the timestamp of when the failover occurred. + // +required + StartTime metav1.Time `json:"failoverTime"` - // OriginCluster represents the cluster name from which the workload failed over from - OriginCluster string `json:"originCluster,omitempty"` + // FromCluster is the cluster name from which application was migrated. + // +required + FromCluster string `json:"fromCluster"` - // DestinationCluster represents the cluster name from which the workload failed over to - DestinationCluster string `json:"destinationCluster,omitempty"` + // ClustersBeforeFailover records the clusters where the application was running prior to failover. + // +required + ClusterBeforeFailover []string `json:"originalCluster"` - // PersistedDuringFailover contains the fields required by the stateful application to resume from that state after failover - PersistedDuringFailover []PersistedFailoverItem `json:"persistedFailoverItem,omitempty"` + // ClustersAfterFailover records the clusters where the application is running after failover. + // +optional + ClusterAfterFailover []string `json:"targetCluster,omitempty"` + + // PreservedLabelState represents the application state information collected from the original cluster, + // and it will be injected into the new cluster in the form of application labels. + // +optional + PreservedLabelState map[string]string `json:"preservedLabelState,omitempty"` } -``` -The FailoverHistoryItem object contains information relevant to a failover and an additional object called "PersistedDuringFailover" which keeps track of the metadata (both the fields that need to be persisted and how to access those fields) that is required by the stateful operation to resume processing from that state. +// FailoverReason represents the reason for the failover. +type FailoverReason string -``` -type PersistedDuringFailover struct { - - // LabelName represents the name of the line that will be persisted for the replica - // in case there is a failover to a new cluster. - LabelName string `json:"labelName,omitempty"` - - // PersistedItem is a pointer to the status item that should be persisted to the rescheduled - // replica during a failover. This should be input in the form: obj.status. - PersistedStatusItem string `json:"persistedStatusItem,omitempty"` -} +const ( + // ClusterFailover represents the failover is due to cluster issues. + ClusterFailover FailoverReason = "ClusterFailover" + + // ApplicationFailover represents the failover is due to application issues. + ApplicationFailover FailoverReason = "ApplicationFailover" // Failover due to application issues, handled by health interpretation. +) ``` -The PersistedDuringFailover object keeps track of the value of a field during failover so that this information can be used to resume processing from this point onward. This object consists of two fields, LabelName and PersistedStatusItem which are both defined in the propogation policy. +The FailoverHistoryItem object contains information relevant to a failover in addition to the `PreservedLabelState` which will be configured by the tenant within their PropagationPolicy. -#### PropagationPolicy API Change +### PropagationPolicy API -We propose to add two fields to a propogation policy to enable stateful failover. -1. persistedFields.maxHistory: This sets the max limit on the amount of stateful failover history that is persisted before the older entries are overwritten. If this is set to 5, the resourcebinding will store a maximum of 5 failover entries in FailoverHistory before it overwrites the older history. -2. persistedFields.fields: This is a list of the fields that are required by the stateful application to be persisted during failover to resume processing from a particular case. This takes in a list of field names as well as how to access them from the spec. +We propose to add two fields to the propogation policy to enable stateful failover: -Example propagation policy for Flink jobs that uses the persistedFields.maxHistory and persistedFields.fields: +1. `ApplicationFailoverBehavior.MaxHistory`: This sets the max limit on the amount of stateful failover history that is persisted before the older entries are overwritten. If this is set to 5, the resourcebinding will store a maximum of 5 failover entries in FailoverHistory before it overwrites the older history. +2. `ApplicationFailoverBehavior.StatePreservation`: This is a list of the fields that are required by the stateful application to be persisted during failover to resume processing from a particular case. This takes in a list of field names as well as how to access them from the spec. +```go +// ApplicationFailoverBehavior indicates application failover behaviors. +type ApplicationFailoverBehavior struct { + // FailoverHistoryItems will automatically be appended to the ResourceBindingStatus. + // If the default setting of 5 is too excessive, the max history can be configured here. + // + // +optional + MaxHistory int64 `json:"maxHistory,omitempty"` + + // StatePreservation defines the policy for preserving and restoring state data + // during failover events for stateful applications. + // + // When an application fails over from one cluster to another, this policy enables + // the extraction of critical data from the original resource configuration. + // Upon successful migration, the extracted data is then re-injected into the new + // resource, ensuring that the application can resume operation with its previous + // state intact. + // This is particularly useful for stateful applications where maintaining data + // consistency across failover events is crucial. + // If not specified, means no state data will be preserved. + // +optional + StatePreservation *StatePreservation `json:"statePreservation,omitempty"` +} + +// StatePreservation defines the policy for preserving state during failover events. +type StatePreservation struct { + // Rules contains a list of StatePreservationRule configurations. + // Each rule specifies a JSONPath expression targeting specific pieces of + // state data to be preserved during failover events. An AliasLabelName is associated + // with each rule, serving as a label key when the preserved data is passed + // to the new cluster. + // +required + Rules []StatePreservationRule `json:"rules"` + + // Note: We probably need more policies to control how to feed the new cluster with the + // preserved state data in the future. Such as: + // - Is it always acceptable to feed data as the label? Is there a need for annotation? + // - If each label name should be started with a prefix, like `karmada.io/failover-preserving-: ` + // Sure, we can default with the label, this structure just makes room for future extensions. + // + // For instance, we can introduce a policy if someone wants to control how the preserving state + // feed to new clusters. This is probably not included this time. + // RestorePolicy determines when and how the preserved state should be restored. + // RestorePolicy RestorePolicy `json:"restorePolicy"` +} + +// StatePreservationRule defines a single rule for state preservation. +// It includes a JSONPath expression and an alias name that will be used +// as a label key when passing state information to the new cluster. +type StatePreservationRule struct { + // AliasLabelName is the name that will be used as a label key when the preserved + // data is passed to the new cluster. This facilitates the injection of the + // preserved state back into the application resources during recovery. + // +required + AliasLabelName string `json:"aliasLabelName"` + + // JSONPath is the JSONPath template used to identify the state data + // to be preserved from the original resource configuration. + // The JSONPath syntax follows the Kubernetes specification: + // https://kubernetes.io/docs/reference/kubectl/jsonpath/ + // + // Note: The JSONPath expression will start searching from the "status" field of + // the API resource object by default. For example, to extract the "availableReplicas" + // from a Deployment, the JSONPath expression should be "{.availableReplicas}", not + // "{.status.availableReplicas}". + // + // +required + JSONPath string `json:"jsonPath"` +} ``` + +Example propagation policy for Flink jobs that uses the persistedFields.maxHistory and persistedFields.fields: + +```yaml apiVersion: policy.karmada.io/v1alpha1 kind: PropagationPolicy metadata: - name: flinkdep-policy + name: flink-deployment-policy spec: failover: application: decisionConditions: tolerationSeconds: 90 - purgeMode: Graciously - gracePeriodSeconds: 10 - persistedFields.maxHistory: 5 - persistedFields.fields: - - LabelName: jobID - PersistedStatusItem: obj.status.jobStatus.jobID + maxHistory: 5 + purgeMode: Immediately + statePreservation: + rules: + - aliasLabelName: jobId + jsonPath: "obj.status.jobStatus.jobID" resourceSelectors: - apiVersion: flink.apache.org/v1beta1 kind: FlinkDeployment - namespace: <> + namespace: example-namespace propagateDeps: true placement: replicaScheduling: