storj/satellite/satellitedb/offers.go
Yingrong Zhao 09b0c2a630
create db implementation for offer table (#2031)
* init marketing service

Fix linting error

Create offerdb implementation

Create offers service

Add update method

Create offer table and migration

Fix linting error

fix conflicts

Insert new data

Change duration to have clear indication to be based on days

add error wrapper

Change from using uuid to int for id field

* Create Marketing service

* make error virable name more readable

* add condition in update service method to check offer status

* generate lock file

Change get to listAllOffers

* Add method for getting current offer

wip

* add check for expires_at in update method

* Fix conflicts

* add copyright header

* Fix linting error

* only allow update to active offers

* add isDefault argument to GetCurrent

* Update lock file

* add migration file

* finish migrate for adding credit_in_cents for both award and invitee

* save 100 years as expiration date for default offers

* create crud test for offers

* add GetCurrent test

* modify doc

* Fix GetCurrent to work with default offer

* fix linting issue

* add more tests and address feedbacks

* fix migration file

* add type column back to match with mockup design

* add type column back to match with mockup design

* move doc changes to new pr

* add comments

* change GetCurrent to GetCurrentByType

* fix typo
2019-06-04 15:17:01 -04:00

165 lines
5.2 KiB
Go

// Copyright (C) 2019 Storj Labs, Inc.
// See LICENSE for copying information
package satellitedb
import (
"context"
"database/sql"
"time"
"github.com/zeebo/errs"
"storj.io/storj/satellite/marketing"
dbx "storj.io/storj/satellite/satellitedb/dbx"
)
type offers struct {
db *dbx.DB
}
// ListAll returns all offers from the db
func (offers *offers) ListAll(ctx context.Context) ([]marketing.Offer, error) {
offersDbx, err := offers.db.All_Offer(ctx)
if err != nil {
return nil, marketing.OffersErr.Wrap(err)
}
return offersFromDBX(offersDbx)
}
// GetCurrent returns an offer that has not expired based on offer type
func (offers *offers) GetCurrentByType(ctx context.Context, offerType marketing.OfferType) (*marketing.Offer, error) {
var statement string
const columns = "id, name, description, award_credit_in_cents, invitee_credit_in_cents, award_credit_duration_days, invitee_credit_duration_days, redeemable_cap, num_redeemed, expires_at, created_at, status, type"
statement = `
WITH o AS (
SELECT ` + columns + ` FROM offers WHERE status=? AND type=? AND expires_at>? AND num_redeemed < redeemable_cap
)
SELECT ` + columns + ` FROM o
UNION ALL
SELECT ` + columns + ` FROM offers
WHERE type=? AND status=?
AND NOT EXISTS (
SELECT id FROM o
) order by created_at desc;`
rows := offers.db.DB.QueryRowContext(ctx, offers.db.Rebind(statement), marketing.Active, offerType, time.Now().UTC(), offerType, marketing.Default)
o := marketing.Offer{}
err := rows.Scan(&o.ID, &o.Name, &o.Description, &o.AwardCreditInCents, &o.InviteeCreditInCents, &o.AwardCreditDurationDays, &o.InviteeCreditDurationDays, &o.RedeemableCap, &o.NumRedeemed, &o.ExpiresAt, &o.CreatedAt, &o.Status, &o.Type)
if err == sql.ErrNoRows {
return nil, marketing.OffersErr.New("no current offer")
}
if err != nil {
return nil, marketing.OffersErr.Wrap(err)
}
return &o, nil
}
// Create inserts a new offer into the db
func (offers *offers) Create(ctx context.Context, o *marketing.NewOffer) (*marketing.Offer, error) {
currentTime := time.Now()
if o.ExpiresAt.Before(currentTime) {
return nil, marketing.OffersErr.New("expiration time: %v can't be before: %v", o.ExpiresAt, currentTime)
}
tx, err := offers.db.Open(ctx)
if err != nil {
return nil, marketing.OffersErr.Wrap(err)
}
// If there's an existing current offer, update its status to Done and set its expires_at to be NOW()
statement := offers.db.Rebind(`
UPDATE offers SET status=?, expires_at=?
WHERE status=? AND type=? AND expires_at>?;
`)
_, err = tx.Tx.ExecContext(ctx, statement, marketing.Done, currentTime, o.Status, o.Type, currentTime)
if err != nil {
return nil, marketing.OffersErr.Wrap(errs.Combine(err, tx.Rollback()))
}
offerDbx, err := tx.Create_Offer(ctx,
dbx.Offer_Name(o.Name),
dbx.Offer_Description(o.Description),
dbx.Offer_AwardCreditInCents(o.AwardCreditInCents),
dbx.Offer_InviteeCreditInCents(o.InviteeCreditInCents),
dbx.Offer_AwardCreditDurationDays(o.AwardCreditDurationDays),
dbx.Offer_InviteeCreditDurationDays(o.InviteeCreditDurationDays),
dbx.Offer_RedeemableCap(o.RedeemableCap),
dbx.Offer_ExpiresAt(o.ExpiresAt),
dbx.Offer_Status(int(o.Status)),
dbx.Offer_Type(int(o.Type)),
)
if err != nil {
return nil, marketing.OffersErr.Wrap(errs.Combine(err, tx.Rollback()))
}
newOffer, err := convertDBOffer(offerDbx)
if err != nil {
return nil, marketing.OffersErr.Wrap(errs.Combine(err, tx.Rollback()))
}
return newOffer, marketing.OffersErr.Wrap(tx.Commit())
}
// Update modifies an offer entry's status and amount of offers redeemed based on offer id
func (offers *offers) Update(ctx context.Context, o *marketing.UpdateOffer) error {
updateFields := dbx.Offer_Update_Fields{
Status: dbx.Offer_Status(int(o.Status)),
NumRedeemed: dbx.Offer_NumRedeemed(o.NumRedeemed),
ExpiresAt: dbx.Offer_ExpiresAt(o.ExpiresAt),
}
offerID := dbx.Offer_Id(o.ID)
_, err := offers.db.Update_Offer_By_Id(ctx, offerID, updateFields)
if err != nil {
return marketing.OffersErr.Wrap(err)
}
return nil
}
func offersFromDBX(offersDbx []*dbx.Offer) ([]marketing.Offer, error) {
var offers []marketing.Offer
var errList errs.Group
for _, offerDbx := range offersDbx {
offer, err := convertDBOffer(offerDbx)
if err != nil {
errList.Add(err)
continue
}
offers = append(offers, *offer)
}
return offers, errList.Err()
}
func convertDBOffer(offerDbx *dbx.Offer) (*marketing.Offer, error) {
if offerDbx == nil {
return nil, marketing.OffersErr.New("offerDbx parameter is nil")
}
o := marketing.Offer{
ID: offerDbx.Id,
Name: offerDbx.Name,
Description: offerDbx.Description,
AwardCreditInCents: offerDbx.AwardCreditInCents,
InviteeCreditInCents: offerDbx.InviteeCreditInCents,
RedeemableCap: offerDbx.RedeemableCap,
NumRedeemed: offerDbx.NumRedeemed,
ExpiresAt: offerDbx.ExpiresAt,
AwardCreditDurationDays: offerDbx.AwardCreditDurationDays,
InviteeCreditDurationDays: offerDbx.InviteeCreditDurationDays,
CreatedAt: offerDbx.CreatedAt,
Status: marketing.OfferStatus(offerDbx.Status),
Type: marketing.OfferType(offerDbx.Type),
}
return &o, nil
}