schwab-sdk 0.4.1

Async Rust client for the Charles Schwab Trader API and real-time market-data streaming.
Documentation
//! Declarative macros shared across the API endpoint modules.

/// Declare a public string-valued enum with a `Unknown(String)` catch-all so
/// wire values added after this crate was published deserialize cleanly,
/// plus the matching `Serialize` / `Deserialize` / `From<X> for String`
/// / `From<String> for X` impls.
///
/// The generated enum is `#[non_exhaustive]`, so adding a known variant in a
/// later crate version is not a breaking change for downstream `match` arms.
/// Together with `Unknown(String)`, this covers both forward-compat hazards:
/// `Unknown` keeps unknown wire values parseable, `non_exhaustive` keeps new
/// known variants from breaking caller `match`es.
///
/// The expansion requires `serde::{Serialize, Deserialize}` and the `strum`
/// derive macros to be reachable at the call site (Rust resolves derive
/// macros at the use site, not the definition site, so the macro deliberately
/// emits the unqualified `Serialize` / `Deserialize` paths the call site is
/// expected to have in scope).
macro_rules! string_enum {
    (
        $(#[$meta:meta])*
        $name:ident {
            $( $(#[$variant_meta:meta])* $variant:ident = $wire:literal ),* $(,)?
        }
    ) => {
        $(#[$meta])*
        #[derive(
            Debug, Clone, PartialEq, Eq, Hash,
            strum::Display, strum::EnumString,
            serde::Serialize, serde::Deserialize,
        )]
        #[serde(into = "String", from = "String")]
        #[non_exhaustive]
        pub enum $name {
            $(
                $(#[$variant_meta])*
                #[strum(serialize = $wire)]
                $variant,
            )*
            /// A wire value Schwab sent that this crate does not recognize.
            /// The raw string is preserved so callers can still route on it.
            #[strum(default)]
            Unknown(String),
        }

        impl From<$name> for String {
            fn from(v: $name) -> Self { v.to_string() }
        }

        impl From<String> for $name {
            fn from(v: String) -> Self {
                match v.as_str() {
                    $($wire => $name::$variant,)*
                    _ => $name::Unknown(v),
                }
            }
        }
    };
}

pub(crate) use string_enum;