|
- /*
- * Licensed to the Apache Software Foundation (ASF) under one or more
- * contributor license agreements. See the NOTICE file distributed with
- * this work for additional information regarding copyright ownership.
- * The ASF licenses this file to You 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 exec
-
- import (
- "context"
- "database/sql/driver"
-
- "github.com/seata/seata-go/pkg/datasource/sql/parser"
- "github.com/seata/seata-go/pkg/datasource/sql/types"
- "github.com/seata/seata-go/pkg/datasource/sql/undo"
- "github.com/seata/seata-go/pkg/datasource/sql/undo/builder"
- )
-
- func init() {
- undo.RegisterUndoLogBuilder(types.MultiExecutor, builder.GetMySQLMultiUndoLogBuilder)
- }
-
- var (
- atExecutors = make(map[types.DBType]func() SQLExecutor)
- xaExecutors = make(map[types.DBType]func() SQLExecutor)
- )
-
- // RegisterATExecutor AT executor
- func RegisterATExecutor(dt types.DBType, builder func() SQLExecutor) {
- atExecutors[dt] = builder
- }
-
- // RegisterXAExecutor XA executor
- func RegisterXAExecutor(dt types.DBType, builder func() SQLExecutor) {
- xaExecutors[dt] = func() SQLExecutor {
- return builder()
- }
- }
-
- type (
- CallbackWithNamedValue func(ctx context.Context, query string, args []driver.NamedValue) (types.ExecResult, error)
-
- CallbackWithValue func(ctx context.Context, query string, args []driver.Value) (types.ExecResult, error)
-
- SQLExecutor interface {
- Interceptors(interceptors []SQLHook)
- ExecWithNamedValue(ctx context.Context, execCtx *types.ExecContext, f CallbackWithNamedValue) (types.ExecResult, error)
- ExecWithValue(ctx context.Context, execCtx *types.ExecContext, f CallbackWithNamedValue) (types.ExecResult, error)
- }
- )
-
- // BuildExecutor use db type and transaction type to build an executor. the executor can
- // add custom hook, and intercept the user's business sql to generate the undo log.
- func BuildExecutor(dbType types.DBType, transactionMode types.TransactionMode, query string) (SQLExecutor, error) {
- parseContext, err := parser.DoParser(query)
- if err != nil {
- return nil, err
- }
-
- hooks := make([]SQLHook, 0, 4)
- hooks = append(hooks, commonHook...)
- hooks = append(hooks, hookSolts[parseContext.SQLType]...)
-
- if transactionMode == types.XAMode {
- e := xaExecutors[dbType]()
- e.Interceptors(hooks)
- return e, nil
- }
-
- e := atExecutors[dbType]()
- e.Interceptors(hooks)
- return e, nil
- }
-
- type BaseExecutor struct {
- hooks []SQLHook
- ex SQLExecutor
- }
-
- // Interceptors
- func (e *BaseExecutor) Interceptors(interceptors []SQLHook) {
- e.hooks = interceptors
- }
-
- // ExecWithNamedValue
- func (e *BaseExecutor) ExecWithNamedValue(ctx context.Context, execCtx *types.ExecContext, f CallbackWithNamedValue) (types.ExecResult, error) {
- for i := range e.hooks {
- e.hooks[i].Before(ctx, execCtx)
- }
-
- defer func() {
- for i := range e.hooks {
- e.hooks[i].After(ctx, execCtx)
- }
- }()
-
- if e.ex != nil {
- return e.ex.ExecWithNamedValue(ctx, execCtx, f)
- }
-
- return f(ctx, execCtx.Query, execCtx.NamedValues)
- }
-
- // ExecWithValue
- func (e *BaseExecutor) ExecWithValue(ctx context.Context, execCtx *types.ExecContext, f CallbackWithNamedValue) (types.ExecResult, error) {
- for i := range e.hooks {
- e.hooks[i].Before(ctx, execCtx)
- }
-
- defer func() {
- for i := range e.hooks {
- e.hooks[i].After(ctx, execCtx)
- }
- }()
-
- if e.ex != nil {
- return e.ex.ExecWithValue(ctx, execCtx, f)
- }
-
- nvargs := make([]driver.NamedValue, len(execCtx.Values))
- for i, value := range execCtx.Values {
- nvargs = append(nvargs, driver.NamedValue{
- Value: value,
- Ordinal: i,
- })
- }
-
- return f(ctx, execCtx.Query, nvargs)
- }
|