mod client;
pub mod models;
mod advanced;
mod analysis;
mod bulk;
mod calendars;
mod commodities;
mod company;
mod crypto;
mod dividends_splits;
mod estimates;
mod etf_mutual_funds;
mod forex;
mod fund_holdings;
mod fundamentals;
mod indexes;
mod insider_trading;
mod institutional;
mod market_performance;
mod news;
mod prices;
mod screener;
mod stock_list;
mod technical_indicators;
use crate::error::{FinanceError, Result};
use crate::rate_limiter::RateLimiter;
use client::FmpClientBuilder;
use std::sync::{Arc, OnceLock};
use std::time::Duration;
pub use advanced::*;
pub use analysis::*;
pub use bulk::*;
pub use calendars::*;
pub use commodities::*;
pub use company::*;
pub use crypto::*;
pub use dividends_splits::*;
pub use estimates::*;
pub use etf_mutual_funds::*;
pub use forex::*;
pub use fund_holdings::*;
pub use fundamentals::*;
pub use indexes::*;
pub use insider_trading::*;
pub use institutional::*;
pub use market_performance::*;
pub use models::*;
pub use news::*;
pub use prices::*;
pub use screener::*;
pub use stock_list::*;
pub use technical_indicators::*;
const FMP_RATE_PER_SEC: f64 = 5.0;
struct FmpSingleton {
api_key: String,
timeout: Duration,
limiter: Arc<RateLimiter>,
}
static FMP_SINGLETON: OnceLock<FmpSingleton> = OnceLock::new();
pub fn init(api_key: impl Into<String>) -> Result<()> {
init_with_timeout(api_key, Duration::from_secs(30))
}
pub fn init_with_timeout(api_key: impl Into<String>, timeout: Duration) -> Result<()> {
FMP_SINGLETON
.set(FmpSingleton {
api_key: api_key.into(),
timeout,
limiter: Arc::new(RateLimiter::new(FMP_RATE_PER_SEC)),
})
.map_err(|_| FinanceError::InvalidParameter {
param: "fmp".to_string(),
reason: "FMP client already initialized".to_string(),
})
}
pub(crate) fn build_client() -> Result<client::FmpClient> {
let s = FMP_SINGLETON
.get()
.ok_or_else(|| FinanceError::InvalidParameter {
param: "fmp".to_string(),
reason: "FMP not initialized. Call fmp::init(api_key) first.".to_string(),
})?;
FmpClientBuilder::new(&s.api_key)
.timeout(s.timeout)
.build_with_limiter(Arc::clone(&s.limiter))
}
#[cfg(test)]
pub(crate) fn build_test_client(base_url: &str) -> Result<client::FmpClient> {
FmpClientBuilder::new("test-key")
.timeout(Duration::from_secs(5))
.base_url(base_url)
.build_with_limiter(Arc::new(RateLimiter::new(100.0)))
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_init_errors_on_double_init() {
let _ = init("test-key-1");
let result = init("test-key-2");
assert!(matches!(result, Err(FinanceError::InvalidParameter { .. })));
}
}