From 15290e50342d1548c08c6703d1ecb2dd7ed51c16 Mon Sep 17 00:00:00 2001 From: Jason Song Date: Sun, 8 Jan 2023 16:09:49 +0800 Subject: [PATCH 1/3] fix: half committer --- models/db/context.go | 20 ++++-- models/db/context_committer_test.go | 102 ++++++++++++++++++++++++++++ 2 files changed, 118 insertions(+), 4 deletions(-) create mode 100644 models/db/context_committer_test.go diff --git a/models/db/context.go b/models/db/context.go index 455f3d1c5de74..3e23cfc185fd9 100644 --- a/models/db/context.go +++ b/models/db/context.go @@ -98,19 +98,31 @@ type Committer interface { // halfCommitter is a wrapper of Committer. // It can be closed early, but can't be committed early, it is useful for reusing a transaction. type halfCommitter struct { - Committer + committer Committer + committed bool } -func (*halfCommitter) Commit() error { - // do nothing +func (c *halfCommitter) Commit() error { + c.committed = true + // should do nothing, and the parent committer will commit later return nil } +func (c *halfCommitter) Close() error { + if c.committed { + // it's "commit and close", should do nothing, and the parent committer will commit later + return nil + } + + // it's "rollback and close", let the parent committer rollback right now + return c.committer.Close() +} + // TxContext represents a transaction Context, // it will reuse the existing transaction in the parent context or create a new one. func TxContext(parentCtx context.Context) (*Context, Committer, error) { if sess, ok := inTransaction(parentCtx); ok { - return newContext(parentCtx, sess, true), &halfCommitter{Committer: sess}, nil + return newContext(parentCtx, sess, true), &halfCommitter{committer: sess}, nil } sess := x.NewSession() diff --git a/models/db/context_committer_test.go b/models/db/context_committer_test.go new file mode 100644 index 0000000000000..9c40ed2c93a29 --- /dev/null +++ b/models/db/context_committer_test.go @@ -0,0 +1,102 @@ +// Copyright 2022 The Gitea Authors. All rights reserved. +// SPDX-License-Identifier: MIT + +package db // it's not db_test, because this file is for testing the private type halfCommitter + +import ( + "fmt" + "testing" + + "github.com/stretchr/testify/assert" +) + +type MockCommitter struct { + wants []string + gots []string +} + +func NewMockCommitter(wants ...string) *MockCommitter { + return &MockCommitter{ + wants: wants, + } +} + +func (c *MockCommitter) Commit() error { + c.gots = append(c.gots, "commit") + return nil +} + +func (c *MockCommitter) Close() error { + c.gots = append(c.gots, "close") + return nil +} + +func (c *MockCommitter) Assert(t *testing.T) { + assert.Equal(t, c.wants, c.gots, "want operations %v, but got %v", c.wants, c.gots) +} + +func Test_halfCommitter(t *testing.T) { + /* + Do something like: + + ctx, committer, err := db.TxContext(db.DefaultContext) + if err != nil { + return nil + } + defer committer.Close() + + // ... + + if err != nil { + return nil + } + + // ... + + return committer.Commit() + */ + + testWithCommiter := func(committer Committer, f func(committer Committer) error) { + if err := f(&halfCommitter{committer: committer}); err == nil { + committer.Commit() + } + committer.Close() + } + + t.Run("commit and close", func(t *testing.T) { + mockCommitter := NewMockCommitter("commit", "close") + + testWithCommiter(mockCommitter, func(committer Committer) error { + defer committer.Close() + return committer.Commit() + }) + + mockCommitter.Assert(t) + }) + + t.Run("rollback and close", func(t *testing.T) { + mockCommitter := NewMockCommitter("close", "close") + + testWithCommiter(mockCommitter, func(committer Committer) error { + defer committer.Close() + if true { + return fmt.Errorf("error") + } + return committer.Commit() + }) + + mockCommitter.Assert(t) + }) + + t.Run("close and commit", func(t *testing.T) { + mockCommitter := NewMockCommitter("close", "close") + + testWithCommiter(mockCommitter, func(committer Committer) error { + committer.Close() + committer.Commit() + return fmt.Errorf("error") + }) + + mockCommitter.Assert(t) + }) +} From 5872d1ade7477382f8c7ab0a2b86344ad380a6ff Mon Sep 17 00:00:00 2001 From: Jason Song Date: Mon, 9 Jan 2023 11:07:22 +0800 Subject: [PATCH 2/3] fix: WithTx --- models/db/context.go | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/models/db/context.go b/models/db/context.go index 3e23cfc185fd9..911dbd1c6f7fe 100644 --- a/models/db/context.go +++ b/models/db/context.go @@ -138,7 +138,12 @@ func TxContext(parentCtx context.Context) (*Context, Committer, error) { // this function will reuse it otherwise will create a new one and close it when finished. func WithTx(parentCtx context.Context, f func(ctx context.Context) error) error { if sess, ok := inTransaction(parentCtx); ok { - return f(newContext(parentCtx, sess, true)) + err := f(newContext(parentCtx, sess, true)) + if err != nil { + // rollback immediately, in case the caller ignores returned error and tries to commit the transaction. + _ = sess.Close() + } + return err } return txWithNoCheck(parentCtx, f) } From cbf56c09cc7e660b651b57350b105f08c355688b Mon Sep 17 00:00:00 2001 From: Jason Song Date: Mon, 9 Jan 2023 20:32:34 +0800 Subject: [PATCH 3/3] fix: typo --- models/db/context_committer_test.go | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/models/db/context_committer_test.go b/models/db/context_committer_test.go index 9c40ed2c93a29..38e91f22edb41 100644 --- a/models/db/context_committer_test.go +++ b/models/db/context_committer_test.go @@ -56,7 +56,7 @@ func Test_halfCommitter(t *testing.T) { return committer.Commit() */ - testWithCommiter := func(committer Committer, f func(committer Committer) error) { + testWithCommitter := func(committer Committer, f func(committer Committer) error) { if err := f(&halfCommitter{committer: committer}); err == nil { committer.Commit() } @@ -66,7 +66,7 @@ func Test_halfCommitter(t *testing.T) { t.Run("commit and close", func(t *testing.T) { mockCommitter := NewMockCommitter("commit", "close") - testWithCommiter(mockCommitter, func(committer Committer) error { + testWithCommitter(mockCommitter, func(committer Committer) error { defer committer.Close() return committer.Commit() }) @@ -77,7 +77,7 @@ func Test_halfCommitter(t *testing.T) { t.Run("rollback and close", func(t *testing.T) { mockCommitter := NewMockCommitter("close", "close") - testWithCommiter(mockCommitter, func(committer Committer) error { + testWithCommitter(mockCommitter, func(committer Committer) error { defer committer.Close() if true { return fmt.Errorf("error") @@ -91,7 +91,7 @@ func Test_halfCommitter(t *testing.T) { t.Run("close and commit", func(t *testing.T) { mockCommitter := NewMockCommitter("close", "close") - testWithCommiter(mockCommitter, func(committer Committer) error { + testWithCommitter(mockCommitter, func(committer Committer) error { committer.Close() committer.Commit() return fmt.Errorf("error")