|
| 1 | +use axum::{extract::State, Json}; |
| 2 | +use serde::{Deserialize, Serialize}; |
| 3 | +use crate::{ |
| 4 | + controllers::generate_handlers, |
| 5 | + AppState |
| 6 | +}; |
| 7 | + |
| 8 | +fn default_filter() -> String { |
| 9 | + String::from("") |
| 10 | +} |
| 11 | + |
| 12 | +#[derive(Deserialize)] |
| 13 | +pub struct ReferenceProteomeCountParameters { |
| 14 | + #[serde(default = "default_filter")] |
| 15 | + filter: String |
| 16 | +} |
| 17 | + |
| 18 | +#[derive(Deserialize)] |
| 19 | +pub struct ReferenceProteomeFilterParameters { |
| 20 | + #[serde(default = "default_filter")] |
| 21 | + filter: String, |
| 22 | + start: usize, |
| 23 | + end: usize, |
| 24 | + #[serde(default)] |
| 25 | + sort_by: String, // Can be "id", "name", or "rank" |
| 26 | + #[serde(default)] |
| 27 | + sort_descending: bool |
| 28 | +} |
| 29 | + |
| 30 | +#[derive(Serialize)] |
| 31 | +pub struct ReferenceProteomeCountResult { |
| 32 | + count: u32 |
| 33 | +} |
| 34 | + |
| 35 | +fn get_taxon_name_by_id(taxon_store: &datastore::TaxonStore, taxon_id: u32) -> String { |
| 36 | + taxon_store |
| 37 | + .get_name(taxon_id) |
| 38 | + .cloned() |
| 39 | + .unwrap_or_else(|| "Unknown".to_string()) |
| 40 | +} |
| 41 | + |
| 42 | +async fn count_handler( |
| 43 | + State(AppState { datastore, .. }): State<AppState>, |
| 44 | + ReferenceProteomeCountParameters { filter }: ReferenceProteomeCountParameters |
| 45 | +) -> Result<ReferenceProteomeCountResult, ()> { |
| 46 | + let proteome_store = datastore.reference_proteome_store(); |
| 47 | + |
| 48 | + if filter.is_empty() { |
| 49 | + Ok(ReferenceProteomeCountResult { |
| 50 | + count: proteome_store.mapper |
| 51 | + .values() |
| 52 | + .count() as u32 |
| 53 | + }) |
| 54 | + } else { |
| 55 | + Ok(ReferenceProteomeCountResult { |
| 56 | + count: proteome_store.mapper |
| 57 | + .iter() |
| 58 | + .filter(|(key, (taxon_id, _))| { |
| 59 | + let taxon_name = get_taxon_name_by_id(datastore.taxon_store(), *taxon_id); |
| 60 | + |
| 61 | + key.to_lowercase().contains(&filter.to_lowercase()) || |
| 62 | + taxon_id.to_string().contains(&filter) || |
| 63 | + taxon_name.to_lowercase().contains(&filter.to_lowercase()) |
| 64 | + }) |
| 65 | + .count() as u32 |
| 66 | + }) |
| 67 | + } |
| 68 | +} |
| 69 | + |
| 70 | +async fn filter_handler( |
| 71 | + State(AppState { datastore, .. }): State<AppState>, |
| 72 | + ReferenceProteomeFilterParameters { |
| 73 | + filter, |
| 74 | + start, |
| 75 | + end, |
| 76 | + sort_by, |
| 77 | + sort_descending |
| 78 | + }: ReferenceProteomeFilterParameters |
| 79 | +) -> Result<Vec<String>, ()> { |
| 80 | + let proteome_store = datastore.reference_proteome_store(); |
| 81 | + |
| 82 | + let mut filtered_proteomes: Vec<(&String, &(u32, u32))> = proteome_store.mapper |
| 83 | + .iter() |
| 84 | + .filter(|(key, (taxon_id, _))| { |
| 85 | + let taxon_name = get_taxon_name_by_id(datastore.taxon_store(), *taxon_id); |
| 86 | + |
| 87 | + key.to_lowercase().contains(&filter.to_lowercase()) || |
| 88 | + taxon_id.to_string().contains(&filter) || |
| 89 | + taxon_name.to_lowercase().contains(&filter.to_lowercase()) |
| 90 | + }) |
| 91 | + .collect(); |
| 92 | + |
| 93 | + // Sort based on the `sort_by` field |
| 94 | + match sort_by.as_str() { |
| 95 | + "taxon_name" => { |
| 96 | + let sort_fn = |a_taxon_id, b_taxon_id| { |
| 97 | + let taxon_name_a = get_taxon_name_by_id(datastore.taxon_store(), a_taxon_id); |
| 98 | + let taxon_name_b = get_taxon_name_by_id(datastore.taxon_store(), b_taxon_id); |
| 99 | + |
| 100 | + if sort_descending { |
| 101 | + taxon_name_b.cmp(&taxon_name_a) |
| 102 | + } else { |
| 103 | + taxon_name_a.cmp(&taxon_name_b) |
| 104 | + } |
| 105 | + }; |
| 106 | + |
| 107 | + filtered_proteomes.sort_by(|(_, &(a_taxon_id, _)), (_, &(b_taxon_id,_))| { |
| 108 | + sort_fn(a_taxon_id, b_taxon_id) |
| 109 | + }); |
| 110 | + }, |
| 111 | + "protein_count" => { |
| 112 | + filtered_proteomes.sort_by(|(_, &(_, a_protein_count)), (_, &(_, b_protein_count))| { |
| 113 | + if sort_descending { |
| 114 | + b_protein_count.cmp(&a_protein_count) |
| 115 | + } else { |
| 116 | + a_protein_count.cmp(&b_protein_count) |
| 117 | + } |
| 118 | + }); |
| 119 | + }, |
| 120 | + _ => { |
| 121 | + filtered_proteomes.sort_by(|(a_proteome_id, _), (b_proteome_id, _)| { |
| 122 | + if sort_descending { |
| 123 | + b_proteome_id.cmp(a_proteome_id) |
| 124 | + } else { |
| 125 | + a_proteome_id.cmp(b_proteome_id) |
| 126 | + } |
| 127 | + }); |
| 128 | + }, |
| 129 | + } |
| 130 | + |
| 131 | + Ok(filtered_proteomes.into_iter().skip(start).take(end - start).map(|(key, _)| key.to_string()).collect()) |
| 132 | +} |
| 133 | + |
| 134 | +generate_handlers!( |
| 135 | + async fn json_count_handler( |
| 136 | + state => State<AppState>, |
| 137 | + params => ReferenceProteomeCountParameters |
| 138 | + ) -> Result<Json<ReferenceProteomeCountResult>, ()> { |
| 139 | + Ok(Json(count_handler(state, params).await?)) |
| 140 | + } |
| 141 | +); |
| 142 | + |
| 143 | +generate_handlers!( |
| 144 | + async fn json_filter_handler( |
| 145 | + state => State<AppState>, |
| 146 | + params => ReferenceProteomeFilterParameters |
| 147 | + ) -> Result<Json<Vec<String>>, ()> { |
| 148 | + Ok(Json(filter_handler(state, params).await?)) |
| 149 | + } |
| 150 | +); |
0 commit comments