forked from databricks/terraform-provider-databricks
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcommands.go
147 lines (128 loc) · 3.92 KB
/
commands.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
138
139
140
141
142
143
144
145
146
147
package common
import (
"context"
"errors"
"html"
"regexp"
"strings"
)
var (
// IPython's output prefixes
outRE = regexp.MustCompile(`Out\[[\d\s]+\]:\s`)
// HTML tags
tagRE = regexp.MustCompile(`<[^>]*>`)
// just exception content without exception name
exceptionRE = regexp.MustCompile(`.*Exception:\s+(.*)`)
// execution errors resulting from http errors are sometimes hidden in these keys
executionErrorRE = regexp.MustCompile(`ExecutionError: ([\s\S]*)\n(StatusCode=[0-9]*)\n(StatusDescription=.*)\n`)
// usual error message explanation is hidden in this key
errorMessageRE = regexp.MustCompile(`ErrorMessage=(.+)\n`)
)
// WithCommandMock mocks all command executions for this client
func (c *DatabricksClient) WithCommandMock(mock CommandMock) {
c.WithCommandExecutor(func(_ context.Context, _ *DatabricksClient) CommandExecutor {
return commandExecutorMock{
mock: mock,
}
})
}
// WithCommandExecutor sets command executor implementation to use
func (c *DatabricksClient) WithCommandExecutor(cef func(context.Context, *DatabricksClient) CommandExecutor) {
c.commandFactory = cef
}
// CommandExecutor service
func (c *DatabricksClient) CommandExecutor(ctx context.Context) CommandExecutor {
return c.commandFactory(ctx, c)
}
// CommandMock mocks the execution of command
type CommandMock func(commandStr string) CommandResults
// CommandExecutorMock simplifies command testing
type commandExecutorMock struct {
mock CommandMock
}
// Execute mock command with given mock function
func (c commandExecutorMock) Execute(clusterID, language, commandStr string) CommandResults {
return c.mock(commandStr)
}
// CommandExecutor creates a spark context and executes a command and then closes context
type CommandExecutor interface {
Execute(clusterID, language, commandStr string) CommandResults
}
// CommandResults captures results of a command
type CommandResults struct {
ResultType string `json:"resultType,omitempty"`
Summary string `json:"summary,omitempty"`
Cause string `json:"cause,omitempty"`
Data any `json:"data,omitempty"`
Schema any `json:"schema,omitempty"`
Truncated bool `json:"truncated,omitempty"`
IsJSONSchema bool `json:"isJsonSchema,omitempty"`
pos int
}
// Failed tells if command execution failed
func (cr *CommandResults) Failed() bool {
return cr.ResultType == "error"
}
// Text returns plain text results
func (cr *CommandResults) Text() string {
if cr.ResultType != "text" {
return ""
}
return outRE.ReplaceAllLiteralString(cr.Data.(string), "")
}
// Err returns error type
func (cr *CommandResults) Err() error {
if !cr.Failed() {
return nil
}
return errors.New(cr.Error())
}
// Error returns error in a bit more friendly way
func (cr *CommandResults) Error() string {
if cr.ResultType != "error" {
return ""
}
summary := tagRE.ReplaceAllLiteralString(cr.Summary, "")
summary = html.UnescapeString(summary)
exceptionMatches := exceptionRE.FindStringSubmatch(summary)
if len(exceptionMatches) == 2 {
summary = strings.ReplaceAll(exceptionMatches[1], "; nested exception is:", "")
summary = strings.TrimRight(summary, " ")
return summary
}
executionErrorMatches := executionErrorRE.FindStringSubmatch(cr.Cause)
if len(executionErrorMatches) == 4 {
return strings.Join(executionErrorMatches[1:], "\n")
}
errorMessageMatches := errorMessageRE.FindStringSubmatch(cr.Cause)
if len(errorMessageMatches) == 2 {
return errorMessageMatches[1]
}
return summary
}
// Scan scans for results
func (cr *CommandResults) Scan(dest ...any) bool {
if cr.ResultType != "table" {
return false
}
if rows, ok := cr.Data.([]any); ok {
if cr.pos >= len(rows) {
return false
}
if cols, ok := rows[cr.pos].([]any); ok {
for i := range dest {
switch d := dest[i].(type) {
case *string:
*d = cols[i].(string)
case *int:
*d = cols[i].(int)
case *bool:
*d = cols[i].(bool)
}
}
cr.pos++
return true
}
}
return false
}