ib-flex 0.1.2

Pure Rust parser for Interactive Brokers FLEX XML statements
Documentation
# ib-flex

[![crates.io](https://img.shields.io/crates/v/ib-flex.svg)](https://crates.io/crates/ib-flex)
[![docs.rs](https://docs.rs/ib-flex/badge.svg)](https://docs.rs/ib-flex)
[![License](https://img.shields.io/badge/license-MIT%20OR%20Apache--2.0-blue.svg)](LICENSE-MIT)

Type-safe Rust parser for Interactive Brokers FLEX XML statements with comprehensive coverage and financial precision.

## Features

- **High performance** - Fast XML parsing with quick-xml and serde
- **Financial precision** - rust_decimal for all monetary values (no floating point errors)
- **Type-safe** - 15 enums with 100+ variants for comprehensive coverage
- **Comprehensive** - Supports all major FLEX sections and 20 asset categories
- **Well-tested** - Integration tests, property-based tests, edge case coverage
- **Edge case handling** - Warrants, T-Bills, CFDs, fractional shares, cancelled trades
- **Optional API client** - Fetch FLEX statements programmatically from IB

## Installation

Add this to your `Cargo.toml`:

```toml
[dependencies]
ib-flex = "0.1"
```

## Quick Start

```rust
use ib_flex::{parse_activity_flex, detect_statement_type, StatementType};

fn main() -> Result<(), Box<dyn std::error::Error>> {
    let xml = std::fs::read_to_string("flex_statement.xml")?;

    // Automatically detect statement type
    match detect_statement_type(&xml)? {
        StatementType::Activity => {
            let statement = parse_activity_flex(&xml)?;
            println!("Account: {}", statement.account_id);
            println!("Trades: {}", statement.trades.items.len());
        }
        StatementType::TradeConfirmation => {
            let statement = ib_flex::parse_trade_confirmation(&xml)?;
            println!("Account: {}", statement.account_id);
            println!("Trades: {}", statement.trades.items.len());
        }
    }

    Ok(())
}
```

## FLEX Query Setup

Interactive Brokers FLEX queries must be configured in the IB Client Portal:

1. Navigate to: Reports โ†’ Flex Queries โ†’ Create Activity Flex Query
2. Select required sections (Trades, Positions, Cash Transactions, etc.)
3. Choose date format: ISO-8601 (`yyyy-MM-dd`) or compact (`yyyyMMdd`)
4. Set output format to XML
5. Save query and note the Query ID

**Important**: European date formats (`dd/MM/yyyy`) are NOT supported by the IB FLEX API.

๐Ÿ“˜ **For comprehensive setup instructions**, see [FLEX_SETUP.md](FLEX_SETUP.md) which covers all 21 recommended sections, field selections, and configuration options.

## FLEX Web Service API Client (Optional)

The `api-client` feature provides programmatic access to fetch FLEX statements directly from Interactive Brokers without manual downloads.

### Installation with API Client

```toml
[dependencies]
ib-flex = { version = "0.1", features = ["api-client"] }
```

### API Setup

1. Log in to IB Account Management
2. Navigate to: Reports โ†’ Settings โ†’ FlexWeb Service
3. Generate a FLEX Web Service token (keep it secure!)
4. Note your FLEX Query ID from the Flex Queries page

### Usage Example

```rust
use ib_flex::api::FlexApiClient;
use std::time::Duration;

#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
    // Create client with your token
    let client = FlexApiClient::new("YOUR_TOKEN");

    // Step 1: Send request with your query ID
    let reference_code = client.send_request("123456").await?;

    // Step 2: Get statement with automatic retry
    let xml = client.get_statement_with_retry(
        &reference_code,
        10,                           // max retries
        Duration::from_secs(2)        // delay between retries
    ).await?;

    // Step 3: Parse the statement
    let statement = ib_flex::parse_activity_flex(&xml)?;
    println!("Trades: {}", statement.trades.items.len());

    Ok(())
}
```

### API Examples

Run the API examples (requires IB credentials):

```bash
export IB_FLEX_TOKEN="your_token"
export IB_FLEX_QUERY_ID="your_query_id"
cargo run --example fetch_flex_statement --features api-client
cargo run --example api_simple_usage --features api-client
cargo run --example api_with_retry --features api-client
```

## Supported FLEX Sections

### Activity FLEX - Core Types
- โœ… **Trades** - Executions with 40+ fields including P&L, commissions, dates, security details
- โœ… **Open Positions** - Current holdings with 30+ fields
- โœ… **Cash Transactions** - Deposits, withdrawals, interest, fees, dividends
- โœ… **Corporate Actions** - Splits, mergers, spinoffs, dividends (36 action types)
- โœ… **Securities Info** - Reference data for all traded instruments
- โœ… **FX Conversion Rates** - Currency conversion rates for multi-currency accounts

### Extended FLEX Sections
- โœ… **Account Information** - Account metadata and configuration
- โœ… **Change in NAV** - Net asset value changes with transfers and P&L breakdown
- โœ… **Equity Summary** - Asset allocation by category (cash, stocks, options, bonds)
- โœ… **Cash Report** - Detailed cash flow by currency
- โœ… **Trade Confirmations** - Real-time trade execution confirmations
- โœ… **Option EAE** - Option exercises, assignments, and expirations
- โœ… **FX Transactions** - Foreign exchange conversions
- โœ… **Dividend Accruals** - Accrued and open dividend tracking
- โœ… **Interest Accruals** - Interest accrual tracking by currency
- โœ… **Transfers** - Security transfers (ACATS, ATON, FOP, etc.)

### Asset Classes Supported
- โœ… **Stocks (STK)** - Including fractional shares
- โœ… **Options (OPT)** - Calls, puts, assignments, expirations
- โœ… **Futures (FUT)** - All major contracts (ES, NQ, CL, GC, etc.)
- โœ… **Forex (CASH)** - FX trades and positions
- โœ… **Bonds (BOND)** - Treasuries, corporate, municipal
- โœ… **Treasury Bills (BILL)** - With maturity handling
- โœ… **Warrants (WAR)** - Equity warrants
- โœ… **CFDs** - Contract for difference
- โœ… **Funds, Commodities, and more** - 20 asset categories total

### Trade Confirmation FLEX
- โœ… **Trade Confirmations** - Real-time trade execution data
- โœ… **All trade fields** - Full support for all trade attributes
- โœ… **Automatic detection** - Detect statement type from XML

## Performance

Benchmarked on M1 MacBook Pro:

| Statement Type | Transactions | Parse Time |
|---------------|--------------|------------|
| Minimal | 1 trade | ~6.5 ยตs |
| Options | 4 trades | ~65 ยตs |
| Cash | 15 transactions | ~71 ยตs |

Memory efficient with approximately 200 bytes per trade. Parsing 10,000 trades uses ~2MB of memory.

## Type Safety

All financial values use `rust_decimal::Decimal` for precise calculations without floating-point errors. The library includes 15 comprehensive enums with 100+ variants covering:

- **AssetCategory** (20 variants) - STK, OPT, FUT, CASH, BOND, BILL, WAR, CFD, etc.
- **Reorg** (36 variants) - All corporate action types
- **Code** (50+ variants) - Transaction classification codes
- **CashAction** (13 variants) - Cash transaction types
- **OrderType** (13 variants) - Market, limit, stop, etc.
- **BuySell, OpenClose, PutCall, LongShort, TradeType, OptionAction, and more**

## Examples

The repository includes several complete example programs:

### Parsing Examples
1. **parse_activity_statement.rs** - Basic Activity FLEX parsing and display
2. **filter_trades.rs** - Filter by asset class, side, symbol, quantity, P&L, date range
3. **calculate_commissions.rs** - Analyze commission costs by category
4. **parse_trade_confirmation.rs** - Trade Confirmation FLEX parsing

Run parsing examples:
```bash
cargo run --example parse_activity_statement
cargo run --example parse_trade_confirmation
cargo run --example filter_trades
cargo run --example calculate_commissions
```

### Historical Backfill Example
5. **backfill_summary.rs** - Parse multi-statement FLEX XML and display comprehensive summary

To use this example:
1. Create a FLEX query with **Period: Last 180 Calendar Days** (or similar) in [IBKR Client Portal]https://portal.interactivebrokers.com - see [FLEX_SETUP.md]FLEX_SETUP.md
2. Download the XML file manually or via API
3. Run the summary:

```bash
cargo run --example backfill_summary -- path/to/your/backfill.xml
```

Output includes:
- NAV over time with returns, high/low, drawdown
- Trading activity by symbol with P&L attribution
- Cash transaction breakdown (dividends, interest, fees)
- Current positions from latest statement
- Monthly returns table

### API Client Examples (requires `api-client` feature)
6. **fetch_flex_statement.rs** - Complete API workflow with detailed output
7. **api_simple_usage.rs** - Minimal API client usage
8. **api_with_retry.rs** - API client with automatic retry logic

Run API examples:
```bash
export IB_FLEX_TOKEN="your_token"
export IB_FLEX_QUERY_ID="your_query_id"
cargo run --example fetch_flex_statement --features api-client
```

## Development

### Build

```bash
cargo build
```

### Test

```bash
cargo test
```

### Benchmark

```bash
cargo bench
```

### Format

```bash
cargo fmt
```

### Lint

```bash
cargo clippy -- -D warnings
```

## Testing

The library has comprehensive test coverage including:

- **Integration tests** covering all asset classes and edge cases
- **Extended types tests** for FLEX sections
- **Error tests** for malformed XML and invalid data
- **Unit tests** for custom deserializers
- **Doc tests** in inline documentation
- **XML fixtures** including extended types, warrants, T-Bills, CFDs, fractional shares, cancelled trades

### Reliability Testing

The library includes comprehensive reliability tests using:
- **Property-based testing** with proptest for random inputs
- **Stress tests** for large XML files
- **Concurrency tests** for thread safety
- **Memory efficiency tests** for repeated parsing
- **Edge case fuzzing** for malformed inputs

Run tests with:
```bash
cargo test           # All tests
cargo test --doc     # Documentation tests only
```

## Contributing

Contributions welcome! This is an open-source project designed to benefit the Rust trading community.

**Before submitting a PR:**
1. Ensure all tests pass: `cargo test`
2. Run clippy: `cargo clippy -- -D warnings`
3. Format code: `cargo fmt`
4. Add tests for new features
5. Update CHANGELOG.md

Bug reports and feature requests are appreciated. When reporting bugs, please include:
- Anonymized XML sample demonstrating the issue
- Expected vs actual behavior
- Rust version and platform

## Documentation

- [API Documentation]https://docs.rs/ib-flex
- [Implementation Plan]PLAN.md - Full implementation details and statistics
- [Project Guide for Claude Code]CLAUDE.md - Development guide
- [Edge Cases Summary]EDGE_CASES_SUMMARY.md - Comprehensive edge case analysis
- [Types Analysis]TYPES_ANALYSIS.md - Type system breakdown

## Resources

- [IB FLEX Queries Guide]https://www.ibkrguides.com/orgportal/performanceandstatements/flex.htm
- [Activity FLEX Reference]https://www.ibkrguides.com/reportingreference/reportguide/activity%20flex%20query%20reference.htm
- [FLEX Web Service API]https://www.interactivebrokers.com/campus/ibkr-api-page/flex-web-service/

## License

Licensed under either of:

- Apache License, Version 2.0 ([LICENSE-APACHE]LICENSE-APACHE or http://www.apache.org/licenses/LICENSE-2.0)
- MIT license ([LICENSE-MIT]LICENSE-MIT or http://opensource.org/licenses/MIT)

at your option.

## Acknowledgments

- Inspired by [csingley/ibflex]https://github.com/csingley/ibflex (Python) - Comprehensive enum research
- Built with [quick-xml]https://github.com/tafia/quick-xml - Fast XML parsing with serde
- [rust_decimal]https://github.com/paupino/rust-decimal - Financial precision
- [chrono]https://github.com/chronotope/chrono - Date/time handling

---

See [PLAN.md](PLAN.md) for detailed implementation statistics and [EDGE_CASES_SUMMARY.md](EDGE_CASES_SUMMARY.md) for edge case coverage.