Skip to content

Commit 2d8b184

Browse files
committed
Changed Auction logics: previous bidder get his money back when greater bid appears
1 parent 050d4b6 commit 2d8b184

File tree

3 files changed

+84
-70
lines changed

3 files changed

+84
-70
lines changed

auction/readme.md

+4-4
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
This is an auction dApp.
44
You can start auction for any token(s) for some period of time (counted in blocks).
55
If somebody took part in the auction, then the organizer will get the winners bid, and the winner will get token(s).
6-
Everyone who particapated, but didn't win - will have ability to get their money back after the auction end.
6+
Once next bid provided, the previous bid is returned to it's sender
77

88
## Starting auction
99
Everybody can start an auction with invokeScript 'startAuction' attaching Tokens to it.
@@ -15,6 +15,6 @@ Before auction finish anybody can bid. The bid is only accepted if it is greater
1515
If the same address will bid again, his new bid will be added to the previous ones.
1616

1717
#Withdraw
18-
Calling 'withdraw':
19-
- if called by the participant who haven't won - returning his bid
20-
- if called by the winner or organizer - transfering bid of the winner to organizer and token(s) to the winner
18+
Anyone can call 'withdraw':
19+
As a result bid of the winner will be transfered to organizer and token(s) to the winner.
20+
In case noone made bid, organizer will get his token(s) back.

auction/ride/auction.ride

