-
Notifications
You must be signed in to change notification settings - Fork 9
Expand file tree
/
Copy pathsql_value_provider.go
More file actions
85 lines (73 loc) · 2.35 KB
/
sql_value_provider.go
File metadata and controls
85 lines (73 loc) · 2.35 KB
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
package dsunit
import (
"github.com/viant/dsc"
"github.com/viant/toolbox"
"strings"
)
type sequence struct {
seq map[string]int64
}
type sequenceValueProvider struct{}
func (p *sequenceValueProvider) countInsertable(dataset *Dataset) int64 {
var result = 0
for _, row := range dataset.Rows {
for _, pkColumn := range dataset.PkColumns {
value := row.Value(pkColumn)
if textValue, ok := value.(string); ok {
if strings.Contains(textValue, ":seq ") {
result++
}
}
}
}
return int64(result)
}
func (p *sequenceValueProvider) fetchSequence(context toolbox.Context, sequenceName string) (int64, error) {
manager := *context.GetOptional((*dsc.Manager)(nil)).(*dsc.Manager)
dataset := context.GetOptional((*Dataset)(nil)).(*Dataset)
sqlDialectable := *context.GetOptional((*dsc.DatastoreDialect)(nil)).(*dsc.DatastoreDialect)
seq, err := sqlDialectable.GetSequence(manager, sequenceName)
if err != nil {
return 0, err
}
insertableCount := p.countInsertable(dataset)
result := seq - insertableCount
return result, nil
}
func (p *sequenceValueProvider) Get(context toolbox.Context, arguments ...interface{}) (interface{}, error) {
sequenceName := toolbox.AsString(arguments[0])
if !context.Contains((*sequence)(nil)) {
seq, err := p.fetchSequence(context, sequenceName)
if err != nil {
return nil, err
}
var sequenceValue = sequence{seq: make(map[string]int64)}
sequenceValue.seq[sequenceName] = seq
context.Put((*sequence)(nil), &sequenceValue)
}
var sequence = context.GetOptional((*sequence)(nil)).(*sequence)
result := sequence.seq[sequenceName]
sequence.seq[sequenceName]++
return result, nil
}
func newSequenceValueProvider() toolbox.ValueProvider {
var result toolbox.ValueProvider = &sequenceValueProvider{}
return result
}
type queryValueProvider struct{}
func (p *queryValueProvider) Get(context toolbox.Context, arguments ...interface{}) (interface{}, error) {
manager := *context.GetOptional((*dsc.Manager)(nil)).(*dsc.Manager)
sql := toolbox.AsString(arguments[0])
var row = make([]interface{}, 0)
success, err := manager.ReadSingle(&row, sql, nil, nil)
if err != nil {
return nil, dsUnitError{"failed to evalue macro with sql: " + sql + " due to:\n\t" + err.Error()}
}
if !success {
return nil, nil
}
return row[0], nil
}
func newQueryValueProvider() toolbox.ValueProvider {
return &queryValueProvider{}
}