Skip to content
This repository has been archived by the owner on Apr 2, 2024. It is now read-only.

Commit

Permalink
refactor(BUX-312): extract choosing before hook and function for roll…
Browse files Browse the repository at this point in the history
…ing back transaction on error during save.
  • Loading branch information
dorzepowski committed Jan 17, 2024
1 parent e64a646 commit f884f0c
Showing 1 changed file with 36 additions and 41 deletions.
77 changes: 36 additions & 41 deletions model_save.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@ import (

// Save will save the model(s) into the Datastore
func Save(ctx context.Context, model ModelInterface) (err error) {

// Check for a client
c := model.Client()
if c == nil {
Expand All @@ -26,24 +25,9 @@ func Save(ctx context.Context, model ModelInterface) (err error) {
// @siggi: we need this to be in a callback context for Mongo
// NOTE: a DB error is not being returned from here
return ds.NewTx(ctx, func(tx *datastore.Transaction) (err error) {

// Fire the before hooks (parent model)
if model.IsNew() {
if err = model.BeforeCreating(ctx); err != nil {
rollbackErr := tx.Rollback()
if rollbackErr != nil {
err = errors.Wrap(err, rollbackErr.Error())
}
return
}
} else {
if err = model.BeforeUpdating(ctx); err != nil {
rollbackErr := tx.Rollback()
if rollbackErr != nil {
err = errors.Wrap(err, rollbackErr.Error())
}
return
}
parentBeforeHook := _beforeHook(model)
if err = parentBeforeHook(ctx); err != nil {
return _closeTxWithError(tx, err)
}

// Set the record's timestamps
Expand All @@ -55,24 +39,11 @@ func Save(ctx context.Context, model ModelInterface) (err error) {
// Add any child models (fire before hooks)
if children := model.ChildModels(); len(children) > 0 {
for _, child := range children {
if child.IsNew() {
if err = child.BeforeCreating(ctx); err != nil {
rollbackErr := tx.Rollback()
if rollbackErr != nil {
err = errors.Wrap(err, rollbackErr.Error())
}
return
}
} else {
if err = child.BeforeUpdating(ctx); err != nil {
rollbackErr := tx.Rollback()
if rollbackErr != nil {
err = errors.Wrap(err, rollbackErr.Error())
}
return
}
}

childBeforeHook := _beforeHook(child)
if err = childBeforeHook(ctx); err != nil {
return _closeTxWithError(tx, err)
}
// Set the record's timestamps
child.SetRecordTime(child.IsNew())
}
Expand All @@ -92,11 +63,7 @@ func Save(ctx context.Context, model ModelInterface) (err error) {
if err = modelsToSave[index].Client().Datastore().SaveModel(
ctx, modelsToSave[index], tx, modelsToSave[index].IsNew(), false,
); err != nil {
rollbackErr := tx.Rollback()
if rollbackErr != nil {
err = errors.Wrap(err, rollbackErr.Error())
}
return
return _closeTxWithError(tx, err)
}
}

Expand Down Expand Up @@ -148,3 +115,31 @@ func saveToCache(ctx context.Context, keys []string, model ModelInterface, ttl t
}
return nil
}

// _closeTxWithError will close the transaction with the given error
// It's crucial to run this rollback to prevent hanging db connections.
func _closeTxWithError(tx *datastore.Transaction, baseError error) error {
if tx == nil {
if baseError != nil {
return baseError
}
return errors.New("transaction is nil during rollback")
}
if err := tx.Rollback(); err != nil {
if baseError != nil {
return errors.Wrap(baseError, err.Error())
}
return err
}
if baseError != nil {
return baseError
}
return errors.New("closing transaction with error")
}

func _beforeHook(model ModelInterface) func(context.Context) error {
if model.IsNew() {
return model.BeforeCreating
}
return model.BeforeUpdating
}

0 comments on commit f884f0c

Please sign in to comment.