peridot/vendor/go.temporal.io/sdk/internal/error.go
2022-07-07 22:13:21 +02:00

970 lines
32 KiB
Go

// The MIT License
//
// Copyright (c) 2020 Temporal Technologies Inc. All rights reserved.
//
// Copyright (c) 2020 Uber Technologies, Inc.
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
package internal
import (
"errors"
"fmt"
"reflect"
"strings"
"time"
commonpb "go.temporal.io/api/common/v1"
enumspb "go.temporal.io/api/enums/v1"
failurepb "go.temporal.io/api/failure/v1"
"go.temporal.io/sdk/converter"
)
/*
If activity fails then *ActivityError is returned to the workflow code. The error has important information about activity
and actual error which caused activity failure. This internal error can be unwrapped using errors.Unwrap() or checked using errors.As().
Below are the possible types of internal error:
1) *ApplicationError: (this should be the most common one)
*ApplicationError can be returned in two cases:
- If activity implementation returns *ApplicationError by using NewApplicationError() API.
The err would contain a message, details, and NonRetryable flag. Workflow code could check this flag and details to determine
what kind of error it was and take actions based on it. The details is encoded payload which workflow code could extract
to strong typed variable. Workflow code needs to know what the types of the encoded details are before extracting them.
- If activity implementation returns errors other than from NewApplicationError() API. In this case GetOriginalType()
will return orginal type of an error represented as string. Workflow code could check this type to determine what kind of error it was
and take actions based on the type. These errors are retryable by default, unless error type is specified in retry policy.
2) *CanceledError:
If activity was canceled, internal error will be an instance of *CanceledError. When activity cancels itself by
returning NewCancelError() it would supply optional details which could be extracted by workflow code.
3) *TimeoutError:
If activity was timed out (several timeout types), internal error will be an instance of *TimeoutError. The err contains
details about what type of timeout it was.
4) *PanicError:
If activity code panic while executing, temporal activity worker will report it as activity failure to temporal server.
The SDK will present that failure as *PanicError. The err contains a string representation of the panic message and
the call stack when panic was happen.
Workflow code could handle errors based on different types of error. Below is sample code of how error handling looks like.
err := workflow.ExecuteActivity(ctx, MyActivity, ...).Get(ctx, nil)
if err != nil {
var applicationErr *ApplicationError
if errors.As(err, &applicationError) {
// handle activity errors (created via NewApplicationError() API)
if !applicationErr.NonRetryable() {
// manually retry activity
}
var detailMsg string // assuming activity return error by NewApplicationError("message", true, "string details")
applicationErr.Details(&detailMsg) // extract strong typed details
// handle activity errors (errors created other than using NewApplicationError() API)
switch err.Type() {
case "CustomErrTypeA":
// handle CustomErrTypeA
case CustomErrTypeB:
// handle CustomErrTypeB
default:
// newer version of activity could return new errors that workflow was not aware of.
}
}
var canceledErr *CanceledError
if errors.As(err, &canceledErr) {
// handle cancellation
}
var timeoutErr *TimeoutError
if errors.As(err, &timeoutErr) {
// handle timeout, could check timeout type by timeoutErr.TimeoutType()
switch err.TimeoutType() {
case commonpb.ScheduleToStart:
// Handle ScheduleToStart timeout.
case commonpb.StartToClose:
// Handle StartToClose timeout.
case commonpb.Heartbeat:
// Handle heartbeat timeout.
default:
}
}
var panicErr *PanicError
if errors.As(err, &panicErr) {
// handle panic, message and stack trace are available by panicErr.Error() and panicErr.StackTrace()
}
}
Errors from child workflow should be handled in a similar way, except that instance of *ChildWorkflowExecutionError is returned to
workflow code. It will contains *ActivityError, which in turn will contains on of the errors above.
When panic happen in workflow implementation code, SDK catches that panic and causing the workflow task timeout.
That workflow task will be retried at a later time (with exponential backoff retry intervals).
Workflow consumers will get an instance of *WorkflowExecutionError. This error will contains one of errors above.
*/
type (
// ApplicationError returned from activity implementations with message and optional details.
ApplicationError struct {
temporalError
msg string
errType string
nonRetryable bool
cause error
details converter.EncodedValues
}
// TimeoutError returned when activity or child workflow timed out.
TimeoutError struct {
temporalError
msg string
timeoutType enumspb.TimeoutType
lastHeartbeatDetails converter.EncodedValues
cause error
}
// CanceledError returned when operation was canceled.
CanceledError struct {
temporalError
details converter.EncodedValues
}
// TerminatedError returned when workflow was terminated.
TerminatedError struct {
temporalError
}
// PanicError contains information about panicked workflow/activity.
PanicError struct {
temporalError
value interface{}
stackTrace string
}
// workflowPanicError contains information about panicked workflow.
// Used to distinguish go panic in the workflow code from a PanicError returned from a workflow function.
workflowPanicError struct {
value interface{}
stackTrace string
}
// ContinueAsNewError contains information about how to continue the workflow as new.
ContinueAsNewError struct {
//params *ExecuteWorkflowParams
WorkflowType *WorkflowType
Input *commonpb.Payloads
Header *commonpb.Header
TaskQueueName string
WorkflowExecutionTimeout time.Duration
WorkflowRunTimeout time.Duration
WorkflowTaskTimeout time.Duration
}
// UnknownExternalWorkflowExecutionError can be returned when external workflow doesn't exist
UnknownExternalWorkflowExecutionError struct{}
// ServerError can be returned from server.
ServerError struct {
temporalError
msg string
nonRetryable bool
cause error
}
// ActivityError is returned from workflow when activity returned an error.
// Unwrap this error to get actual cause.
ActivityError struct {
temporalError
scheduledEventID int64
startedEventID int64
identity string
activityType *commonpb.ActivityType
activityID string
retryState enumspb.RetryState
cause error
}
// ChildWorkflowExecutionError is returned from workflow when child workflow returned an error.
// Unwrap this error to get actual cause.
ChildWorkflowExecutionError struct {
temporalError
namespace string
workflowID string
runID string
workflowType string
initiatedEventID int64
startedEventID int64
retryState enumspb.RetryState
cause error
}
// ChildWorkflowExecutionAlreadyStartedError is set as the cause of
// ChildWorkflowExecutionError when failure is due the child workflow having
// already started.
ChildWorkflowExecutionAlreadyStartedError struct{}
// WorkflowExecutionError is returned from workflow.
// Unwrap this error to get actual cause.
WorkflowExecutionError struct {
workflowID string
runID string
workflowType string
cause error
}
// ActivityNotRegisteredError is returned if worker doesn't support activity type.
ActivityNotRegisteredError struct {
activityType string
supportedTypes []string
}
temporalError struct {
messenger
originalFailure *failurepb.Failure
}
failureHolder interface {
setFailure(*failurepb.Failure)
failure() *failurepb.Failure
}
messenger interface {
message() string
}
)
var (
// Should be "errorString".
goErrType = reflect.TypeOf(errors.New("")).Elem().Name()
// ErrNoData is returned when trying to extract strong typed data while there is no data available.
ErrNoData = errors.New("no data available")
// ErrTooManyArg is returned when trying to extract strong typed data with more arguments than available data.
ErrTooManyArg = errors.New("too many arguments")
// ErrActivityResultPending is returned from activity's implementation to indicate the activity is not completed when
// activity method returns. Activity needs to be completed by Client.CompleteActivity() separately. For example, if an
// activity require human interaction (like approve an expense report), the activity could return activity.ErrResultPending
// which indicate the activity is not done yet. Then, when the waited human action happened, it needs to trigger something
// that could report the activity completed event to temporal server via Client.CompleteActivity() API.
ErrActivityResultPending = errors.New("not error: do not autocomplete, using Client.CompleteActivity() to complete")
)
// NewApplicationError create new instance of *ApplicationError with message, type, and optional details.
func NewApplicationError(msg string, errType string, nonRetryable bool, cause error, details ...interface{}) error {
applicationErr := &ApplicationError{
msg: msg,
errType: errType,
nonRetryable: nonRetryable,
cause: cause}
// When return error to user, use EncodedValues as details and data is ready to be decoded by calling Get
if len(details) == 1 {
if d, ok := details[0].(*EncodedValues); ok {
applicationErr.details = d
return applicationErr
}
}
// When create error for server, use ErrorDetailsValues as details to hold values and encode later
applicationErr.details = ErrorDetailsValues(details)
return applicationErr
}
// NewTimeoutError creates TimeoutError instance.
// Use NewHeartbeatTimeoutError to create heartbeat TimeoutError.
func NewTimeoutError(msg string, timeoutType enumspb.TimeoutType, cause error, lastHeartbeatDetails ...interface{}) error {
timeoutErr := &TimeoutError{
msg: msg,
timeoutType: timeoutType,
cause: cause,
}
if len(lastHeartbeatDetails) == 1 {
if d, ok := lastHeartbeatDetails[0].(*EncodedValues); ok {
timeoutErr.lastHeartbeatDetails = d
return timeoutErr
}
}
timeoutErr.lastHeartbeatDetails = ErrorDetailsValues(lastHeartbeatDetails)
return timeoutErr
}
// NewHeartbeatTimeoutError creates TimeoutError instance.
func NewHeartbeatTimeoutError(details ...interface{}) error {
return NewTimeoutError("heartbeat timeout", enumspb.TIMEOUT_TYPE_HEARTBEAT, nil, details...)
}
// NewCanceledError creates CanceledError instance.
func NewCanceledError(details ...interface{}) error {
if len(details) == 1 {
if d, ok := details[0].(*EncodedValues); ok {
return &CanceledError{details: d}
}
}
return &CanceledError{details: ErrorDetailsValues(details)}
}
// NewServerError create new instance of *ServerError with message.
func NewServerError(msg string, nonRetryable bool, cause error) error {
return &ServerError{msg: msg, nonRetryable: nonRetryable, cause: cause}
}
// NewActivityError creates ActivityError instance.
func NewActivityError(
scheduledEventID int64,
startedEventID int64,
identity string,
activityType *commonpb.ActivityType,
activityID string,
retryState enumspb.RetryState,
cause error,
) *ActivityError {
return &ActivityError{
scheduledEventID: scheduledEventID,
startedEventID: startedEventID,
identity: identity,
activityType: activityType,
activityID: activityID,
retryState: retryState,
cause: cause,
}
}
// NewChildWorkflowExecutionError creates ChildWorkflowExecutionError instance.
func NewChildWorkflowExecutionError(
namespace string,
workflowID string,
runID string,
workflowType string,
initiatedEventID int64,
startedEventID int64,
retryState enumspb.RetryState,
cause error,
) *ChildWorkflowExecutionError {
return &ChildWorkflowExecutionError{
namespace: namespace,
workflowID: workflowID,
runID: runID,
workflowType: workflowType,
initiatedEventID: initiatedEventID,
startedEventID: startedEventID,
retryState: retryState,
cause: cause,
}
}
// NewWorkflowExecutionError creates WorkflowExecutionError instance.
func NewWorkflowExecutionError(
workflowID string,
runID string,
workflowType string,
cause error,
) *WorkflowExecutionError {
return &WorkflowExecutionError{
workflowID: workflowID,
runID: runID,
workflowType: workflowType,
cause: cause,
}
}
func (e *temporalError) setFailure(f *failurepb.Failure) {
e.originalFailure = f
}
func (e *temporalError) failure() *failurepb.Failure {
return e.originalFailure
}
// IsCanceledError returns whether error in CanceledError.
func IsCanceledError(err error) bool {
var canceledErr *CanceledError
return errors.As(err, &canceledErr)
}
// NewContinueAsNewError creates ContinueAsNewError instance
// If the workflow main function returns this error then the current execution is ended and
// the new execution with same workflow ID is started automatically with options
// provided to this function.
// ctx - use context to override any options for the new workflow like run timeout, task timeout, task queue.
// if not mentioned it would use the defaults that the current workflow is using.
// ctx := WithWorkflowRunTimeout(ctx, 30 * time.Minute)
// ctx := WithWorkflowTaskTimeout(ctx, 5 * time.Second)
// ctx := WithWorkflowTaskQueue(ctx, "example-group")
// wfn - workflow function. for new execution it can be different from the currently running.
// args - arguments for the new workflow.
//
func NewContinueAsNewError(ctx Context, wfn interface{}, args ...interface{}) error {
i := getWorkflowOutboundInterceptor(ctx)
// Put header on context before executing
ctx = workflowContextWithNewHeader(ctx)
return i.NewContinueAsNewError(ctx, wfn, args...)
}
func (wc *workflowEnvironmentInterceptor) NewContinueAsNewError(
ctx Context,
wfn interface{},
args ...interface{},
) error {
// Validate type and its arguments.
options := getWorkflowEnvOptions(ctx)
if options == nil {
panic("context is missing required options for continue as new")
}
env := getWorkflowEnvironment(ctx)
workflowType, input, err := getValidatedWorkflowFunction(wfn, args, options.DataConverter, env.GetRegistry())
if err != nil {
panic(err)
}
header, err := workflowHeaderPropagated(ctx, options.ContextPropagators)
if err != nil {
return err
}
return &ContinueAsNewError{
WorkflowType: workflowType,
Input: input,
Header: header,
TaskQueueName: options.TaskQueueName,
WorkflowExecutionTimeout: options.WorkflowExecutionTimeout,
WorkflowRunTimeout: options.WorkflowRunTimeout,
WorkflowTaskTimeout: options.WorkflowTaskTimeout,
}
}
// NewActivityNotRegisteredError creates a new ActivityNotRegisteredError.
func NewActivityNotRegisteredError(activityType string, supportedTypes []string) error {
return &ActivityNotRegisteredError{activityType: activityType, supportedTypes: supportedTypes}
}
// Error from error interface.
func (e *ApplicationError) Error() string {
msg := e.message()
if e.errType != "" {
msg = fmt.Sprintf("%s (type: %s, retryable: %v)", msg, e.errType, !e.nonRetryable)
}
if e.cause != nil {
msg = fmt.Sprintf("%s: %v", msg, e.cause)
}
return msg
}
func (e *ApplicationError) message() string {
return e.msg
}
// Type returns error type represented as string.
// This type can be passed explicitly to ApplicationError constructor.
// Also any other Go error is converted to ApplicationError and type is set automatically using reflection.
// For example instance of "MyCustomError struct" will be converted to ApplicationError and Type() will return "MyCustomError" string.
func (e *ApplicationError) Type() string {
return e.errType
}
// HasDetails return if this error has strong typed detail data.
func (e *ApplicationError) HasDetails() bool {
return e.details != nil && e.details.HasValues()
}
// Details extracts strong typed detail data of this custom error. If there is no details, it will return ErrNoData.
func (e *ApplicationError) Details(d ...interface{}) error {
if !e.HasDetails() {
return ErrNoData
}
return e.details.Get(d...)
}
// NonRetryable indicated if error is not retryable.
func (e *ApplicationError) NonRetryable() bool {
return e.nonRetryable
}
func (e *ApplicationError) Unwrap() error {
return e.cause
}
// Error from error interface
func (e *TimeoutError) Error() string {
msg := fmt.Sprintf("%s (type: %v)", e.message(), e.timeoutType)
if e.cause != nil {
msg = fmt.Sprintf("%s: %v", msg, e.cause)
}
return msg
}
func (e *TimeoutError) message() string {
return e.msg
}
func (e *TimeoutError) Unwrap() error {
return e.cause
}
// TimeoutType return timeout type of this error
func (e *TimeoutError) TimeoutType() enumspb.TimeoutType {
return e.timeoutType
}
// HasLastHeartbeatDetails return if this error has strong typed detail data.
func (e *TimeoutError) HasLastHeartbeatDetails() bool {
return e.lastHeartbeatDetails != nil && e.lastHeartbeatDetails.HasValues()
}
// LastHeartbeatDetails extracts strong typed detail data of this error. If there is no details, it will return ErrNoData.
func (e *TimeoutError) LastHeartbeatDetails(d ...interface{}) error {
if !e.HasLastHeartbeatDetails() {
return ErrNoData
}
return e.lastHeartbeatDetails.Get(d...)
}
// Error from error interface
func (e *CanceledError) Error() string {
return e.message()
}
func (e *CanceledError) message() string {
return "canceled"
}
// HasDetails return if this error has strong typed detail data.
func (e *CanceledError) HasDetails() bool {
return e.details != nil && e.details.HasValues()
}
// Details extracts strong typed detail data of this error.
func (e *CanceledError) Details(d ...interface{}) error {
if !e.HasDetails() {
return ErrNoData
}
return e.details.Get(d...)
}
func newPanicError(value interface{}, stackTrace string) error {
return &PanicError{value: value, stackTrace: stackTrace}
}
func newWorkflowPanicError(value interface{}, stackTrace string) error {
return &workflowPanicError{value: value, stackTrace: stackTrace}
}
// Error from error interface
func (e *PanicError) Error() string {
return e.message()
}
func (e *PanicError) message() string {
return fmt.Sprintf("%v", e.value)
}
// StackTrace return stack trace of the panic
func (e *PanicError) StackTrace() string {
return e.stackTrace
}
// Error from error interface
func (e *workflowPanicError) Error() string {
return fmt.Sprintf("%v", e.value)
}
// StackTrace return stack trace of the panic
func (e *workflowPanicError) StackTrace() string {
return e.stackTrace
}
// Error from error interface
func (e *ContinueAsNewError) Error() string {
return e.message()
}
func (e *ContinueAsNewError) message() string {
return "continue as new"
}
// newTerminatedError creates NewTerminatedError instance
func newTerminatedError() *TerminatedError {
return &TerminatedError{}
}
// Error from error interface
func (e *TerminatedError) Error() string {
return e.message()
}
func (e *TerminatedError) message() string {
return "terminated"
}
// newUnknownExternalWorkflowExecutionError creates UnknownExternalWorkflowExecutionError instance
func newUnknownExternalWorkflowExecutionError() *UnknownExternalWorkflowExecutionError {
return &UnknownExternalWorkflowExecutionError{}
}
// Error from error interface
func (e *UnknownExternalWorkflowExecutionError) Error() string {
return "unknown external workflow execution"
}
// Error from error interface
func (e *ServerError) Error() string {
msg := e.message()
if e.cause != nil {
msg = fmt.Sprintf("%s: %v", msg, e.cause)
}
return msg
}
func (e *ServerError) message() string {
return e.msg
}
func (e *ServerError) Unwrap() error {
return e.cause
}
func (e *ActivityError) Error() string {
msg := fmt.Sprintf("%s (type: %s, scheduledEventID: %d, startedEventID: %d, identity: %s)", e.message(), e.activityType.GetName(), e.scheduledEventID, e.startedEventID, e.identity)
if e.cause != nil {
msg = fmt.Sprintf("%s: %v", msg, e.cause)
}
return msg
}
func (e *ActivityError) message() string {
return "activity error"
}
func (e *ActivityError) Unwrap() error {
return e.cause
}
// ScheduledEventID returns event id of the scheduled workflow task corresponding to the activity.
func (e *ActivityError) ScheduledEventID() int64 {
return e.scheduledEventID
}
// StartedEventID returns event id of the started workflow task corresponding to the activity.
func (e *ActivityError) StartedEventID() int64 {
return e.startedEventID
}
// Identity returns identity of the worker that attempted activity execution.
func (e *ActivityError) Identity() string {
return e.identity
}
// ActivityType returns declared type of the activity.
func (e *ActivityError) ActivityType() *commonpb.ActivityType {
return e.activityType
}
// ActivityID return assigned identifier for the activity.
func (e *ActivityError) ActivityID() string {
return e.activityID
}
// RetryState returns details on why activity failed.
func (e *ActivityError) RetryState() enumspb.RetryState {
return e.retryState
}
// Error from error interface
func (e *ChildWorkflowExecutionError) Error() string {
msg := fmt.Sprintf("%s (type: %s, workflowID: %s, runID: %s, initiatedEventID: %d, startedEventID: %d)",
e.message(), e.workflowType, e.workflowID, e.runID, e.initiatedEventID, e.startedEventID)
if e.cause != nil {
msg = fmt.Sprintf("%s: %v", msg, e.cause)
}
return msg
}
func (e *ChildWorkflowExecutionError) message() string {
return "child workflow execution error"
}
func (e *ChildWorkflowExecutionError) Unwrap() error {
return e.cause
}
// Error from error interface
func (*ChildWorkflowExecutionAlreadyStartedError) Error() string {
return "child workflow execution already started"
}
// Error from error interface
func (e *WorkflowExecutionError) Error() string {
msg := fmt.Sprintf("workflow execution error (type: %s, workflowID: %s, runID: %s)",
e.workflowType, e.workflowID, e.runID)
if e.cause != nil {
msg = fmt.Sprintf("%s: %v", msg, e.cause)
}
return msg
}
func (e *WorkflowExecutionError) Unwrap() error {
return e.cause
}
func (e *ActivityNotRegisteredError) Error() string {
supported := strings.Join(e.supportedTypes, ", ")
return fmt.Sprintf("unable to find activityType=%v. Supported types: [%v]", e.activityType, supported)
}
func convertErrDetailsToPayloads(details converter.EncodedValues, dc converter.DataConverter) *commonpb.Payloads {
switch d := details.(type) {
case ErrorDetailsValues:
data, err := encodeArgs(dc, d)
if err != nil {
panic(err)
}
return data
case *EncodedValues:
return d.values
default:
panic(fmt.Sprintf("unknown error details type %T", details))
}
}
// IsRetryable returns if error retryable or not.
func IsRetryable(err error, nonRetryableTypes []string) bool {
if err == nil {
return false
}
var terminatedErr *TerminatedError
var canceledErr *CanceledError
var workflowPanicErr *workflowPanicError
if errors.As(err, &terminatedErr) || errors.As(err, &canceledErr) || errors.As(err, &workflowPanicErr) {
return false
}
var timeoutErr *TimeoutError
if errors.As(err, &timeoutErr) {
return timeoutErr.timeoutType == enumspb.TIMEOUT_TYPE_START_TO_CLOSE || timeoutErr.timeoutType == enumspb.TIMEOUT_TYPE_HEARTBEAT
}
var applicationErr *ApplicationError
var errType string
if errors.As(err, &applicationErr) {
if applicationErr.nonRetryable {
return false
}
errType = applicationErr.errType
} else {
// If it is generic Go error.
errType = getErrType(err)
}
for _, nonRetryableType := range nonRetryableTypes {
if nonRetryableType == errType {
return false
}
}
return true
}
func getErrType(err error) string {
var t reflect.Type
for t = reflect.TypeOf(err); t.Kind() == reflect.Ptr; t = t.Elem() {
}
if t.Name() == goErrType {
return ""
}
return t.Name()
}
// ConvertErrorToFailure converts error to failure.
func ConvertErrorToFailure(err error, dc converter.DataConverter) *failurepb.Failure {
if err == nil {
return nil
}
if fh, ok := err.(failureHolder); ok {
if fh.failure() != nil {
return fh.failure()
}
}
failure := &failurepb.Failure{
Source: "GoSDK",
}
if m, ok := err.(messenger); ok && m != nil {
failure.Message = m.message()
} else {
failure.Message = err.Error()
}
switch err := err.(type) {
case *ApplicationError:
failureInfo := &failurepb.ApplicationFailureInfo{
Type: err.errType,
NonRetryable: err.nonRetryable,
Details: convertErrDetailsToPayloads(err.details, dc),
}
failure.FailureInfo = &failurepb.Failure_ApplicationFailureInfo{ApplicationFailureInfo: failureInfo}
case *CanceledError:
failureInfo := &failurepb.CanceledFailureInfo{
Details: convertErrDetailsToPayloads(err.details, dc),
}
failure.FailureInfo = &failurepb.Failure_CanceledFailureInfo{CanceledFailureInfo: failureInfo}
case *PanicError:
failureInfo := &failurepb.ApplicationFailureInfo{
Type: getErrType(err),
}
failure.FailureInfo = &failurepb.Failure_ApplicationFailureInfo{ApplicationFailureInfo: failureInfo}
failure.StackTrace = err.StackTrace()
case *workflowPanicError:
failureInfo := &failurepb.ApplicationFailureInfo{
Type: getErrType(&PanicError{}),
NonRetryable: true,
}
failure.FailureInfo = &failurepb.Failure_ApplicationFailureInfo{ApplicationFailureInfo: failureInfo}
failure.StackTrace = err.StackTrace()
case *TimeoutError:
failureInfo := &failurepb.TimeoutFailureInfo{
TimeoutType: err.timeoutType,
LastHeartbeatDetails: convertErrDetailsToPayloads(err.lastHeartbeatDetails, dc),
}
failure.FailureInfo = &failurepb.Failure_TimeoutFailureInfo{TimeoutFailureInfo: failureInfo}
case *TerminatedError:
failureInfo := &failurepb.TerminatedFailureInfo{}
failure.FailureInfo = &failurepb.Failure_TerminatedFailureInfo{TerminatedFailureInfo: failureInfo}
case *ServerError:
failureInfo := &failurepb.ServerFailureInfo{
NonRetryable: err.nonRetryable,
}
failure.FailureInfo = &failurepb.Failure_ServerFailureInfo{ServerFailureInfo: failureInfo}
case *ActivityError:
failureInfo := &failurepb.ActivityFailureInfo{
ScheduledEventId: err.scheduledEventID,
StartedEventId: err.startedEventID,
Identity: err.identity,
ActivityType: err.activityType,
ActivityId: err.activityID,
RetryState: err.retryState,
}
failure.FailureInfo = &failurepb.Failure_ActivityFailureInfo{ActivityFailureInfo: failureInfo}
case *ChildWorkflowExecutionError:
failureInfo := &failurepb.ChildWorkflowExecutionFailureInfo{
Namespace: err.namespace,
WorkflowExecution: &commonpb.WorkflowExecution{
WorkflowId: err.workflowID,
RunId: err.runID,
},
WorkflowType: &commonpb.WorkflowType{Name: err.workflowType},
InitiatedEventId: err.initiatedEventID,
StartedEventId: err.startedEventID,
RetryState: err.retryState,
}
failure.FailureInfo = &failurepb.Failure_ChildWorkflowExecutionFailureInfo{ChildWorkflowExecutionFailureInfo: failureInfo}
default: // All unknown errors are considered to be retryable ApplicationFailureInfo.
failureInfo := &failurepb.ApplicationFailureInfo{
Type: getErrType(err),
NonRetryable: false,
}
failure.FailureInfo = &failurepb.Failure_ApplicationFailureInfo{ApplicationFailureInfo: failureInfo}
}
failure.Cause = ConvertErrorToFailure(errors.Unwrap(err), dc)
return failure
}
// ConvertFailureToError converts failure to error.
func ConvertFailureToError(failure *failurepb.Failure, dc converter.DataConverter) error {
if failure == nil {
return nil
}
var err error
if failure.GetApplicationFailureInfo() != nil {
applicationFailureInfo := failure.GetApplicationFailureInfo()
details := newEncodedValues(applicationFailureInfo.GetDetails(), dc)
switch applicationFailureInfo.GetType() {
case getErrType(&PanicError{}):
err = newPanicError(failure.GetMessage(), failure.GetStackTrace())
default:
err = NewApplicationError(
failure.GetMessage(),
applicationFailureInfo.GetType(),
applicationFailureInfo.GetNonRetryable(),
ConvertFailureToError(failure.GetCause(), dc),
details)
}
} else if failure.GetCanceledFailureInfo() != nil {
details := newEncodedValues(failure.GetCanceledFailureInfo().GetDetails(), dc)
err = NewCanceledError(details)
} else if failure.GetTimeoutFailureInfo() != nil {
timeoutFailureInfo := failure.GetTimeoutFailureInfo()
lastHeartbeatDetails := newEncodedValues(timeoutFailureInfo.GetLastHeartbeatDetails(), dc)
err = NewTimeoutError(
failure.GetMessage(),
timeoutFailureInfo.GetTimeoutType(),
ConvertFailureToError(failure.GetCause(), dc),
lastHeartbeatDetails)
} else if failure.GetTerminatedFailureInfo() != nil {
err = newTerminatedError()
} else if failure.GetServerFailureInfo() != nil {
err = NewServerError(failure.GetMessage(), failure.GetServerFailureInfo().GetNonRetryable(), ConvertFailureToError(failure.GetCause(), dc))
} else if failure.GetResetWorkflowFailureInfo() != nil {
err = NewApplicationError(failure.GetMessage(), "", true, ConvertFailureToError(failure.GetCause(), dc), failure.GetResetWorkflowFailureInfo().GetLastHeartbeatDetails())
} else if failure.GetActivityFailureInfo() != nil {
activityTaskInfoFailure := failure.GetActivityFailureInfo()
err = NewActivityError(
activityTaskInfoFailure.GetScheduledEventId(),
activityTaskInfoFailure.GetStartedEventId(),
activityTaskInfoFailure.GetIdentity(),
activityTaskInfoFailure.GetActivityType(),
activityTaskInfoFailure.GetActivityId(),
activityTaskInfoFailure.GetRetryState(),
ConvertFailureToError(failure.GetCause(), dc),
)
} else if failure.GetChildWorkflowExecutionFailureInfo() != nil {
childWorkflowExecutionFailureInfo := failure.GetChildWorkflowExecutionFailureInfo()
err = NewChildWorkflowExecutionError(
childWorkflowExecutionFailureInfo.GetNamespace(),
childWorkflowExecutionFailureInfo.GetWorkflowExecution().GetWorkflowId(),
childWorkflowExecutionFailureInfo.GetWorkflowExecution().GetRunId(),
childWorkflowExecutionFailureInfo.GetWorkflowType().GetName(),
childWorkflowExecutionFailureInfo.GetInitiatedEventId(),
childWorkflowExecutionFailureInfo.GetStartedEventId(),
childWorkflowExecutionFailureInfo.GetRetryState(),
ConvertFailureToError(failure.GetCause(), dc),
)
}
if err == nil {
// All unknown types are considered to be retryable ApplicationError.
err = NewApplicationError(failure.GetMessage(), "", false, ConvertFailureToError(failure.GetCause(), dc))
}
if fh, ok := err.(failureHolder); ok {
fh.setFailure(failure)
}
return err
}