This repository has been archived by the owner on Feb 24, 2023. It is now read-only.
forked from ecodeclub/eorm
-
Notifications
You must be signed in to change notification settings - Fork 0
/
core.go
106 lines (96 loc) · 3.4 KB
/
core.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
// Copyright 2021 gotomicro
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package eorm
import (
"context"
"reflect"
"github.com/gotomicro/eorm/internal/dialect"
"github.com/gotomicro/eorm/internal/errs"
"github.com/gotomicro/eorm/internal/model"
"github.com/gotomicro/eorm/internal/valuer"
)
type core struct {
ms []Middleware
metaRegistry model.MetaRegistry
dialect dialect.Dialect
valCreator valuer.BasicTypeCreator
}
func getHandler[T any](ctx context.Context, sess session, c core, qc *QueryContext) *QueryResult {
rows, err := sess.queryContext(ctx, qc.q.SQL, qc.q.Args...)
if err != nil {
return &QueryResult{Err: err}
}
if !rows.Next() {
return &QueryResult{Err: errs.ErrNoRows}
}
tp := new(T)
meta := qc.meta
if meta == nil && reflect.TypeOf(tp).Elem().Kind() == reflect.Struct {
// 当通过 RawQuery 方法调用 Get ,如果 T 是 time.Time, sql.Scanner 的实现,
// 内置类型或者基本类型时, 在这里都会报错,但是这种情况我们认为是可以接受的
// 所以在此将报错忽略,因为基本类型取值用不到 meta 里的数据
meta, _ = c.metaRegistry.Get(tp)
}
val := c.valCreator.NewBasicTypeValue(tp, meta)
if err = val.SetColumns(rows); err != nil {
return &QueryResult{Err: err}
}
return &QueryResult{Result: tp}
}
func get[T any](ctx context.Context, sess session, core core, qc *QueryContext) *QueryResult {
var handler HandleFunc = func(ctx context.Context, queryContext *QueryContext) *QueryResult {
return getHandler[T](ctx, sess, core, queryContext)
}
ms := core.ms
for i := len(ms) - 1; i >= 0; i-- {
handler = ms[i](handler)
}
return handler(ctx, qc)
}
func getMultiHandler[T any](ctx context.Context, sess session, c core, qc *QueryContext) *QueryResult {
rows, err := sess.queryContext(ctx, qc.q.SQL, qc.q.Args...)
if err != nil {
return &QueryResult{Err: err}
}
res := make([]*T, 0, 16)
meta := qc.meta
if meta == nil {
t := new(T)
if reflect.TypeOf(t).Elem().Kind() == reflect.Struct {
// 当通过 RawQuery 方法调用 Get ,如果 T 是 time.Time, sql.Scanner 的实现,
// 内置类型或者基本类型时, 在这里都会报错,但是这种情况我们认为是可以接受的
// 所以在此将报错忽略,因为基本类型取值用不到 meta 里的数据
meta, _ = c.metaRegistry.Get(t)
}
}
for rows.Next() {
tp := new(T)
val := c.valCreator.NewBasicTypeValue(tp, meta)
if err = val.SetColumns(rows); err != nil {
return &QueryResult{Err: err}
}
res = append(res, tp)
}
return &QueryResult{Result: res}
}
func getMulti[T any](ctx context.Context, sess session, core core, qc *QueryContext) *QueryResult {
var handler HandleFunc = func(ctx context.Context, queryContext *QueryContext) *QueryResult {
return getMultiHandler[T](ctx, sess, core, queryContext)
}
ms := core.ms
for i := len(ms) - 1; i >= 0; i-- {
handler = ms[i](handler)
}
return handler(ctx, qc)
}