汎用フレームワークを使用した Go での堅牢な SQL トランザクション実行の構築
Go で SQL データベースを操作する場合、アトミック性を確保し、複数ステップのトランザクション中にロールバックを管理することが困難になることがあります。この記事では、柔軟なジェネリックスを使用して、Go で SQL トランザクションを実行するための堅牢で再利用可能でテスト可能なフレームワークを作成する方法を説明します。
トランザクション内で複数の依存データベース操作を実行するための SqlWriteExec ユーティリティを構築します。ステートレス操作とステートフル操作の両方をサポートし、依存関係をシームレスに管理しながら関連エンティティを挿入するなどの高度なワークフローを可能にします。
SQL トランザクションのフレームワークが必要なのはなぜですか?
実際のアプリケーションでは、データベース操作が分離されることはほとんどありません。次のシナリオを検討してください:
ユーザーを挿入し、そのインベントリをアトミックに更新します。
注文の作成と支払いの処理により、一貫性が確保されます。
複数の手順が関係するため、データの整合性を確保するには障害時のロールバックの管理が重要になります。
Txn 管理での作業。
データベース TXN を作成している場合は、コア ロジックを作成する前に考慮する必要があるいくつかのボイラー プレートが存在する可能性があります。この txn 管理は Java では Spring Boot によって管理され、Java でコードを書いている間はあまり気にすることはありませんでしたが、golang ではそうではありません。簡単な例を以下に示します
func basicTxn(db *sql.DB) error { // start a transaction tx, err := db.Begin() if err != nil { return err } defer func() { if r := recover(); r != nil { tx.Rollback() } else if err != nil { tx.Rollback() } else { tx.Commit() } }() // insert data into the orders table _, err = tx.Exec("INSERT INTO orders (id, customer_name, order_date) VALUES (1, 'John Doe', '2022-01-01')") if err != nil { return err } return nil }
すべての関数に対してロールバック/コミット コードを繰り返すことは期待できません。ここには 2 つのオプションがあります。遅延で実行されるときに txn をコミット/ロールバックする関数を戻り値の型として提供するクラスを作成するか、すべての txn 関数をまとめてラップして一度に実行するラッパー クラスを作成するかのいずれかです。
私は後者の選択を選択しました。コードの変更を以下に示します。
func TestSqlWriteExec_CreateOrderTxn(t *testing.T) { db := setupDatabase() // create a new SQL Write Executor err := dbutils.NewSqlTxnExec[OrderRequest, OrderProcessingResponse](context.TODO(), db, nil, &OrderRequest{CustomerName: "CustomerA", ProductID: 1, Quantity: 10}). StatefulExec(InsertOrder). StatefulExec(UpdateInventory). StatefulExec(InsertShipment). Commit() // check if the transaction was committed successfully if err != nil { t.Fatal(err) return } verifyTransactionSuccessful(t, db) t.Cleanup( func() { cleanup(db) db.Close() }, ) }
func InsertOrder(ctx context.Context, txn *sql.Tx, order *OrderRequest, orderProcessing *OrderProcessingResponse) error { // Insert Order result, err := txn.Exec("INSERT INTO orders (customer_name, product_id, quantity) VALUES (, , )", order.CustomerName, order.ProductID, order.Quantity) if err != nil { return err } // Get the inserted Order ID orderProcessing.OrderID, err = result.LastInsertId() return err } func UpdateInventory(ctx context.Context, txn *sql.Tx, order *OrderRequest, orderProcessing *OrderProcessingResponse) error { // Update Inventory if it exists and the quantity is greater than the quantity check if it exists result, err := txn.Exec("UPDATE inventory SET product_quantity = product_quantity - WHERE id = AND product_quantity >= ", order.Quantity, order.ProductID) if err != nil { return err } // Get the number of rows affected rowsAffected, err := result.RowsAffected() if rowsAffected == 0 { return errors.New("Insufficient inventory") } return err } func InsertShipment(ctx context.Context, txn *sql.Tx, order *OrderRequest, orderProcessing *OrderProcessingResponse) error { // Insert Shipment result, err := txn.Exec("INSERT INTO shipping_info (customer_name, shipping_address) VALUES (, 'Shipping Address')", order.CustomerName) if err != nil { return err } // Get the inserted Shipping ID orderProcessing.ShippingID, err = result.LastInsertId() return err }
このコードは非常に正確かつ簡潔になります。
コアロジックの実装方法
アイデアは、txn を単一の go struct に分離して、複数の txn を受け入れられるようにすることです。 txn とは、クラス用に作成した txn を使用してアクションを実行する関数を意味します。
type TxnFn[T any] func(ctx context.Context, txn *sql.Tx, processingReq *T) error type StatefulTxnFn[T any, R any] func(ctx context.Context, txn *sql.Tx, processingReq *T, processedRes *R) error
これら 2 つは、txn を受け取って何かを処理する関数タイプです。次に、 を実装するデータ層で次のような関数を作成し、引数の挿入と関数の実行を担当するエグゼキューター クラスにそれを渡します。
// SQL Write Executor is responsible when executing write operations // For dependent writes you may need to add the dependent data to processReq and proceed to the next function call type SqlTxnExec[T any, R any] struct { db *sql.DB txn *sql.Tx txnFns []TxnFn[T] statefulTxnFns []StatefulTxnFn[T, R] processingReq *T processedRes *R ctx context.Context err error }
これはすべての txn_fn の詳細を保存する場所であり、txn のコミットを試行する Commit() メソッドがあります。
func (s *SqlTxnExec[T, R]) Commit() (err error) { defer func() { if p := recover(); p != nil { s.txn.Rollback() panic(p) } else if err != nil { err = errors.Join(err, s.txn.Rollback()) } else { err = errors.Join(err, s.txn.Commit()) } return }() for _, writeFn := range s.txnFns { if err = writeFn(s.ctx, s.txn, s.processingReq); err != nil { return } } for _, statefulWriteFn := range s.statefulTxnFns { if err = statefulWriteFn(s.ctx, s.txn, s.processingReq, s.processedRes); err != nil { return } } return }
リポジトリにはさらに多くの例とテストがあります -
https://github.com/mahadev-k/go-utils/tree/main/examples
私たちは現在、分散システムとコンセンサスプロトコルに偏っていますが、依然として SQL を使用しており、SQL はまだ存在しています。
誰かがこれに貢献してこれを構築したい場合はお知らせください!!
ここまで読んでいただきありがとうございます!!
https://in.linkedin.com/in/mahadev-k-934520223
https://x.com/mahadev_k_
以上が汎用フレームワークを使用した Go での堅牢な SQL トランザクション実行の構築の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

ホットAIツール

Undresser.AI Undress
リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover
写真から衣服を削除するオンライン AI ツール。

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

Video Face Swap
完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

人気の記事

ホットツール

メモ帳++7.3.1
使いやすく無料のコードエディター

SublimeText3 中国語版
中国語版、とても使いやすい

ゼンドスタジオ 13.0.1
強力な PHP 統合開発環境

ドリームウィーバー CS6
ビジュアル Web 開発ツール

SublimeText3 Mac版
神レベルのコード編集ソフト(SublimeText3)

ホットトピック











OpenSSLは、安全な通信で広く使用されているオープンソースライブラリとして、暗号化アルゴリズム、キー、証明書管理機能を提供します。ただし、その歴史的バージョンにはいくつかの既知のセキュリティの脆弱性があり、その一部は非常に有害です。この記事では、Debian SystemsのOpenSSLの共通の脆弱性と対応測定に焦点を当てます。 Debianopensslの既知の脆弱性:OpenSSLは、次のようないくつかの深刻な脆弱性を経験しています。攻撃者は、この脆弱性を、暗号化キーなどを含む、サーバー上の不正な読み取りの敏感な情報に使用できます。

バックエンド学習パス:フロントエンドからバックエンドへの探査の旅は、フロントエンド開発から変わるバックエンド初心者として、すでにNodeJSの基盤を持っています...

Go Crawler Collyのキュースレッドの問題は、Go言語でColly Crawler Libraryを使用する問題を調査します。 �...

Beegoormフレームワークでは、モデルに関連付けられているデータベースを指定する方法は?多くのBEEGOプロジェクトでは、複数のデータベースを同時に操作する必要があります。 Beegoを使用する場合...

Go言語での文字列印刷の違い:printlnとstring()関数を使用する効果の違いはGOにあります...

redisstreamを使用してGo言語でメッセージキューを実装する問題は、GO言語とRedisを使用することです...

Golandのカスタム構造ラベルが表示されない場合はどうすればよいですか?ゴーランドを使用するためにGolandを使用する場合、多くの開発者はカスタム構造タグに遭遇します...
