fugle-marketdata-core 0.6.0

Internal kernel for the Fugle market data SDK. End users should depend on `fugle-marketdata` instead.
Documentation
//! Tickers endpoint - GET /futopt/intraday/tickers

use crate::{
    errors::MarketDataError,
    models::futopt::{ContractType, FutOptTicker, FutOptType},
    rest::client::RestClient,
};

/// Request builder for FutOpt intraday tickers (batch) endpoint
pub struct TickersRequestBuilder<'a> {
    client: &'a RestClient,
    typ: Option<FutOptType>,
    exchange: Option<String>,
    session: Option<String>,
    contract_type: Option<ContractType>,
}

impl<'a> TickersRequestBuilder<'a> {
    /// Create a new tickers request builder
    pub(crate) fn new(client: &'a RestClient) -> Self {
        Self {
            client,
            typ: None,
            exchange: None,
            session: None,
            contract_type: None,
        }
    }

    /// Set the contract type (FUTURE or OPTION) - required
    pub fn typ(mut self, typ: FutOptType) -> Self {
        self.typ = Some(typ);
        self
    }

    /// Set the exchange filter (e.g., "TAIFEX")
    pub fn exchange(mut self, exchange: &str) -> Self {
        self.exchange = Some(exchange.to_string());
        self
    }

    /// Query after-hours session data
    ///
    /// Sets `session=afterhours` query parameter
    pub fn after_hours(mut self) -> Self {
        self.session = Some("afterhours".to_string());
        self
    }

    /// Set the contract type filter (I, R, B, C, S, E)
    pub fn contract_type(mut self, contract_type: ContractType) -> Self {
        self.contract_type = Some(contract_type);
        self
    }

    /// Execute the request and return the tickers
    ///
    /// # Errors
    /// Returns [`MarketDataError`] on transport, deserialization, validation,
    /// or non-2xx API failures.
    pub fn send(self) -> Result<Vec<FutOptTicker>, MarketDataError> {
        // type is required for tickers endpoint
        let typ = self.typ.ok_or_else(|| MarketDataError::ConfigError(
            "type parameter is required for tickers endpoint".to_string(),
        ))?;

        // Build URL with query parameters
        let mut query_params = Vec::new();
        query_params.push(format!("type={}", typ.as_str()));

        if let Some(exchange) = &self.exchange {
            query_params.push(format!("exchange={}", exchange));
        }
        if let Some(session) = &self.session {
            query_params.push(format!("session={}", session));
        }
        if let Some(contract_type) = &self.contract_type {
            query_params.push(format!("contractType={}", contract_type.as_code()));
        }

        let url = format!(
            "{}/futopt/intraday/tickers?{}",
            self.client.get_base_url(),
            query_params.join("&")
        );

        // Make request
        let request = self.client.agent().get(&url);
        let request = self.client.auth().apply_to_request(request);

        let response = self.client.execute(request)?;
        let tickers: Vec<FutOptTicker> = response
            .into_json()
            .map_err(|e| MarketDataError::Other(e.into()))?;

        Ok(tickers)
    }
}

#[cfg(test)]
mod tests {
    use super::*;
    use crate::rest::Auth;

    #[test]
    fn test_tickers_builder_requires_type() {
        let client = RestClient::new(Auth::SdkToken("test".to_string()));
        let builder = TickersRequestBuilder::new(&client);

        let result = builder.send();
        assert!(result.is_err());
        assert!(matches!(result.unwrap_err(), MarketDataError::ConfigError(_)));
    }

    #[test]
    fn test_tickers_builder_with_type() {
        let client = RestClient::new(Auth::SdkToken("test".to_string()));
        let builder = TickersRequestBuilder::new(&client).typ(FutOptType::Future);

        assert_eq!(builder.typ, Some(FutOptType::Future));
    }

    #[test]
    fn test_tickers_builder_full_params() {
        let client = RestClient::new(Auth::SdkToken("test".to_string()));
        let builder = TickersRequestBuilder::new(&client)
            .typ(FutOptType::Option)
            .exchange("TAIFEX")
            .after_hours()
            .contract_type(ContractType::Index);

        assert_eq!(builder.typ, Some(FutOptType::Option));
        assert_eq!(builder.exchange, Some("TAIFEX".to_string()));
        assert_eq!(builder.session, Some("afterhours".to_string()));
        assert_eq!(builder.contract_type, Some(ContractType::Index));
    }

    #[test]
    fn test_tickers_builder_chaining() {
        let client = RestClient::new(Auth::SdkToken("test".to_string()));
        let builder = TickersRequestBuilder::new(&client)
            .typ(FutOptType::Future)
            .exchange("TAIFEX")
            .contract_type(ContractType::Stock);

        // Verify all fields are set
        assert!(builder.typ.is_some());
        assert!(builder.exchange.is_some());
        assert!(builder.contract_type.is_some());
    }
}