blob: 9ba88c93586898a08038694cb8d09b74f80262fc [file] [log] [blame]
/*
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 dalgorm
import (
"fmt"
"github.com/apache/incubator-devlake/core/dal"
"github.com/apache/incubator-devlake/core/errors"
)
// DalgormTransaction represents a gorm transaction which using the same underlying
// session for all queries
type DalgormTransaction struct {
*Dalgorm
}
var _ dal.Transaction = (*DalgormTransaction)(nil)
// Rollback the transaction
func (t *DalgormTransaction) Rollback() errors.Error {
r := t.db.Rollback()
if r.Error != nil {
return errors.Default.Wrap(r.Error, "failed to rollback transaction")
}
return nil
}
// Commit the transaction
func (t *DalgormTransaction) Commit() errors.Error {
r := t.db.Commit()
if r.Error != nil {
return errors.Default.Wrap(r.Error, "failed to commit transaction")
}
return nil
}
func (t *DalgormTransaction) LockTables(lockTables dal.LockTables) errors.Error {
switch t.Dialect() {
case "mysql":
// mysql lock all tables at once, each lock would release all previous locks
clause := ""
for _, lockTable := range lockTables {
if clause != "" {
clause += ", "
}
clause += lockTable.TableName()
if lockTable.Exclusive {
clause += " WRITE"
} else {
clause += " READ"
}
}
return t.Exec(fmt.Sprintf("LOCK TABLES %s", clause))
case "postgres":
for _, lockTable := range lockTables {
var clause string
if lockTable.Exclusive {
clause = "EXCLUSIVE"
} else {
clause = "SHARE"
}
stmt := fmt.Sprintf("LOCK TABLE %s IN %s MODE;", lockTable.TableName(), clause)
err := t.Exec(stmt)
if err != nil {
return err
}
}
return nil
default:
panic(fmt.Errorf("unknown dialect %s", t.Dialect()))
}
}
func (t *DalgormTransaction) UnlockTables() errors.Error {
switch t.Dialect() {
case "mysql":
// mysql would not release lock automatically on Rollback
// according to https://dev.mysql.com/doc/refman/8.0/en/lock-tables.html
return t.Exec("UNLOCK TABLES")
case "postgres":
// pg has no unlock tables
return nil
default:
panic(fmt.Errorf("unknown dialect %s", t.Dialect()))
}
}
func newTransaction(dalgorm *Dalgorm) *DalgormTransaction {
return &DalgormTransaction{
Dalgorm: NewDalgorm(dalgorm.db.Begin()),
}
}