Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update summary on shipped state #605

Merged
merged 7 commits into from
Dec 26, 2016
Merged

Update summary on shipped state #605

merged 7 commits into from
Dec 26, 2016

Conversation

tonypizzicato
Copy link
Contributor

@tonypizzicato tonypizzicato commented Dec 21, 2016

  • Fix transitions between stock item states (e.g.decrease onHand count on Shipped)
  • Introduce Shipped state for consistency
  • Implement transactions proxy*
  • Run summary update in transaction
  • Wrap all existing transactions in a wrapper

Transactions proxy

Transactions proxy are implemented as a wrapper over gorm.DB


type Transaction struct {
	db      *gorm.DB
	wrapped bool
	Error   error
}

func NewTransaction(db *gorm.DB) *Transaction {
	_, inTxn := db.CommonDB().(*sql.Tx)

	return &Transaction{db, inTxn, nil}
}

func (txn *Transaction) runTransaction(methodName string, whenTxn bool) *Transaction {
	if txn.wrapped {
		return txn
	}

	if _, isTxn := txn.db.CommonDB().(*sql.Tx); isTxn == whenTxn {
		methodValue := reflect.ValueOf(txn.db).MethodByName(methodName)
		if !methodValue.IsValid() {
			panic(fmt.Sprintf("Can't call method %s of Transaction", methodName))
		}

		result := methodValue.Call([]reflect.Value{})

		txn.db = result[0].Interface().(*gorm.DB)
		txn.Error = txn.db.Error
	}

	return txn
}

Transactions is used in repos and knows when it is a repo-itself transaction (created with *sql.DB instance) or when it is already started transaction (instance of *sql.Tx). So it just proxies calls to *gorm.DB when transaction is not started and does nothing when transaction already begun.

Usage

txn := transaction.NewTransaction(repository.db).Begin()

In case when transaction was not started (repository.db is a regular db connection) it just proxies calls to *gorm.DB instance
When repo was instantiated with transaction (means transaction is managed outside repo)

	txn := service.db.Begin()
	shipmentRepo := repositories.NewShipmentRepository(txn)
        ...

	if err := txn.Commit().Error; err != nil {
		txn.Rollback()
		return nil, err
	}

calls to transaction methods in repo itself do nothing.

Running services with outer-managed transactions

To use a service in a service that already started transaction (e.g. https://github.com/FoxComm/highlander/blob/master/middlewarehouse/services/shipment_service.go#L72)
WithTransaction(txn *gorm.DB) IInventoryService method for such services was implemented

Usage

	txn := service.db.Begin()
        ...
	if err := service.inventoryService.WithTransaction(txn).ReserveItems(shipment.OrderRefNum); err != nil {
		txn.Rollback()
		return nil, err
	}

WithTransaction method return a copy of a service with txn *gorm.DB field set to passed transaction.
The same approach was used for repos.

func (service *inventoryService) getUnitRepo() repositories.IStockItemUnitRepository {
	if service.txn == nil {
		return service.unitRepo
	}

	return service.unitRepo.WithTransaction(service.txn)
}

In this case it service runs under outer-managed transaction, it gets a copy of a repo with transaction instead old *gorm.DB instace - service.unitRepo.WithTransaction(service.txn)

Using transaction proxy and WithTransaction approach together

In order to make transactions passing to work this two approaches should be used together.
But, in case of using transaction proxies without WithTransaction (means repos would always have own db instance instead one with started transaction) it would take no effect on code running and repo would start it's own transaction every time as it does right now.
But, in case of using WithTransaction methods without transactions proxy would lead to can't start transaction errors from gorm

@tonypizzicato
Copy link
Contributor Author

Copy link
Contributor

@jmataya jmataya left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A couple small comments, otherwise this looks good!

@@ -28,6 +29,15 @@ func NewShipmentRepository(db *gorm.DB) IShipmentRepository {
return &shipmentRepository{db}
}

// WithTransaction returns a shallow copy of repository with its db changed to txn. The provided txn must be non-nil.
func (repository *shipmentRepository) WithTransaction(txn *gorm.DB) IShipmentRepository {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm... thinking about the signature of this method - why do we pass the DB in from the outside, rather than use repository.db? And if we shouldn't be using any property from repository, why is this a member method on IShipmentRepository?

// WithTransaction returns a shallow copy of repository with its db changed to txn. The provided txn must be non-nil.
func (repository *shipmentRepository) WithTransaction(txn *gorm.DB) IShipmentRepository {
if txn == nil {
panic("nil transaction")
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do we want to panic here or return an error?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this case is an error in the code, so i think it's better to fail asap with a panic

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I can buy that logic.

@@ -97,11 +115,76 @@ func (service *inventoryService) HoldItems(refNum string, skus map[string]int) e
}

// get stock items associated with SKUs
items, err := service.stockItemRepo.GetStockItemsBySKUs(skusList)
items, err := service.getStockItemsBySKUs(skusList)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Really like you pulling this into this service! 👍

@jmataya
Copy link
Contributor

jmataya commented Dec 26, 2016

Talked through my concerns 1:1 with @tonypizzicato. These changes look good to me!

@jmataya jmataya merged commit e63a34a into master Dec 26, 2016
@jmataya jmataya deleted the fix/mwh-summary-update branch December 26, 2016 18:25
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants