methodsCode = `
/// The constructor populates the initial state.
///
/// Method num 1. This is part of the Filecoin calling convention.
/// InitActor#Exec will call the constructor on method_num = 1.
pub fn constructor(params: u32) -> Option<RawBytes> {
let params = sdk::message::params_raw(params).unwrap().1;
let trusted_address = Address::from_bytes(¶ms).unwrap();
// This constant should be part of the SDK.
const INIT_ACTOR_ADDR: ActorID = 1;
// Should add SDK sugar to perform ACL checks more succinctly.
// i.e. the equivalent of the validate_* builtin-actors runtime methods.
// https://github.com/filecoin-project/builtin-actors/blob/master/actors/runtime/src/runtime/fvm.rs#L110-L146
if sdk::message::caller() != INIT_ACTOR_ADDR {
abort!(USR_FORBIDDEN, "constructor invoked by non-init actor");
}
let mut state = State {
trusted_address,
bounties_map: Cid::default(),
};
let mut bounties: Hamt<Blockstore, BountyValue, BytesKey> = Hamt::new(Blockstore);
let bounties_cid = match bounties.flush() {
Ok(map) => map,
Err(_e) => abort!(USR_ILLEGAL_STATE, "failed to create bounties hamt"),
};
state.bounties_map = bounties_cid;
state.save();
None
}
#[derive(Debug, Deserialize_tuple)]
pub struct PostBountyParams {
pub piece_cid: Cid,
pub address: Address,
}
/// Method num 2.
pub fn post_bounty(params: u32) -> Option<RawBytes> {
let params = sdk::message::params_raw(params).unwrap().1;
let params = RawBytes::new(params);
let params: PostBountyParams = params.deserialize().unwrap();
let mut state = State::load();
let mut bounties =
match Hamt::<Blockstore, BountyValue, BytesKey>::load(&state.bounties_map, Blockstore) {
Ok(map) => map,
Err(err) => abort!(USR_ILLEGAL_STATE, "failed to load bounties hamt: {:?}", err),
};
let key = BountyKey {
piece_cid: params.piece_cid,
address: params.address,
};
let raw_bytes = RawBytes::serialize(&key).unwrap();
let bytes = raw_bytes.bytes();
let key = BytesKey::from(bytes);
let mut amount = match bounties.get(&key) {
Ok(Some(bounty_value)) => bounty_value.amount.clone(),
Ok(None) => TokenAmount::from(0),
Err(err) => abort!(
USR_ILLEGAL_STATE,
"failed to query hamt when getting bounty balance: {:?}",
err
),
};
amount += sdk::message::value_received();
if amount > TokenAmount::from(0) {
let bounty_value = BountyValue { amount: amount };
bounties.set(key, bounty_value).unwrap();
// Flush the HAMT to generate the new root CID to update the actor's state.
let cid = match bounties.flush() {
Ok(cid) => cid,
Err(err) => abort!(USR_ILLEGAL_STATE, "failed to flush hamt: {:?}", err),
};
// Update the actor's state.
state.bounties_map = cid;
state.save();
}
None
}
#[derive(Debug, Serialize)]
pub struct PostedBounty {
pub piece_cid: Cid,
pub address: Address,
#[serde(with = "bigint_ser")]
pub amount: TokenAmount,
}
/// Method num 3.
pub fn list_bounties() -> Option<RawBytes> {
let mut bounties_vec = Vec::new();
let state = State::load();
let bounties =
match Hamt::<Blockstore, BountyValue, BytesKey>::load(&state.bounties_map, Blockstore) {
Ok(map) => map,
Err(err) => abort!(USR_ILLEGAL_STATE, "failed to load bounties hamt: {:?}", err),
};
bounties
.for_each(|k, v: &BountyValue| {
let raw_bytes = RawBytes::new(k.as_slice().to_vec());
let key: BountyKey = raw_bytes.deserialize().unwrap();
let posted_bounty = PostedBounty {
piece_cid: key.piece_cid,
address: key.address,
amount: v.amount.clone(),
};
bounties_vec.push(posted_bounty);
Ok(())
})
.unwrap();
Some(RawBytes::serialize(&bounties_vec).unwrap())
}
/// Method num 4.
pub fn lookup_bounty(params: u32) -> Option<RawBytes> {
let params = sdk::message::params_raw(params).unwrap().1;
let params = RawBytes::new(params);
let params: PostBountyParams = params.deserialize().unwrap();
let state = State::load();
let bounties =
match Hamt::<Blockstore, BountyValue, BytesKey>::load(&state.bounties_map, Blockstore) {
Ok(map) => map,
Err(err) => abort!(USR_ILLEGAL_STATE, "failed to load bounties hamt: {:?}", err),
};
let key = BountyKey {
piece_cid: params.piece_cid,
address: params.address,
};
let raw_bytes = RawBytes::serialize(&key).unwrap();
let bytes = raw_bytes.bytes();
let key = BytesKey::from(bytes);
let amount = match bounties.get(&key) {
Ok(Some(bounty_value)) => bounty_value.amount.clone(),
Ok(None) => TokenAmount::from(0),
Err(err) => abort!(
USR_ILLEGAL_STATE,
"failed to query hamt when getting bounty balance: {:?}",
err
),
};
let bounty_value = BountyValue { amount: amount };
Some(RawBytes::serialize(&bounty_value).unwrap())
}
#[derive(Debug, Deserialize_tuple)]
pub struct AwardBountyParams {
pub piece_cid: Cid,
pub address: Address,
pub payout_address: Address,
}
/// Method num 5.
pub fn award_bounty(params: u32) -> Option<RawBytes> {
let params = sdk::message::params_raw(params).unwrap().1;
let params = RawBytes::new(params);
let params: AwardBountyParams = params.deserialize().unwrap();
let mut state = State::load();
let caller = sdk::message::caller();
let address = Address::new_id(caller);
if state.trusted_address != address.clone() {
abort!(
USR_FORBIDDEN,
"caller not trusted {:?} != {:?} (trusted)",
address,
&state.trusted_address
);
}
let mut bounties =
match Hamt::<Blockstore, BountyValue, BytesKey>::load(&state.bounties_map, Blockstore) {
Ok(map) => map,
Err(err) => abort!(USR_ILLEGAL_STATE, "failed to load bounties hamt: {:?}", err),
};
let key = BountyKey {
piece_cid: params.piece_cid,
address: params.address,
};
let raw_bytes = RawBytes::serialize(&key).unwrap();
let bytes = raw_bytes.bytes();
let key = BytesKey::from(bytes);
let amount = match bounties.get(&key) {
Ok(Some(bounty_value)) => bounty_value.amount.clone(),
Ok(None) => TokenAmount::from(0),
Err(err) => abort!(
USR_ILLEGAL_STATE,
"failed to query hamt when getting bounty balance: {:?}",
err
),
};
if amount > TokenAmount::from(0) {
let send_params = RawBytes::default();
let _receipt =
fvm_sdk::send::send(¶ms.payout_address, METHOD_SEND, send_params, amount).unwrap();
bounties.delete(&key).unwrap();
// Flush the HAMT to generate the new root CID to update the actor's state.
let cid = match bounties.flush() {
Ok(cid) => cid,
Err(err) => abort!(USR_ILLEGAL_STATE, "failed to flush hamt: {:?}", err),
};
// Update the actor's state.
state.bounties_map = cid;
state.save();
}
None
}
`.trim()