Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

add task that tracks stake delegation #157

Merged
merged 12 commits into from
Dec 1, 2023
Merged
100 changes: 100 additions & 0 deletions docs/bin/openapi.json
Original file line number Diff line number Diff line change
Expand Up @@ -282,6 +282,47 @@
}
]
},
"DelegationForAddressResponse": {
"properties": {
"txId": {
"type": "string",
"nullable": true
},
"pool": {
"type": "string",
"nullable": true
}
},
"required": [
"txId",
"pool"
],
"type": "object"
},
"DelegationForAddressRequest": {
"properties": {
"until": {
"properties": {
"absoluteSlot": {
"type": "number",
"format": "double"
}
},
"required": [
"absoluteSlot"
],
"type": "object"
},
"address": {
"$ref": "#/components/schemas/Address"
}
},
"required": [
"until",
"address"
],
"type": "object"
},
"PolicyId": {
"type": "string",
"example": "b863bc7369f46136ac1048adb2fa7dae3af944c3bbb2be2f216a8d4f",
Expand Down Expand Up @@ -1008,6 +1049,65 @@
}
}
},
"/delegation/address": {
"post": {
"operationId": "DelegationForAddress",
"responses": {
"200": {
"description": "",
"content": {
"application/json": {
"schema": {
"$ref": "#/components/schemas/DelegationForAddressResponse"
}
}
}
},
"400": {
"description": "",
"content": {
"application/json": {
"schema": {
"$ref": "#/components/schemas/ErrorShape"
}
}
}
},
"409": {
"description": "",
"content": {
"application/json": {
"schema": {
"$ref": "#/components/schemas/ErrorShape"
}
}
}
},
"422": {
"description": "",
"content": {
"application/json": {
"schema": {
"$ref": "#/components/schemas/ErrorShape"
}
}
}
}
},
"security": [],
"parameters": [],
"requestBody": {
"required": true,
"content": {
"application/json": {
"schema": {
"$ref": "#/components/schemas/DelegationForAddressRequest"
}
}
}
}
}
},
"/dex/last-price": {
"post": {
"operationId": "DexLastPrice",
Expand Down
1 change: 1 addition & 0 deletions indexer/entity/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,4 +15,5 @@ pub mod dex_swap;
pub mod native_asset;
pub mod plutus_data;
pub mod plutus_data_hash;
pub mod stake_delegation;
pub mod transaction_metadata;
37 changes: 37 additions & 0 deletions indexer/entity/src/stake_delegation.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
use sea_orm::entity::prelude::*;
use serde::{Deserialize, Serialize};

#[derive(Clone, Debug, PartialEq, Eq, DeriveEntityModel, Deserialize, Serialize)]
#[sea_orm(table_name = "StakeDelegationCredentialRelation")]
pub struct Model {
#[sea_orm(primary_key, column_type = "BigInteger")]
pub id: i64,
pub stake_credential: i64,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

shouldn't we add #[sea_orm(column_type = "BigInteger")]?

Copy link
Contributor Author

@ecioppettini ecioppettini Nov 7, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure that does anything, to be honest. The columns already have bigint type for me, and it's what the docs say should happen I think (https://www.sea-ql.org/SeaORM/docs/next/generate-entity/entity-structure/).

But I can add it just in case anyway.

// pool registrations are not tracked in StakeCredentials,
pub pool_credential: Option<Vec<u8>>,
pub tx_id: i64,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

here as well

}

#[derive(Copy, Clone, Debug, DeriveRelation, EnumIter)]
pub enum Relation {
#[sea_orm(
belongs_to = "super::stake_credential::Entity",
from = "Column::StakeCredential",
to = "super::stake_credential::Column::Id"
)]
StakeCredential,
#[sea_orm(
belongs_to = "super::transaction::Entity",
from = "Column::TxId",
to = "super::transaction::Column::Id"
)]
Transaction,
}

impl Related<super::stake_credential::Entity> for Entity {
fn to() -> RelationDef {
Relation::StakeCredential.def()
}
}

impl ActiveModelBehavior for ActiveModel {}
2 changes: 2 additions & 0 deletions indexer/execution_plans/default.toml
Original file line number Diff line number Diff line change
Expand Up @@ -63,3 +63,5 @@ readonly=false
readonly=false

[MultieraCip25EntryTask]

[MultieraAddressDelegationTask]
2 changes: 2 additions & 0 deletions indexer/migration/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ mod m20220528_000012_create_plutus_data_table;
mod m20220808_000013_create_transaction_reference_input_table;
mod m20221031_000014_create_dex_table;
mod m20230223_000015_modify_block_table;
mod m20230927_231206_create_stake_delegation_table;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

let's probably keep the naming convention (date + number)?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

good catch, I didn't notice that. The weird thing is that I created this file with sea-orm-cli, so I'm not sure why it's choosing that name


pub struct Migrator;

Expand All @@ -41,6 +42,7 @@ impl MigratorTrait for Migrator {
Box::new(m20220808_000013_create_transaction_reference_input_table::Migration),
Box::new(m20221031_000014_create_dex_table::Migration),
Box::new(m20230223_000015_modify_block_table::Migration),
Box::new(m20230927_231206_create_stake_delegation_table::Migration),
]
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
use entity::prelude::{StakeCredential, StakeCredentialColumn, Transaction, TransactionColumn};
use entity::stake_delegation::*;
use sea_schema::migration::prelude::*;

