Skip to content

Commit

Permalink
compliance: compliance-server: db: Define migrations and query db
Browse files Browse the repository at this point in the history
  • Loading branch information
joeykraut committed Jun 29, 2024
1 parent 1a68366 commit 731ed22
Show file tree
Hide file tree
Showing 12 changed files with 174 additions and 7 deletions.
2 changes: 1 addition & 1 deletion compliance/compliance-api/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -13,5 +13,5 @@ pub enum ComplianceStatus {
/// The wallet is compliant
Compliant,
/// The wallet is not compliant
NotCompliant,
NotCompliant { reason: String },

Check failure on line 16 in compliance/compliance-api/src/lib.rs

View workflow job for this annotation

GitHub Actions / clippy

missing documentation for a struct field

error: missing documentation for a struct field --> compliance/compliance-api/src/lib.rs:16:20 | 16 | NotCompliant { reason: String }, | ^^^^^^^^^^^^^^
}

Check failure on line 17 in compliance/compliance-api/src/lib.rs

View workflow job for this annotation

GitHub Actions / clippy

missing documentation for the crate

error: missing documentation for the crate --> compliance/compliance-api/src/lib.rs:1:1 | 1 | / use serde::{Deserialize, Serialize}; 2 | | 3 | | /// The response type for a compliance check 4 | | #[derive(Debug, Clone, Serialize, Deserialize)] ... | 16 | | NotCompliant { reason: String }, 17 | | } | |_^ | = note: requested on the command line with `-D missing-docs`

Check failure on line 17 in compliance/compliance-api/src/lib.rs

View workflow job for this annotation

GitHub Actions / clippy

missing documentation for the crate

error: missing documentation for the crate --> compliance/compliance-api/src/lib.rs:1:1 | 1 | / use serde::{Deserialize, Serialize}; 2 | | 3 | | /// The response type for a compliance check 4 | | #[derive(Debug, Clone, Serialize, Deserialize)] ... | 16 | | NotCompliant { reason: String }, 17 | | } | |_^ | = help: for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#missing_docs_in_private_items = note: requested on the command line with `-D clippy::missing-docs-in-private-items`
3 changes: 3 additions & 0 deletions compliance/compliance-server/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,9 @@ http-body-util = "0.1.0"
warp = "0.3"
compliance-api = { path = "../compliance-api" }

# === Database === #
diesel = { version = "2.2", features = ["postgres", "r2d2"] }

# === Renegade Dependencies === #
renegade-util = { workspace = true }

Expand Down
9 changes: 9 additions & 0 deletions compliance/compliance-server/diesel.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
# For documentation on how to configure this file,
# see https://diesel.rs/guides/configuring-diesel-cli

[print_schema]
file = "src/schema.rs"
custom_type_derives = ["diesel::query_builder::QueryId", "Clone"]

[migrations_directory]
dir = "/Users/joeykraut/work/relayer-extensions/compliance/compliance-server/migrations"
Empty file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
-- This file was automatically created by Diesel to setup helper functions
-- and other internal bookkeeping. This file is safe to edit, any future
-- changes will be added to existing projects as new migrations.

DROP FUNCTION IF EXISTS diesel_manage_updated_at(_tbl regclass);
DROP FUNCTION IF EXISTS diesel_set_updated_at();
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
-- This file was automatically created by Diesel to setup helper functions
-- and other internal bookkeeping. This file is safe to edit, any future
-- changes will be added to existing projects as new migrations.




