Skip to content

Latest commit

 

History

History
130 lines (96 loc) · 6.91 KB

README.md

File metadata and controls

130 lines (96 loc) · 6.91 KB

Donations With Automatic Matching Up to a Maximum

Someone (let's call them "Matcher") wants to pledge to match donations to a certain account (the "Recipient") and says "I'll match donations up to X amount." So he/she sends the max amount to the controlling contract (sort of like escrow) and earmarks those funds for Recipient.

For simplicity (to avoid complexity of expiration dates and cron jobs and whatever), any Matcher can at any time rescind any unclaimed funds. If the promised funds from that Matcher towards a specific Recipient become 0, the Matcher is removed (is no longer listed as a Matcher related to that Recipient).

Any other account (other Donors) can choose to donate to the Recipient account (via the controlling contract, which is this project).

On each donation:

  1. Funds that the Donor deposited into this escrow contract get transferred immediately to the Recipient.
  2. For each and every "Matcher" account currently associated with the Recipient, the following happens:
    1. This escrow contract will automatically transfer to the Recipient an amount (called "matchedAmount") that equals the minimum of the donor's donated amount and that Matcher's remaining commitment to this Recipient.
    2. The Matcher's commitment will be decreased by that "matchedAmount".

Usage

  1. clone this repo to a local folder
  2. run yarn
  3. Read https://docs.near.org/docs/tools/near-cli#near-call and decide whether you want to use --depositYocto or --deposit in the steps below.

For localnet (work in progress; consider using testnet below for now):

  1. export NEAR_ENV=local
  2. [how to deploy contract locally?]
near create-account justatemporarylocalaccount.node0 --masterAccount node0 --initialBalance 1000 --keyPath ~/.near/localnet/node0/validator_key.json

near create-account recipient.justatemporarylocalaccount.node0 --masterAccount justatemporarylocalaccount.node0 --initialBalance 10

near create-account matcher1.justatemporarylocalaccount.node0 --masterAccount justatemporarylocalaccount.node0 --initialBalance 20

near create-account matcher2.justatemporarylocalaccount.node0 --masterAccount justatemporarylocalaccount.node0 --initialBalance 20

near create-account donor.justatemporarylocalaccount.node0 --masterAccount justatemporarylocalaccount.node0 --initialBalance 20

export PARENT=justatemporarylocalaccount.node0
export MATCHER1=matcher1.justatemporarylocalaccount.node0
export MATCHER2=matcher2.justatemporarylocalaccount.node0
export RECIPIENT=recipient.justatemporarylocalaccount.node0
export DONOR=donor.justatemporarylocalaccount.node0

For testnet:

  1. export NEAR_ENV=testnet

  2. ./scripts/1.dev-deploy.sh

  3. Follow the instructions from the output of the previous line. It will tell you to run something like export CONTRACT=dev-1638053233399-4079004334.

  4. You will need at least 3 other NEAR accounts: one to act as a recipient (such as a charity), one to act as a regular donor, and one to act as a "matcher" (someone who commits to match others' donations to a certain recipient).

    • If you don't already have 3 testnet accounts that you want to use, you can create one (to serve as Matcher) at https://wallet.testnet.near.org/. Then you can create RECIPIENT and DONOR accounts as sub-accounts of that one. E.g.:
    near create-account recipient.ryancwalsh.testnet --masterAccount ryancwalsh.testnet --initialBalance 10
    near create-account matcher1.ryancwalsh.testnet --masterAccount ryancwalsh.testnet --initialBalance 20
    near create-account matcher2.ryancwalsh.testnet --masterAccount ryancwalsh.testnet --initialBalance 20
    near create-account donor.ryancwalsh.testnet --masterAccount ryancwalsh.testnet --initialBalance 20
    
  5. Call export commands to define RECIPIENT, MATCHER, and DONOR with the accountIds from the previous steps. E.g.:

    export PARENT=ryancwalsh.testnet
    export MATCHER1=matcher1.ryancwalsh.testnet
    export MATCHER2=matcher2.ryancwalsh.testnet
    export RECIPIENT=recipient.ryancwalsh.testnet
    export DONOR=donor.ryancwalsh.testnet
    

Now try using the contract (on localnet or testnet):

near state $MATCHER1 |  sed -n "s/.*formattedAmount: '\([^\\]*\).*'/\1/p"
near state $MATCHER2 |  sed -n "s/.*formattedAmount: '\([^\\]*\).*'/\1/p"
near state $RECIPIENT |  sed -n "s/.*formattedAmount: '\([^\\]*\).*'/\1/p"
near state $DONOR |  sed -n "s/.*formattedAmount: '\([^\\]*\).*'/\1/p"
near call $CONTRACT offerMatchingFunds "{\"recipient\": \"$RECIPIENT\"}" --accountId $MATCHER1 --deposit 9 --gas=15000000000000
near call $CONTRACT offerMatchingFunds "{\"recipient\": \"$RECIPIENT\"}" --accountId $MATCHER2 --deposit 1 --gas=15000000000000
near view $CONTRACT getCommitments "{\"recipient\": \"$RECIPIENT\"}"
near state $MATCHER1 |  sed -n "s/.*formattedAmount: '\([^\\]*\).*'/\1/p"
near state $MATCHER2 |  sed -n "s/.*formattedAmount: '\([^\\]*\).*'/\1/p"

(The result should reflect the values from above, and the CLI/Explorer should now also show Matcher1's balance as ~11 and Matcher2's balance as ~19.)

near call $CONTRACT rescindMatchingFunds "{\"recipient\": \"$RECIPIENT\", \"requestedAmount\": \"2000000000000000000000000\"}" --accountId $MATCHER1 --gas=90000000000000
near view $CONTRACT getCommitments "{\"recipient\": \"$RECIPIENT\"}"
near state $MATCHER1 |  sed -n "s/.*formattedAmount: '\([^\\]*\).*'/\1/p"

(Matcher1 should now only have 7 committed to this Recipient. The CLI/Explorer should now also show Matcher1's balance as ~13.)

near call $CONTRACT donate "{\"recipient\": \"$RECIPIENT\"}" --accountId $DONOR --deposit 4 --gas 300000000000000
near state $MATCHER1 |  sed -n "s/.*formattedAmount: '\([^\\]*\).*'/\1/p"
near state $MATCHER2 |  sed -n "s/.*formattedAmount: '\([^\\]*\).*'/\1/p"
near state $RECIPIENT |  sed -n "s/.*formattedAmount: '\([^\\]*\).*'/\1/p"
near state $DONOR |  sed -n "s/.*formattedAmount: '\([^\\]*\).*'/\1/p"
near view $CONTRACT getCommitments "{\"recipient\": \"$RECIPIENT\"}"

(Only Matcher1 should be committed to 3.) (The CLI/Explorer should now show: Recipient's balance as 10+4+4+1 = 19 Matcher1's balance as still ~13 Matcher2's balance as still ~19 Donor's balance is ~16.)

near call $CONTRACT rescindMatchingFunds "{\"recipient\": \"$RECIPIENT\", \"requestedAmount\": \"9999000000000000000000000000\"}" --accountId $MATCHER1 --gas=90000000000000
near state $MATCHER1 |  sed -n "s/.*formattedAmount: '\([^\\]*\).*'/\1/p"
near view $CONTRACT getCommitments "{\"recipient\": \"$RECIPIENT\"}"

(The CLI/Explorer should now show Matcher1's balance as ~16 and getCommitments as empty.)

Optionally nuke the match relationships if they weren't already emptied: near call $CONTRACT deleteAllMatchesAssociatedWithRecipient "{\"recipient\": \"$RECIPIENT\"}" --accountId $CONTRACT --gas=15000000000000

Optionally clean up accounts with:

near delete $DONOR $PARENT
near delete $RECIPIENT $PARENT
near delete $MATCHER1 $PARENT
near delete $MATCHER2 $PARENT