pub struct Migration;

impl MigrationName for Migration {
fn name(&self) -> &str {
"m20230927_231206_create_stake_delegation_table"
}
}

#[async_trait::async_trait]
impl MigrationTrait for Migration {
async fn up(&self, manager: &SchemaManager) -> Result<(), DbErr> {
manager
.create_table(
Table::create()
.table(Entity)
.if_not_exists()
.col(
ColumnDef::new(Column::Id)
.big_integer()
.not_null()
.auto_increment(),
)
.col(
ColumnDef::new(Column::StakeCredential)
.big_integer()
.not_null(),
)
.col(ColumnDef::new(Column::TxId).big_integer().not_null())
.col(ColumnDef::new(Column::PoolCredential).binary())
.foreign_key(
ForeignKey::create()
.name("fk-stake_delegation-credential_id")
.from(Entity, Column::StakeCredential)
.to(StakeCredential, StakeCredentialColumn::Id)
.on_delete(ForeignKeyAction::Cascade),
)
.foreign_key(
ForeignKey::create()
.name("fk-stake_delegation-tx_id")
.from(Entity, Column::TxId)
.to(Transaction, TransactionColumn::Id)
.on_delete(ForeignKeyAction::Cascade),
)
.primary_key(
Index::create()
.table(Entity)
.name("stake_delegation_credential-pk")
.col(Column::Id),
)
.to_owned(),
)
.await
}

async fn down(&self, manager: &SchemaManager) -> Result<(), DbErr> {
manager
.drop_table(Table::drop().table(Entity).to_owned())
.await
}
}
1 change: 1 addition & 0 deletions indexer/tasks/src/multiera/mod.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
pub mod dex;
pub mod multiera_address;
pub mod multiera_address_credential_relations;
pub mod multiera_address_delegation;
pub mod multiera_asset_mint;
pub mod multiera_block;
pub mod multiera_cip25entry;
Expand Down
104 changes: 104 additions & 0 deletions indexer/tasks/src/multiera/multiera_address_delegation.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,104 @@
use crate::{
multiera::multiera_stake_credentials::MultieraStakeCredentialTask,
types::{AddressCredentialRelationValue, TxCredentialRelationValue},
};
use cardano_multiplatform_lib::{
address::{BaseAddress, EnterpriseAddress, PointerAddress, RewardAddress},
byron::ByronAddress,
};
use entity::{
prelude::*,
sea_orm::{prelude::*, DatabaseTransaction},
};
use pallas::ledger::{
primitives::{alonzo::Certificate, Fragment},
traverse::{MultiEraBlock, MultiEraCert, MultiEraOutput, MultiEraTx},
};
use sea_orm::{Order, QueryOrder, Set};
use std::collections::{BTreeMap, BTreeSet};
use std::ops::Deref;

use super::{
multiera_address_credential_relations::QueuedAddressCredentialRelation,
multiera_txs::MultieraTransactionTask, relation_map::RelationMap,
};
use crate::config::EmptyConfig::EmptyConfig;
use crate::dsl::database_task::BlockGlobalInfo;
use crate::dsl::task_macro::*;

carp_task! {
name MultieraAddressDelegationTask;
configuration EmptyConfig;
doc "Tracks stake delegation actions to pools.";
era multiera;
dependencies [MultieraStakeCredentialTask];
read [multiera_txs, multiera_stake_credential];
write [];
should_add_task |block, _properties| {
// recall: txs may have no outputs if they just burn all inputs as fee
// TODO: this runs slightly more than it should
!block.1.is_empty()
};
execute |previous_data, task| handle(
task.db_tx,
task.block,
&previous_data.multiera_txs,
&previous_data.multiera_stake_credential,
);
merge_result |_previous_data, _result| {};
}

async fn handle(
db_tx: &DatabaseTransaction,
block: BlockInfo<'_, MultiEraBlock<'_>, BlockGlobalInfo>,
multiera_txs: &[TransactionModel],
multiera_stake_credential: &BTreeMap<Vec<u8>, StakeCredentialModel>,
) -> Result<(), DbErr> {
for (tx_body, cardano_transaction) in block.1.txs().iter().zip(multiera_txs) {
for cert in tx_body.certs() {
{
let tx_id = cardano_transaction.id;
let cert = &cert;
match cert.as_alonzo().unwrap() {
Certificate::StakeDelegation(credential, pool) => {
let credential = credential.encode_fragment().unwrap();

let stake_credential_id = multiera_stake_credential
.get(&credential.to_vec())
.unwrap()
.id;

entity::stake_delegation::ActiveModel {
stake_credential: Set(stake_credential_id),
pool_credential: Set(Some(pool.to_vec())),
tx_id: Set(tx_id),
..Default::default()
}
.save(db_tx)
.await?;
}
Certificate::StakeDeregistration(credential) => {
let credential = credential.encode_fragment().unwrap();

let stake_credential_id = multiera_stake_credential
.get(&credential.to_vec())
.unwrap()
.id;

entity::stake_delegation::ActiveModel {
stake_credential: Set(stake_credential_id),
pool_credential: Set(None),
tx_id: Set(tx_id),
..Default::default()
}
.save(db_tx)
.await?;
}
_ => {}
};
};
}
}

Ok(())
}
Loading