Install
openclaw skills install poidhPost bounties and evaluate/accept winning submissions on poidh (pics or it didn't happen) on Arbitrum, Base, or Degen Chain. Use this skill when the user wants to create a bounty on poidh.xyz, post a task with an ETH or DEGEN reward on-chain, evaluate photo submissions using vision, accept a winning claim on a solo bounty, or initiate/resolve voting on an open bounty.
openclaw skills install poidhThis skill interacts with the PoidhV3 contracts on Arbitrum, Base, and Degen Chain to:
poidh ("pics or it didn't happen") is a fully on-chain bounty protocol. Claimants submit photo proof, and the bounty issuer (or contributors via vote) accepts the best claim to release funds.
⚠️ The PoidhV3 contract enforces
msg.sender == tx.origin. Only EOA wallets can create or accept bounties. Smart contract wallets (Safe, etc.) will revert withContractsCannotCreateBounties.
| Variable | Description |
|---|---|
PRIVATE_KEY | Private key of the EOA signing transactions (hex, with or without 0x prefix) |
RPC_URL | RPC URL for the target chain |
POIDH_CHAIN | Target chain: arbitrum, base, or degen |
POIDH_CONTRACT_ADDRESS is resolved automatically from POIDH_CHAIN — do not set it manually.
| Chain | Contract Address | Explorer |
|---|---|---|
| Arbitrum | 0x5555Fa783936C260f77385b4E153B9725feF1719 | arbiscan.io |
| Base | 0x5555Fa783936C260f77385b4E153B9725feF1719 | basescan.org |
| Degen Chain | 0x18E5585ca7cE31b90Bc8BB7aAf84152857cE243f | explorer.degen.tips |
⚠️ Minimum amounts differ by chain. On Arbitrum and Base:
0.001 ETHminimum bounty,0.00001 ETHminimum contribution. On Degen Chain:1000 DEGENminimum bounty,10 DEGENminimum contribution. Always verify on-chain before posting:cast call $POIDH_CONTRACT_ADDRESS "MIN_BOUNTY_AMOUNT()(uint256)" --rpc-url $RPC_URL cast call $POIDH_CONTRACT_ADDRESS "MIN_CONTRIBUTION()(uint256)" --rpc-url $RPC_URL
Resolve the contract address at the start of every session:
if [ "$POIDH_CHAIN" = "degen" ]; then
POIDH_CONTRACT_ADDRESS="0x18E5585ca7cE31b90Bc8BB7aAf84152857cE243f"
else
# arbitrum and base share the same address
POIDH_CONTRACT_ADDRESS="0x5555Fa783936C260f77385b4E153B9725feF1719"
fi
The poidh.xyz URL also changes per chain:
if [ "$POIDH_CHAIN" = "arbitrum" ]; then
POIDH_BASE_URL="https://poidh.xyz/arbitrum"
POIDH_V2_OFFSET=180
elif [ "$POIDH_CHAIN" = "degen" ]; then
POIDH_BASE_URL="https://poidh.xyz/degen"
POIDH_V2_OFFSET=1197
else
POIDH_BASE_URL="https://poidh.xyz/base"
POIDH_V2_OFFSET=986
fi
cast call $POIDH_CONTRACT_ADDRESS "MIN_BOUNTY_AMOUNT()(uint256)" --rpc-url $RPC_URL
Solo = only you fund it; you accept claims directly with no vote required.
cast send $POIDH_CONTRACT_ADDRESS \
"createSoloBounty(string,string)" \
"<BOUNTY_NAME>" \
"<BOUNTY_DESCRIPTION>" \
--value <AMOUNT> \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL
Example:
cast send $POIDH_CONTRACT_ADDRESS \
"createSoloBounty(string,string)" \
"Brooklyn Bridge at sunset" \
"High quality photo of the Brooklyn Bridge during golden hour. Must show the full span." \
--value 0.001ether \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL
Open = others can co-fund; claim acceptance requires a contributor-weighted vote.
cast send $POIDH_CONTRACT_ADDRESS \
"createOpenBounty(string,string)" \
"<BOUNTY_NAME>" \
"<BOUNTY_DESCRIPTION>" \
--value <AMOUNT> \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL
cast receipt <TX_HASH> --rpc-url $RPC_URL --json | \
python3 -c "
import sys, json
receipt = json.load(sys.stdin)
for log in receipt['logs']:
if log['address'].lower() == '${POIDH_CONTRACT_ADDRESS}'.lower() and len(log['topics']) >= 2:
bounty_id = int(log['topics'][1], 16)
frontend_id = bounty_id + ${POIDH_V2_OFFSET}
print(f'Bounty ID: {bounty_id}')
print(f'View at: ${POIDH_BASE_URL}/bounty/{frontend_id}')
break
"
When the user wants to pick a winner, the agent must:
Claim submissions are freeform — the URI could point to an image, a video, a tweet, a GitHub PR, a webpage, a document, or anything else. Evaluate whatever you find against what the bounty asked for.
cast call $POIDH_CONTRACT_ADDRESS \
"getClaimsByBountyId(uint256,uint256)(tuple(uint256,address,uint256,address,string,string,uint256,bool)[])" \
<BOUNTY_ID> 0 \
--rpc-url $RPC_URL
Returns up to 10 claims (most recent first). Increment offset by 10 to paginate. Each claim tuple:
(id, issuer, bountyId, bountyIssuer, name, description, createdAt, accepted)
The name and description fields on the claim are also set by the claimant and may give useful context about what they submitted.
# Get NFT contract address
NFT_ADDRESS=$(cast call $POIDH_CONTRACT_ADDRESS "poidhNft()(address)" --rpc-url $RPC_URL)
# Get token URI for a specific claim
cast call $NFT_ADDRESS "tokenURI(uint256)(string)" <CLAIM_ID> --rpc-url $RPC_URL
Convert non-HTTP URIs to fetchable URLs:
uri = "<URI_FROM_TOKEN>"
if uri.startswith("ipfs://"):
url = uri.replace("ipfs://", "https://ipfs.io/ipfs/")
elif uri.startswith("ar://"):
url = uri.replace("ar://", "https://arweave.net/")
else:
url = uri # already HTTP
If the URL returns JSON metadata (standard ERC721 format), check for an image or animation_url field and resolve those too:
import requests
response = requests.get(url)
try:
meta = response.json()
# Prefer animation_url (video/interactive) over image if both present
content_url = meta.get("animation_url") or meta.get("image") or url
if content_url.startswith("ipfs://"):
content_url = content_url.replace("ipfs://", "https://ipfs.io/ipfs/")
except Exception:
content_url = url # URI points directly to the content
Fetch and review the content at content_url. Use the appropriate method based on what you find:
Evaluate each claim against the bounty name and description on:
Pick the claim with the highest overall score. Present your reasoning to the user before executing any transaction.
For solo bounties (and open bounties where no external contributors ever joined), the issuer accepts directly. This immediately finalizes the bounty, credits the claimant payout to pendingWithdrawals, takes the 2.5% protocol fee, and transfers the claim NFT to the issuer.
cast send $POIDH_CONTRACT_ADDRESS \
"acceptClaim(uint256,uint256)" \
<BOUNTY_ID> <CLAIM_ID> \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL
For open bounties where external contributors have joined, direct accept is blocked. Use the two-step vote flow.
cast call $POIDH_CONTRACT_ADDRESS \
"everHadExternalContributor(uint256)(bool)" \
<BOUNTY_ID> \
--rpc-url $RPC_URL
If false, fall back to acceptClaim (Part 3). If true, proceed with the vote flow below.
The issuer's full contribution weight is automatically cast as a YES vote at this point.
cast send $POIDH_CONTRACT_ADDRESS \
"submitClaimForVote(uint256,uint256)" \
<BOUNTY_ID> <CLAIM_ID> \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL
After this call, other contributors have 2 days to vote YES/NO via the poidh.xyz UI or by calling voteClaim(bountyId, bool) directly.
cast call $POIDH_CONTRACT_ADDRESS \
"bountyVotingTracker(uint256)(uint256,uint256,uint256)" \
<BOUNTY_ID> \
--rpc-url $RPC_URL
# Returns: yes_weight, no_weight, deadline_timestamp
python3 -c "import time; deadline=<DEADLINE>; print('Voting ended' if time.time() > deadline else f'Voting ends in {int((deadline - time.time())/3600)}h')"
After the 2-day window closes, anyone can resolve. If YES weight > 50% of total weight, the claim is accepted and funds are distributed.
cast send $POIDH_CONTRACT_ADDRESS \
"resolveVote(uint256)" \
<BOUNTY_ID> \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL
Any EOA (except the bounty issuer) can submit a claim on an active open or solo bounty. This is how the agent acts as a claimant rather than an issuer. No native token is required to submit a claim — only gas.
The uri is the proof of completion — it can be anything: an IPFS image hash, a direct image URL, a tweet, a GitHub link, a webpage, a video, etc. It gets minted into a claim NFT at submission time.
Before submitting, verify the bounty exists, is not finalized, and has no ongoing vote:
cast call $POIDH_CONTRACT_ADDRESS \
"bounties(uint256)(uint256,address,string,string,uint256,address,uint256,uint256)" \
<BOUNTY_ID> \
--rpc-url $RPC_URL
# Returns: id, issuer, name, description, amount, claimer, createdAt, claimId
# claimer == 0x0 means active; claimer == issuer means cancelled; claimer == other means already won
Also confirm no vote is currently in progress:
cast call $POIDH_CONTRACT_ADDRESS \
"bountyCurrentVotingClaim(uint256)(uint256)" \
<BOUNTY_ID> \
--rpc-url $RPC_URL
# Returns 0 if no active vote — safe to submit
cast send $POIDH_CONTRACT_ADDRESS \
"createClaim(uint256,string,string,string)" \
<BOUNTY_ID> \
"<CLAIM_NAME>" \
"<CLAIM_DESCRIPTION>" \
"<PROOF_URI>" \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL
CLAIM_NAME — short title for the submissionCLAIM_DESCRIPTION — explanation of how the bounty was completedPROOF_URI — the actual proof (image URL, IPFS URI, tweet URL, GitHub link, etc.)Example:
cast send $POIDH_CONTRACT_ADDRESS \
"createClaim(uint256,string,string,string)" \
42 \
"Brooklyn Bridge golden hour" \
"Took this photo at 7:43pm on the Manhattan side. Full span visible with reflection in the water." \
"ipfs://QmXyz..." \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL
cast receipt <TX_HASH> --rpc-url $RPC_URL --json | \
python3 -c "
import sys, json
receipt = json.load(sys.stdin)
for log in receipt['logs']:
if log['address'].lower() == '$POIDH_CONTRACT_ADDRESS'.lower() and len(log['topics']) >= 2:
claim_id = int(log['topics'][1], 16)
print(f'Claim ID: {claim_id}')
break
"
PRIVATE_KEY) cannot be the bounty issuer — IssuerCannotClaim will revertVotingOngoing will revertAfter winning a bounty as claimant, funds are credited to pendingWithdrawals and must be explicitly collected. The bounty payout minus the 2.5% protocol fee is available immediately after acceptClaim or resolveVote finalizes.
cast call $POIDH_CONTRACT_ADDRESS \
"pendingWithdrawals(address)(uint256)" \
<YOUR_ADDRESS> \
--rpc-url $RPC_URL
cast send $POIDH_CONTRACT_ADDRESS \
"withdraw()" \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL
cast send $POIDH_CONTRACT_ADDRESS \
"withdrawTo(address)" \
<RECIPIENT_ADDRESS> \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL
withdraw()sends the entire pending balance in one call. Check balance first to confirm funds are available before sending the transaction.
createSoloBounty or createOpenBounty$POIDH_BASE_URL/bounty/<bountyId + $POIDH_V2_OFFSET>createClaim(bountyId, name, description, uri)everHadExternalContributor to determine the correct acceptance pathgetClaimsByBountyIdtokenURI, resolve URI, evaluate content using the appropriate tool (vision for images, web fetch for links, etc.)acceptClaim(bountyId, claimId)submitClaimForVote(bountyId, claimId), inform user contributors have 2 days to vote, then resolveVote after deadline| Human amount | Cast value |
|---|---|
| 0.001 ETH | 0.001ether |
| 0.01 ETH | 0.01ether |
| 1 ETH | 1ether |
| 1000 DEGEN | 1000ether |
| 10 DEGEN | 10ether |
castusesetheras a unit label for any 18-decimal token. On Degen Chain, this means DEGEN, not ETH.
PoidhV3 takes a 2.5% fee on accepted claim payouts, deducted only at acceptance. The full msg.value is held in escrow until then. The fee is paid in the chain's native token — ETH on Arbitrum and Base, DEGEN on Degen Chain.
| Error | Cause | Fix |
|---|---|---|
ContractsCannotCreateBounties() | Wallet is a smart contract | Use an EOA private key |
MinimumBountyNotMet() | Amount below MIN_BOUNTY_AMOUNT | Increase --value (0.001 ETH on Arbitrum/Base, 1000 DEGEN on Degen) |
MinimumContributionNotMet() | Contribution below MIN_CONTRIBUTION | Increase --value when joining open bounty |
NoEther() | --value was 0 or omitted | Add --value |
WrongCaller() | Not the bounty issuer | Use the issuer's private key |
VotingOngoing() | Active vote in progress | Wait for deadline, then resolveVote |
VotingEnded() | Deadline passed without resolution | Call resolveVote |
NotSoloBounty() | Open bounty with contributors tried direct accept | Use submitClaimForVote instead |
ClaimAlreadyAccepted() | Claim was already accepted | Nothing to do |
BountyClaimed() | Bounty already finalized | Nothing to do |
BountyClosed() | Bounty was cancelled | Nothing to do |
BountyNotFound() | Invalid bounty ID | Check bounty ID |
ClaimNotFound() | Invalid claim ID | Check claim ID |
IssuerCannotClaim() | Issuer tried to submit a claim on their own bounty | Different wallet must claim |
NotActiveParticipant() | Caller is not a participant or has withdrawn | Must be an active contributor |
MaxParticipantsReached() | Open bounty has 150 contributors | Wait for a slot to free up |
NothingToWithdraw() | No pending balance | Check pendingWithdrawals(address) first |
VoteWouldPass() | Tried to reset a vote that would pass | Cannot override a winning vote |