+64-65
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@
33
{-# SCRIPT_TYPE ACCOUNT #-}
44

55

6-
76
let maxAuctionDuration = 1440 * 30
87
# ~ 30 days
98

@@ -12,13 +11,13 @@ let maxAuctionDuration = 1440 * 30
1211
func startAuction(duration: Int, startPrice: Int, priceAssetId:String) = {
1312

1413
let auctionId = toBase58String(i.transactionId)
15-
let auctionEndHeight = lastBlock.height + duration
14+
let endHeight = lastBlock.height + duration
1615

1716
let pmt = extract(i.payment)
1817

1918
if (duration > maxAuctionDuration) then throw("Duration is too long. Must be less than " + toString(maxAuctionDuration)) else
2019
WriteSet(
21-
[ DataEntry(auctionId, auctionEndHeight),
20+
[ DataEntry(auctionId, endHeight),
2221
DataEntry(auctionId + "_organizer", i.caller.bytes.toBase58String()),
2322
DataEntry(auctionId + "_lot_assetId", if (isDefined(pmt.assetId)) then toBase58String(value(pmt.assetId)) else "WAVES"),
2423
DataEntry(auctionId + "_lot_amount", pmt.amount),
@@ -34,27 +33,44 @@ func bid(auctionId: String) = {
3433
let pmtAssetIdStr = if (isDefined(pmt.assetId)) then toBase58String(value(pmt.assetId)) else "WAVES"
3534
let callerAddressStr = i.caller.bytes.toBase58String()
3635

37-
let auctionEndHeight = getIntegerValue(this, auctionId)
38-
let auctionPriceAssetId = getStringValue(this, auctionId + "_priceAssetId")
39-
let auctionWinAmount = getInteger(this, auctionId + "_winAmount")
40-
41-
let bidderAmount = getInteger(this, auctionId + "_bidder_" + callerAddressStr)
42-
let curBidderAmount = if isDefined(bidderAmount) then value(bidderAmount) else 0
36+
let endHeight = getIntegerValue(this, auctionId)
37+
let startPrice = getIntegerValue(this, auctionId + "_startPrice")
38+
let priceAssetId = getStringValue(this, auctionId + "_priceAssetId")
39+
let winAmount = getInteger(this, auctionId + "_winAmount")
40+
let winner = getString(this, auctionId + "_winner")
4341

44-
let totalBidAmount = curBidderAmount + pmt.amount
42+
let bidFromTheSameUser = isDefined(winner) && value(winner) == callerAddressStr
43+
let totalBidAmount = pmt.amount + if bidFromTheSameUser then
44+
value(winAmount) else 0
4545

46-
if (lastBlock.height >= auctionEndHeight) then
46+
if (lastBlock.height >= endHeight) then
4747
throw("Auction already finished") else
48-
if (auctionPriceAssetId != pmtAssetIdStr) then
49-
throw("Bid must be in asset '" + auctionPriceAssetId + "'") else
50-
if (isDefined(auctionWinAmount) && totalBidAmount <= value(auctionWinAmount)) then
51-
throw("Bid must be more then " + toString(totalBidAmount))
48+
if (priceAssetId != pmtAssetIdStr) then
49+
throw("Bid must be in asset '" + priceAssetId + "'") else
50+
if (isDefined(winAmount) && totalBidAmount <= value(winAmount) ||
51+
!isDefined(winAmount) && totalBidAmount <= startPrice) then
52+
throw("Bid must be more then "
53+
+ toString(if isDefined(winAmount) then value(winAmount) else startPrice))
5254
else
53-
WriteSet([
54-
DataEntry(auctionId + "_bidder_" + callerAddressStr, totalBidAmount),
55-
DataEntry(auctionId + "_winner", callerAddressStr),
56-
DataEntry(auctionId + "_winAmount", totalBidAmount)
57-
])
55+
if (bidFromTheSameUser || !isDefined(winner)) then
56+
WriteSet([
57+
DataEntry(auctionId + "_winner", callerAddressStr),
58+
DataEntry(auctionId + "_winAmount", totalBidAmount)
59+
])
60+
else {
61+
let previousBidderAddr = addressFromStringValue(value(winner))
62+
let priceAsset = if (priceAssetId == "WAVES" || priceAssetId == "") then unit else fromBase58String(priceAssetId)
63+
ScriptResult(
64+
WriteSet([
65+
DataEntry(auctionId + "_winner", callerAddressStr),
66+
DataEntry(auctionId + "_winAmount", totalBidAmount)
67+
]),
68+
TransferSet([
69+
ScriptTransfer(previousBidderAddr, value(winAmount), priceAsset)
70+
])
71+
)
72+
}
73+
5874
}
5975

6076

@@ -65,59 +81,42 @@ func withdraw(auctionId: String) = {
6581
let pmtAssetIdStr = if (isDefined(pmt.assetId)) then toBase58String(value(pmt.assetId)) else "WAVES"
6682
let callerAddressStr = i.caller.bytes.toBase58String()
6783

68-
let auctionEndHeight = getIntegerValue(this, auctionId)
69-
let auctionOrganizer = getStringValue(this, auctionId + "_organizer")
70-
let auctionWinner = getString(this, auctionId + "_winner")
71-
let auctionLotAssetId = getStringValue(this, auctionId + "_lot_assetId")
72-
let auctionLotAmount = getIntegerValue(this, auctionId + "_lot_amount")
73-
let auctionPriceAssetId = getStringValue(this, auctionId + "_priceAssetId")
74-
let auctionWinAmount = getIntegerValue(this, auctionId + "_winAmount")
84+
let endHeight = getIntegerValue(this, auctionId)
85+
let organizer = getStringValue(this, auctionId + "_organizer")
86+
let winner = getString(this, auctionId + "_winner")
87+
let lotAssetId = getStringValue(this, auctionId + "_lot_assetId")
88+
let lotAmount = getIntegerValue(this, auctionId + "_lot_amount")
89+
let priceAssetId = getStringValue(this, auctionId + "_priceAssetId")
90+
let winAmount = getIntegerValue(this, auctionId + "_winAmount")
7591

76-
let auctionLotAsset = if (auctionLotAssetId == "WAVES") then unit else fromBase58String(auctionLotAssetId)
77-
let auctionPriceAsset = if (auctionPriceAssetId == "WAVES" || auctionPriceAssetId == "") then unit else fromBase58String(auctionPriceAssetId)
78-
let auctionWinnerAddr = addressFromStringValue(value(auctionWinner))
79-
let auctionOrganizerAddr = addressFromStringValue(value(auctionOrganizer))
92+
let lotAsset = if (lotAssetId == "WAVES") then unit else fromBase58String(lotAssetId)
93+
let priceAsset = if (priceAssetId == "WAVES" || priceAssetId == "") then unit else fromBase58String(priceAssetId)
94+
let winnerAddr = addressFromStringValue(value(winner))
95+
let organizerAddr = addressFromStringValue(value(organizer))
8096

8197
let betAmount = getInteger(this, auctionId + "_bidder_" + callerAddressStr)
8298

83-
if (lastBlock.height < auctionEndHeight) then
99+
if (lastBlock.height < endHeight) then
84100
throw("Auction is not finished yet") else
85101

86-
if (!isDefined(auctionWinner)) then {
87-
if (callerAddressStr == auctionOrganizer) then
88-
if (!isDefined(getString(this, auctionId + "_lot_passed"))) then
89-
throw("You have already got your lot back")
90-
else
91-
ScriptResult(
92-
WriteSet([DataEntry(auctionId + "_lot_passed", i.caller.bytes.toBase58String())]),
93-
TransferSet([ScriptTransfer(i.caller, auctionLotAmount, auctionLotAsset)])
94-
)
102+
if (!isDefined(winner)) then {
103+
if (isDefined(getString(this, auctionId + "_lot_passed"))) then
104+
throw("Organizer has already got his lot back")
95105
else
96-
throw("You haven't participate in this auction")
106+
ScriptResult(
107+
WriteSet([DataEntry(auctionId + "_lot_passed", organizer)]),
108+
TransferSet([ScriptTransfer(organizerAddr, lotAmount, lotAsset)])
109+
)
97110
}
98111
else {
99-
if (callerAddressStr == auctionOrganizer || callerAddressStr == auctionWinner) then {
100-
# Lot -> winner, winner's bet -> organizer
101-
if (isDefined(getString(this, auctionId + "_lot_passed"))) then
102-
throw("Lot is already passed to the winner, and organizer got his reward")
103-
else
104-
ScriptResult(
105-
WriteSet([DataEntry(auctionId + "_lot_passed", auctionWinnerAddr.bytes.toBase58String())]),
106-
TransferSet([ScriptTransfer(auctionWinnerAddr, auctionLotAmount, auctionLotAsset),
107-
ScriptTransfer(auctionOrganizerAddr, auctionWinAmount, auctionPriceAsset)])
108-
# We don't mark winner's bid as returned here because it's not neccessary
109-
)
110-
}
111-
else {
112-
if (!isDefined(betAmount)) then
113-
throw("You didn't bid anything")
114-
else if (isDefined(getBoolean(this, auctionId + "_bidder_" + callerAddressStr + "_return"))) then
115-
throw("You have already got your bid back")
116-
else
117-
ScriptResult(
118-
WriteSet([DataEntry(auctionId + "_bidder_" + callerAddressStr + "_return", true)]),
119-
TransferSet([ScriptTransfer(i.caller, value(betAmount), auctionPriceAsset)]) # return bet which loose
120-
)
121-
}
112+
# Lot -> winner, winner's bet -> organizer
113+
if (isDefined(getString(this, auctionId + "_lot_passed"))) then
114+
throw("Lot is already passed to the winner, and organizer got his reward")
115+
else
116+
ScriptResult(
117+
WriteSet([DataEntry(auctionId + "_lot_passed", winnerAddr.bytes.toBase58String())]),
118+
TransferSet([ScriptTransfer(winnerAddr, lotAmount, lotAsset),
119+
ScriptTransfer(organizerAddr, winAmount, priceAsset)])
120+
)
122121
}
123122
}

auction/test/auction_test.js

+16-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ const ADD_FEE = 0.004 * WAVES
1010
var issueTxId
1111
var auctionId
1212
var auctionStartTx
13-
13+
var customer2Before
1414

1515
async function rememberBalances(text, forAddress, tokenId) {
1616
const tokenBal = await assetBalance(tokenId, forAddress)
@@ -64,7 +64,17 @@ describe('Auction test Suite', async function(){
6464
console.log("Start auction height : " + auctionStartTx.height)
6565
})
6666

67+
it('Unable to bid less then start price', async function(){
68+
const invTx = invokeScript({fee:INV_FEE, dApp: address(accounts.auction), call: {function:"bid", args:[{type:"string", value: auctionId}]},
69+
payment: [{amount: 999999, assetId:null }]},
70+
accounts.customer2)
71+
expect(broadcast(invTx)).rejectedWith("Bid must be more then 1000000")
72+
})
73+
6774
it('Customer2: bid 0.1 WAVES', async function(){
75+
76+
customer2Before = await balance(address(accounts.customer2))
77+
6878
const invTx = invokeScript({fee:INV_FEE, dApp: address(accounts.auction), call: {function:"bid", args:[{type:"string", value: auctionId}]},
6979
payment: [{amount: 10000000, assetId:null }]},
7080
accounts.customer2)
@@ -86,6 +96,11 @@ describe('Auction test Suite', async function(){
8696
const resp = await broadcast(invTx)
8797
await waitForTx(invTx.id)
8898
})
99+
100+
it('Previous bid returned to bidder', async function(){
101+
const customer2After = await balance(address(accounts.customer2))
102+
expect(customer2After).to.equal(customer2Before - INV_FEE, "Bid must be returned")
103+
})
89104

90105
it('Wait for auction end', async function(){
91106
const timeout = 180000

0 commit comments

Comments
 (0)