ibapi 2.11.2

A Rust implementation of the Interactive Brokers TWS API, providing a reliable and user friendly interface for TWS and IB Gateway. Designed with a focus on simplicity and performance.
Documentation
//! Historical Ticks Mid Point example
//!
//! # Usage
//!
//! ```bash
//! cargo run --features sync --example historical_ticks_mid_point
//! ```

use clap::{arg, Command};
use ibapi::client::blocking::Client;
use time::format_description::FormatItem;
use time::macros::format_description;
use time::{OffsetDateTime, PrimitiveDateTime};

use ibapi::contracts::Contract;
use ibapi::market_data::TradingHours;

fn main() {
    env_logger::init();

    let matches = Command::new("historical_ticks_mid_point")
        .about("Get historical ticks for the specified stock")
        .arg(
            arg!(<INTERVAL_START>)
                .required(true)
                .help("time at start of interval for tick. e.g. 20240415T12:00:00"),
        )
        .arg(arg!(<STOCK_SYMBOL>).required(true).help("stock symbol to retrieve data for"))
        .arg(arg!(--connection_string <VALUE>).default_value("127.0.0.1:4002"))
        .arg(arg!(--number_of_ticks <VALUE>).default_value("100"))
        .get_matches();

    let connection_string = matches.get_one::<String>("connection_string").expect("connection_string is required");
    let interval_raw = matches.get_one::<String>("INTERVAL_START").expect("interval start is required");
    let stock_symbol = matches.get_one::<String>("STOCK_SYMBOL").expect("stock symbol is required");
    let number_of_ticks = 10;

    let client = Client::connect(connection_string, 100).expect("connection failed");

    let interval_start = parse_interval(interval_raw);
    let contract = Contract::stock(stock_symbol.as_str()).build();

    let ticks = client
        .historical_ticks_mid_point(&contract, Some(interval_start), None, number_of_ticks, TradingHours::Regular)
        .expect("historical data request failed");

    for tick in ticks {
        println!("{tick:?}");
    }
}

const INTERVAL_FORMAT: &[FormatItem] = format_description!("[year][month][day]T[hour]:[minute]:[second]Z");

fn parse_interval(text: &str) -> OffsetDateTime {
    PrimitiveDateTime::parse(text, INTERVAL_FORMAT)
        .expect("expected date in format 20230415T12:00:00Z")
        .assume_utc()
}