Skip to content

Commit

Permalink
Merge pull request #582 from formancehq/feat/transaction-bulk
Browse files Browse the repository at this point in the history
feat: add capability to create transaction at controller level
  • Loading branch information
gfyrag authored Nov 22, 2024
2 parents 382c5e9 + d0f36bb commit e5abb9c
Show file tree
Hide file tree
Showing 32 changed files with 931 additions and 435 deletions.
1 change: 1 addition & 0 deletions Earthfile
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ generate:
RUN go install github.com/princjef/gomarkdoc/cmd/gomarkdoc@latest
COPY (+tidy/*) /src/
COPY --dir (+sources/src/*) /src/

WORKDIR /src
RUN go generate ./...
SAVE ARTIFACT internal AS LOCAL internal
Expand Down
2 changes: 2 additions & 0 deletions docs/api/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -498,6 +498,8 @@ Accept: application/json
|Name|In|Type|Required|Description|
|---|---|---|---|---|
|ledger|path|string|true|Name of the ledger.|
|continueOnFailure|query|boolean|false|Continue on failure|
|atomic|query|boolean|false|Make bulk atomic|
|body|body|[V2Bulk](#schemav2bulk)|false|none|

> Example responses
Expand Down
43 changes: 43 additions & 0 deletions internal/api/common/mocks_ledger_controller_test.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

43 changes: 43 additions & 0 deletions internal/api/v1/mocks_ledger_controller_test.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

31 changes: 24 additions & 7 deletions internal/api/v2/controllers_bulk.go
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,13 @@ func bulkHandler(bulkMaxSize int) http.HandlerFunc {

w.Header().Set("Content-Type", "application/json")

ret, errorsInBulk, err := ProcessBulk(r.Context(), common.LedgerFromContext(r.Context()), b, api.QueryParamBool(r, "continueOnFailure"))
ret, errorsInBulk, err := ProcessBulk(
r.Context(),
common.LedgerFromContext(r.Context()),
b,
api.QueryParamBool(r, "continueOnFailure"),
api.QueryParamBool(r, "atomic"),
)
if err != nil {
api.InternalServerError(w, r, err)
return
Expand Down Expand Up @@ -122,12 +128,7 @@ func (req *TransactionRequest) ToRunScript(allowUnboundedOverdrafts bool) (*ledg
}, nil
}

func ProcessBulk(
ctx context.Context,
l ledgercontroller.Controller,
bulk Bulk,
continueOnFailure bool,
) ([]Result, bool, error) {
func ProcessBulk(ctx context.Context, l ledgercontroller.Controller, bulk Bulk, continueOnFailure bool, atomic bool) ([]Result, bool, error) {

for i, element := range bulk {
switch element.Action {
Expand Down Expand Up @@ -166,6 +167,15 @@ func ProcessBulk(
errorsInBulk = true
}

if atomic {
if err := l.BeginTX(ctx, nil); err != nil {
return nil, errorsInBulk, fmt.Errorf("error starting transaction: %s", err)
}
defer func() {
_ = l.Rollback(ctx)
}()
}

for i, element := range bulk {
switch element.Action {
case ActionCreateTransaction:
Expand Down Expand Up @@ -370,5 +380,12 @@ func ProcessBulk(
}
}
}

if atomic {
if err := l.Commit(ctx); err != nil {
return nil, errorsInBulk, fmt.Errorf("error committing transaction: %s", err)
}
}

return ret, errorsInBulk, nil
}
69 changes: 68 additions & 1 deletion internal/api/v2/controllers_bulk_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -349,9 +349,76 @@ func TestBulk(t *testing.T) {
}},
expectError: true,
},
{
name: "with atomic",
body: `[
{
"action": "ADD_METADATA",
"data": {
"targetId": "world",
"targetType": "ACCOUNT",
"metadata": {
"foo": "bar"
}
}
},
{
"action": "ADD_METADATA",
"data": {
"targetId": "world",
"targetType": "ACCOUNT",
"metadata": {
"foo2": "bar2"
}
}
}
]`,
queryParams: map[string][]string{
"atomic": {"true"},
},
expectations: func(mockLedger *LedgerController) {
mockLedger.EXPECT().
BeginTX(gomock.Any(), nil).
Return(nil)

mockLedger.EXPECT().
SaveAccountMetadata(gomock.Any(), ledgercontroller.Parameters[ledgercontroller.SaveAccountMetadata]{
Input: ledgercontroller.SaveAccountMetadata{
Address: "world",
Metadata: metadata.Metadata{
"foo": "bar",
},
},
}).
Return(&ledger.Log{}, nil)

mockLedger.EXPECT().
SaveAccountMetadata(gomock.Any(), ledgercontroller.Parameters[ledgercontroller.SaveAccountMetadata]{
Input: ledgercontroller.SaveAccountMetadata{
Address: "world",
Metadata: metadata.Metadata{
"foo2": "bar2",
},
},
}).
Return(&ledger.Log{}, nil)

mockLedger.EXPECT().
Commit(gomock.Any()).
Return(nil)

mockLedger.EXPECT().
Rollback(gomock.Any()).
Return(nil)
},
expectResults: []Result{{
ResponseType: ActionAddMetadata,
}, {
ResponseType: ActionAddMetadata,
}},
},
}
for _, testCase := range testCases {
testCase := testCase
t.Run(testCase.name, func(t *testing.T) {

systemController, ledgerController := newTestingSystemController(t, true)
Expand Down
43 changes: 43 additions & 0 deletions internal/api/v2/mocks_ledger_controller_test.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

5 changes: 5 additions & 0 deletions internal/controller/ledger/controller.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ package ledger

import (
"context"
"database/sql"
"github.com/formancehq/go-libs/v2/metadata"
"github.com/formancehq/ledger/internal/machine/vm"

Expand All @@ -13,6 +14,10 @@ import (
//go:generate mockgen -write_source_comment=false -write_package_comment=false -source controller.go -destination controller_generated_test.go -package ledger . Controller

type Controller interface {
BeginTX(ctx context.Context, options *sql.TxOptions) error
Commit(ctx context.Context) error
Rollback(ctx context.Context) error

// IsDatabaseUpToDate check if the ledger store is up to date, including the bucket and the ledger specifics
// It returns true if up to date
IsDatabaseUpToDate(ctx context.Context) (bool, error)
Expand Down
Loading

0 comments on commit e5abb9c

Please sign in to comment.