-- Sets up a trigger for the given table to automatically set a column called
-- `updated_at` whenever the row is modified (unless `updated_at` was included
-- in the modified columns)
--
-- # Example
--
-- ```sql
-- CREATE TABLE users (id SERIAL PRIMARY KEY, updated_at TIMESTAMP NOT NULL DEFAULT NOW());
--
-- SELECT diesel_manage_updated_at('users');
-- ```
CREATE OR REPLACE FUNCTION diesel_manage_updated_at(_tbl regclass) RETURNS VOID AS $$
BEGIN
EXECUTE format('CREATE TRIGGER set_updated_at BEFORE UPDATE ON %s
FOR EACH ROW EXECUTE PROCEDURE diesel_set_updated_at()', _tbl);
END;
$$ LANGUAGE plpgsql;

CREATE OR REPLACE FUNCTION diesel_set_updated_at() RETURNS trigger AS $$
BEGIN
IF (
NEW IS DISTINCT FROM OLD AND
NEW.updated_at IS NOT DISTINCT FROM OLD.updated_at
) THEN
NEW.updated_at := current_timestamp;
END IF;
RETURN NEW;
END;
$$ LANGUAGE plpgsql;
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
-- Drop the wallet compliance table
DROP TABLE IF EXISTS wallet_compliance;
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
-- Create a table for caching wallet compliance information
CREATE TABLE IF NOT EXISTS wallet_compliance (
address TEXT PRIMARY KEY,
is_compliant BOOLEAN NOT NULL,
reason TEXT NOT NULL,
created_at TIMESTAMP NOT NULL DEFAULT NOW(),
expires_at TIMESTAMP NOT NULL DEFAULT NOW() + INTERVAL '1 year'
);
58 changes: 58 additions & 0 deletions compliance/compliance-server/src/db.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
//! Database helpers for the server
use std::time::SystemTime;

use compliance_api::ComplianceStatus;
use diesel::{ExpressionMethods, Insertable, PgConnection, QueryDsl, Queryable, RunQueryDsl};
use renegade_util::err_str;

use crate::{
error::ComplianceServerError,
schema::{
wallet_compliance,
wallet_compliance::dsl::{address as address_col, wallet_compliance as compliance_table},
},
};

// ----------
// | Models |
// ----------

/// A compliance entry for a wallet
#[derive(Debug, Clone, Queryable, Insertable)]
#[table_name = "wallet_compliance"]
pub struct ComplianceEntry {
pub address: String,
pub is_compliant: bool,
pub reason: String,
pub created_at: SystemTime,
pub expires_at: SystemTime,
}

impl ComplianceEntry {
/// Get the compliance status for an entry
pub fn compliance_status(&self) -> ComplianceStatus {
if self.is_compliant {
ComplianceStatus::Compliant
} else {
ComplianceStatus::NotCompliant { reason: self.reason.clone() }
}
}
}

// -----------
// | Queries |
// -----------

/// Get a compliance entry by address
pub fn get_compliance_entry(
address: &str,
conn: &mut PgConnection,
) -> Result<Option<ComplianceEntry>, ComplianceServerError> {
let query = compliance_table
.filter(address_col.eq(address))
.load::<ComplianceEntry>(conn)
.map_err(err_str!(ComplianceServerError::Db))?;

Ok(query.first().cloned())
}
9 changes: 7 additions & 2 deletions compliance/compliance-server/src/error.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,11 +6,16 @@ use warp::reject::Reject;

/// The error type emitted by the compliance server
#[derive(Debug, Clone)]
pub enum ComplianceServerError {}
pub enum ComplianceServerError {
/// An error with a database query
Db(String),
}

impl Display for ComplianceServerError {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "ComplianceServerError")
match self {
ComplianceServerError::Db(e) => write!(f, "Database error: {}", e),
}
}
}
impl Error for ComplianceServerError {}
Expand Down
37 changes: 33 additions & 4 deletions compliance/compliance-server/src/main.rs
Original file line number Diff line number Diff line change
@@ -1,10 +1,22 @@
use std::sync::Arc;

use clap::Parser;
use compliance_api::{ComplianceCheckResponse, ComplianceStatus};
use diesel::pg::PgConnection;
use diesel::r2d2::{ConnectionManager, Pool};
use error::ComplianceServerError;
use renegade_util::err_str;
use renegade_util::telemetry::{setup_system_logger, LevelFilter};
use warp::{reply::Json, Filter};

use crate::db::get_compliance_entry;

pub mod db;
pub mod error;
pub mod schema;

/// The type of the connection pool
type ConnectionPool = Arc<Pool<ConnectionManager<PgConnection>>>;

/// The CLI for the compliance server
#[derive(Debug, Clone, Parser)]
Expand All @@ -16,13 +28,21 @@ struct Cli {
/// The Chainalysis API key
#[arg(long)]
chainalysis_api_key: String,
/// The url of the compliance database
#[arg(long)]
db_url: String,
}

#[tokio::main]
async fn main() {
setup_system_logger(LevelFilter::INFO);
let cli = Cli::parse();

// Create the connection pool
let manager = ConnectionManager::<PgConnection>::new(cli.db_url.clone());
let pool = Pool::builder().build(manager).expect("Failed to create pool");
let pool = Arc::new(pool);

// Get compliance information for a wallet
let chainalysis_key = cli.chainalysis_api_key.clone();
let compliance_check = warp::get()
Expand All @@ -31,8 +51,10 @@ async fn main() {
.and(warp::path::param::<String>()) // wallet_address
.and_then(move |wallet_address| {
let key = chainalysis_key.clone();
let pool = pool.clone();

async move {
handle_req(wallet_address, &key).await
handle_req(wallet_address, &key, pool).await
}
});

Expand All @@ -49,8 +71,10 @@ async fn main() {
async fn handle_req(
wallet_address: String,
chainalysis_api_key: &str,
pool: ConnectionPool,
) -> Result<Json, warp::Rejection> {
let compliance_status = check_wallet_compliance(wallet_address, chainalysis_api_key).await?;
let compliance_status =
check_wallet_compliance(wallet_address, chainalysis_api_key, pool).await?;
let resp = ComplianceCheckResponse { compliance_status };
Ok(warp::reply::json(&resp))
}
Expand All @@ -59,10 +83,15 @@ async fn handle_req(
async fn check_wallet_compliance(
wallet_address: String,
chainalysis_api_key: &str,
pool: ConnectionPool,
) -> Result<ComplianceStatus, ComplianceServerError> {
// 1. Check the DB first
let mut conn = pool.get().map_err(err_str!(ComplianceServerError::Db))?;
let compliance_entry = get_compliance_entry(&wallet_address, &mut conn)?;
if let Some(compliance_entry) = compliance_entry {
return Ok(compliance_entry.compliance_status());
}

// 2. If not present, check the chainalysis API

todo!()
Ok(ComplianceStatus::Compliant)
}
11 changes: 11 additions & 0 deletions compliance/compliance-server/src/schema.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
// @generated automatically by Diesel CLI.

diesel::table! {
wallet_compliance (address) {
address -> Text,
is_compliant -> Bool,
reason -> Text,
created_at -> Timestamp,
expires_at -> Timestamp,
}
}

0 comments on commit 731ed22

Please sign in to comment.