/
error.go
137 lines (122 loc) · 7.1 KB
/
error.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
// Copyright (c) 2017 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 workflow
import (
"go.uber.org/cadence/.gen/go/shared"
"go.uber.org/cadence/internal"
)
/*
Below are the possible errors that activity or child workflow could return:
1) *workflow.CustomError: (this should be the most common one)
If activity or child workflow implementation returns *CustomError by using NewCustomError() API, workflow code would receive *CustomError.
The err would contain a Reason and Details. The reason is what activity specified to NewCustomError(), which workflow
code could check to determine what kind of error it was and take actions based on the reason. The details is encoded
[]byte which workflow code could extract strong typed data. Workflow code needs to know what the types of the encoded
details are before extracting them.
2) *workflow.GenericError:
If activity or child workflow implementation returns errors other than from NewCustomError() API,
workflow code would receive *GenericError.
Use err.Error() to get the string representation of the actual error.
3) *workflow.CanceledError:
If activity or child workflow was canceled, workflow code will receive instance of *CanceledError.
When activity or child workflow finishes cleanup it can indicate it by returning error created through
NewCancelError() and could supply optional details which could be extracted by workflow code.
4) *workflow.TimeoutError:
If activity or child workflow was timed out (several timeout types), workflow code will receive instance of
*TimeoutError. The err contains details about what type of timeout it was.
5) *workflow.PanicError:
If activity code panics while executing, cadence activity worker will report it as activity failure to cadence server.
The cadence client library will present that failure as *PanicError to workflow code. The err contains a string
representation of the panic message and the call stack when panic was happen.
Note that there should be no *PanicError from child workflow. When panic happen in workflow implementation code,
cadence client library catches that panic and causing the decision timeout. That decision task will be retried at
a later time (with exponential backoff retry intervals). Eventually either decision code is fixed to not panic or
a workflow execution times out. In the timeout case the parent workflow receives TimeoutError.
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(nil)
if err != nil {
switch err := err.(type) {
case *workflowCustomError:
// handle activity errors (created via NewCustomError() API)
switch err.Reason() {
case CustomErrReasonA: // assume CustomErrReasonA is constant defined by activity implementation
var detailMsg string // assuming activity return error by NewCustomError(CustomErrReasonA, "string details")
err.Details(&detailMsg) // extract strong typed details (corresponding to CustomErrReasonA)
// handle CustomErrReasonA
case CustomErrReasonB:
// handle CustomErrReasonB
default:
// newer version of activity could return new errors that workflow was not aware of.
}
case *workflow.GenericError:
// handle generic error (errors created other than using NewCustomError() API)
case *workflow.CanceledError:
// handle cancellation
case *workflow.TimeoutError:
// handle timeout, could check timeout type by err.TimeoutType()
case *workflow.PanicError:
// handle panic
}
}
*/
type (
// GenericError is returned from activity or child workflow when an implementations return error
// other than from workflow.NewCustomError() API.
GenericError = internal.GenericError
// TimeoutError returned when activity or child workflow timed out.
TimeoutError = internal.TimeoutError
// TerminatedError returned when workflow was terminated.
TerminatedError = internal.TerminatedError
// PanicError contains information about panicked workflow/activity.
PanicError = internal.PanicError
// ContinueAsNewError can be returned by a workflow implementation function and indicates that
// the workflow should continue as new with the same WorkflowID, but new RunID and new history.
ContinueAsNewError = internal.ContinueAsNewError
// UnknownExternalWorkflowExecutionError can be returned when external workflow doesn't exist
UnknownExternalWorkflowExecutionError = internal.UnknownExternalWorkflowExecutionError
)
// 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 execution timeout, decision task timeout, task list.
// if not mentioned it would use the defaults that the current workflow is using.
// ctx := WithExecutionStartToCloseTimeout(ctx, 30 * time.Minute)
// ctx := WithWorkflowTaskStartToCloseTimeout(ctx, time.Minute)
// ctx := WithWorkflowTaskList(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{}) *ContinueAsNewError {
return internal.NewContinueAsNewError(ctx, wfn, args...)
}
// NewTimeoutError creates TimeoutError instance.
// Use NewHeartbeatTimeoutError to create heartbeat TimeoutError
// WARNING: This function is public only to support unit testing of workflows.
// It shouldn't be used by application level code.
func NewTimeoutError(timeoutType shared.TimeoutType, details ...interface{}) *TimeoutError {
return internal.NewTimeoutError(timeoutType, details...)
}
// NewHeartbeatTimeoutError creates TimeoutError instance
// WARNING: This function is public only to support unit testing of workflows.
// It shouldn't be used by application level code.
func NewHeartbeatTimeoutError(details ...interface{}) *TimeoutError {
return internal.NewHeartbeatTimeoutError(details...)
}