✨ (Cargo.toml): Bump package version to 0.1.8 for new changes
♻️ (item_update.rs): Refactor ItemUpdate struct and related methods to store only non-null changed fields ♻️ (ls_client.rs): Refactor data update handling to store updates in a HashMap and call on_item_update for each listener 🐛 (ls_client.rs): Fix item index off-by-one error in data update handling 🐛 (main.rs): Update on_item_update implementation to handle new ItemUpdate structure ♻️ (subscription_listener.rs): Refactor on_item_update method to take a reference to ItemUpdate
This commit is contained in:
parent
82a9d2f070
commit
5af8a69942
@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "lightstreamer-client"
|
name = "lightstreamer-client"
|
||||||
version = "0.1.7"
|
version = "0.1.8"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
authors = ["Daniel López Azaña <daniloaz@gmail.com>"]
|
authors = ["Daniel López Azaña <daniloaz@gmail.com>"]
|
||||||
description = "A Rust client for Lightstreamer, designed to facilitate real-time communication with Lightstreamer servers."
|
description = "A Rust client for Lightstreamer, designed to facilitate real-time communication with Lightstreamer servers."
|
||||||
|
@ -28,7 +28,7 @@ pub struct ItemUpdate {
|
|||||||
pub item_name: Option<String>,
|
pub item_name: Option<String>,
|
||||||
pub item_pos: usize,
|
pub item_pos: usize,
|
||||||
pub fields: HashMap<String, Option<String>>,
|
pub fields: HashMap<String, Option<String>>,
|
||||||
pub changed_fields: HashMap<String, Option<String>>,
|
pub changed_fields: HashMap<String, String>,
|
||||||
pub is_snapshot: bool,
|
pub is_snapshot: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -46,7 +46,7 @@ impl ItemUpdate {
|
|||||||
///
|
///
|
||||||
/// # Returns
|
/// # Returns
|
||||||
/// A map containing the values for each field changed with the last server update.
|
/// A map containing the values for each field changed with the last server update.
|
||||||
pub fn get_changed_fields(&self) -> HashMap<String, Option<String>> {
|
pub fn get_changed_fields(&self) -> HashMap<String, String> {
|
||||||
self.changed_fields.clone()
|
self.changed_fields.clone()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -60,7 +60,7 @@ impl ItemUpdate {
|
|||||||
///
|
///
|
||||||
/// # Returns
|
/// # Returns
|
||||||
/// A map containing the values for each field changed with the last server update.
|
/// A map containing the values for each field changed with the last server update.
|
||||||
pub fn get_changed_fields_by_position(&self) -> HashMap<usize, Option<String>> {
|
pub fn get_changed_fields_by_position(&self) -> HashMap<usize, String> {
|
||||||
self.changed_fields
|
self.changed_fields
|
||||||
.iter()
|
.iter()
|
||||||
.map(|(name, value)| (self.get_field_position(name), value.clone()))
|
.map(|(name, value)| (self.get_field_position(name), value.clone()))
|
||||||
|
@ -315,7 +315,7 @@ impl LightstreamerClient {
|
|||||||
let mut request_id: usize = 0;
|
let mut request_id: usize = 0;
|
||||||
let mut _session_id: Option<String> = None;
|
let mut _session_id: Option<String> = None;
|
||||||
let mut subscription_id: usize = 0;
|
let mut subscription_id: usize = 0;
|
||||||
let mut item_updates: Vec<Vec<ItemUpdate>> = Vec::new();
|
let mut subscription_item_updates: HashMap<usize, HashMap<usize, ItemUpdate>> = HashMap::new();
|
||||||
loop {
|
loop {
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
message = read_stream.next() => {
|
message = read_stream.next() => {
|
||||||
@ -445,7 +445,7 @@ impl LightstreamerClient {
|
|||||||
// Data updates from server.
|
// Data updates from server.
|
||||||
//
|
//
|
||||||
"u" => {
|
"u" => {
|
||||||
println!("Received data update from server: '{}'", clean_text);
|
//println!("Received data update from server: '{}'", clean_text);
|
||||||
// Parse arguments from the received message.
|
// Parse arguments from the received message.
|
||||||
let arguments = clean_text.split(",").collect::<Vec<&str>>();
|
let arguments = clean_text.split(",").collect::<Vec<&str>>();
|
||||||
//
|
//
|
||||||
@ -464,7 +464,7 @@ impl LightstreamerClient {
|
|||||||
//
|
//
|
||||||
let item_index = arguments.get(2).unwrap_or(&"").parse::<usize>().unwrap_or(0);
|
let item_index = arguments.get(2).unwrap_or(&"").parse::<usize>().unwrap_or(0);
|
||||||
let item = match subscription.get_items() {
|
let item = match subscription.get_items() {
|
||||||
Some(items) => items.get(item_index),
|
Some(items) => items.get(item_index-1),
|
||||||
None => {
|
None => {
|
||||||
println!("No items found in subscription: {:?}", subscription);
|
println!("No items found in subscription: {:?}", subscription);
|
||||||
continue;
|
continue;
|
||||||
@ -486,13 +486,16 @@ impl LightstreamerClient {
|
|||||||
} else {
|
} else {
|
||||||
// If item doesn't exist in item_updates yet, the first update
|
// If item doesn't exist in item_updates yet, the first update
|
||||||
// is always a snapshot.
|
// is always a snapshot.
|
||||||
if let Some(item_updates) = item_updates.get(subscription_index) {
|
if let Some(item_updates) = subscription_item_updates.get(&(subscription_index)) {
|
||||||
if let Some(_) = item_updates.get(item_index) {
|
if let Some(_) = item_updates.get(&(item_index)) {
|
||||||
|
// Item update already exists in item_updates, so it's not a snapshot.
|
||||||
false
|
false
|
||||||
} else {
|
} else {
|
||||||
|
// Item update doesn't exist in item_updates, so the first update is always a snapshot.
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
// Item updates not found for subscription, so the first update is always a snapshot.
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -597,29 +600,36 @@ impl LightstreamerClient {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//println!("Field values: {:?}", field_map);
|
// Store only item_update's changed fields.
|
||||||
|
let changed_fields: HashMap<String, String> = field_map.iter()
|
||||||
let changed_fields: HashMap<String, Option<String>> = field_map.iter()
|
.filter_map(|(k, v)| {
|
||||||
.map(|(k, v)| (k.clone(), v.clone()))
|
if let Some(v) = v {
|
||||||
|
Some((k.clone(), v.clone()))
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
})
|
||||||
.collect();
|
.collect();
|
||||||
//println!("Changed fields: {:?}", changed_fields);
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// Take the proper item_update from item_updates and update it with changed fields.
|
// Take the proper item_update from item_updates and update it with changed fields.
|
||||||
// If the item_update doesn't exist yet, create a new one.
|
// If the item_update doesn't exist yet, create a new one.
|
||||||
//
|
//
|
||||||
match item_updates.get_mut(subscription_index) {
|
let current_item_update: ItemUpdate;
|
||||||
Some(item_updates) => match item_updates.get_mut(item_index) {
|
match subscription_item_updates.get_mut(&(subscription_index)) {
|
||||||
|
Some(item_updates) => match item_updates.get_mut(&(item_index)) {
|
||||||
Some(item_update) => {
|
Some(item_update) => {
|
||||||
//
|
//
|
||||||
// Iterate changed_fields and update existing item_update.fields assigning the new values.
|
// Iterate changed_fields and update existing item_update.fields assigning the new values.
|
||||||
//
|
//
|
||||||
for (field_name, new_value) in &changed_fields {
|
for (field_name, new_value) in &changed_fields {
|
||||||
if item_update.fields.contains_key(field_name) {
|
if item_update.fields.contains_key(field_name) {
|
||||||
item_update.fields.insert((*field_name).clone(), new_value.clone());
|
item_update.fields.insert((*field_name).clone(), Some(new_value.clone()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
item_update.changed_fields = changed_fields;
|
item_update.changed_fields = changed_fields;
|
||||||
|
item_update.is_snapshot = is_snapshot;
|
||||||
|
current_item_update = item_update.clone();
|
||||||
},
|
},
|
||||||
None => {
|
None => {
|
||||||
// Create a new item_update and add it to item_updates.
|
// Create a new item_update and add it to item_updates.
|
||||||
@ -630,7 +640,8 @@ impl LightstreamerClient {
|
|||||||
changed_fields: changed_fields,
|
changed_fields: changed_fields,
|
||||||
is_snapshot: is_snapshot,
|
is_snapshot: is_snapshot,
|
||||||
};
|
};
|
||||||
item_updates.push(item_update);
|
current_item_update = item_update.clone();
|
||||||
|
item_updates.insert(item_index, item_update);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
None => {
|
None => {
|
||||||
@ -642,28 +653,20 @@ impl LightstreamerClient {
|
|||||||
changed_fields: changed_fields,
|
changed_fields: changed_fields,
|
||||||
is_snapshot: is_snapshot,
|
is_snapshot: is_snapshot,
|
||||||
};
|
};
|
||||||
item_updates.push(vec![item_update]);
|
current_item_update = item_update.clone();
|
||||||
|
let mut item_updates = HashMap::new();
|
||||||
|
item_updates.insert(item_index, item_update);
|
||||||
|
subscription_item_updates.insert(subscription_index, item_updates);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
//println!("Item updates: {:?}", item_updates);
|
// Get mutable subscription listeners directly.
|
||||||
|
let subscription_listeners = subscription.get_listeners();
|
||||||
|
|
||||||
println!("Item updates: {}", serde_json::to_string_pretty(&item_updates).unwrap());
|
// Iterate subscription listeners and call on_item_update for each listener.
|
||||||
println!("\n\n");
|
for listener in subscription_listeners {
|
||||||
|
listener.on_item_update(¤t_item_update);
|
||||||
if item_updates.len() >= 3 {
|
|
||||||
return Ok(());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
if let Some(item_updates) = item_updates.get_mut(subscription_index) {
|
|
||||||
if let Some(item_update) = item_updates.get_mut(item_index) {
|
|
||||||
for (field_name, field_value) in changed_fields {
|
|
||||||
item_update.set_field_value(field_name, field_value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
}
|
}
|
||||||
//
|
//
|
||||||
// Connection confirmation from server.
|
// Connection confirmation from server.
|
||||||
|
31
src/main.rs
31
src/main.rs
@ -43,11 +43,34 @@ async fn setup_signal_hook(shutdown_signal: Arc<Notify>) {
|
|||||||
pub struct MySubscriptionListener {}
|
pub struct MySubscriptionListener {}
|
||||||
|
|
||||||
impl SubscriptionListener for MySubscriptionListener {
|
impl SubscriptionListener for MySubscriptionListener {
|
||||||
fn on_item_update(&mut self, update: ItemUpdate) {
|
fn on_item_update(&self, update: &ItemUpdate) {
|
||||||
println!(
|
println!(
|
||||||
"UPDATE {} {}",
|
"UPDATE for item '{}' => '{}': {}, '{}': {}, '{}': {}, '{}': {}, '{}': {}, '{}': {}, '{}': {}, '{}': {}, '{}': {}, '{}': {}, '{}': {}, '{}': {}",
|
||||||
update.get_value("stock_name").unwrap(),
|
update.item_name.as_ref().unwrap_or(&"N/A".to_string()),
|
||||||
update.get_value("last_price").unwrap()
|
"stock_name",
|
||||||
|
update.get_value("stock_name").unwrap_or(&"N/A".to_string()),
|
||||||
|
"last_price",
|
||||||
|
update.get_value("last_price").unwrap_or(&"N/A".to_string()),
|
||||||
|
"time",
|
||||||
|
update.get_value("time").unwrap_or(&"N/A".to_string()),
|
||||||
|
"pct_change",
|
||||||
|
update.get_value("pct_change").unwrap_or(&"N/A".to_string()),
|
||||||
|
"bid_quantity",
|
||||||
|
update.get_value("bid_quantity").unwrap_or(&"N/A".to_string()),
|
||||||
|
"bid",
|
||||||
|
update.get_value("bid").unwrap_or(&"N/A".to_string()),
|
||||||
|
"ask",
|
||||||
|
update.get_value("ask").unwrap_or(&"N/A".to_string()),
|
||||||
|
"ask_quantity",
|
||||||
|
update.get_value("ask_quantity").unwrap_or(&"N/A".to_string()),
|
||||||
|
"min",
|
||||||
|
update.get_value("min").unwrap_or(&"N/A".to_string()),
|
||||||
|
"max",
|
||||||
|
update.get_value("max").unwrap_or(&"N/A".to_string()),
|
||||||
|
"ref_price",
|
||||||
|
update.get_value("ref_price").unwrap_or(&"N/A".to_string()),
|
||||||
|
"open_price",
|
||||||
|
update.get_value("open_price").unwrap_or(&"N/A".to_string()),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -156,7 +156,7 @@ pub trait SubscriptionListener: Send {
|
|||||||
/// - `update`: a value object containing the updated values for all the fields, together with
|
/// - `update`: a value object containing the updated values for all the fields, together with
|
||||||
/// meta-information about the update itself and some helper methods that can be used to
|
/// meta-information about the update itself and some helper methods that can be used to
|
||||||
/// iterate through all or new values.
|
/// iterate through all or new values.
|
||||||
fn on_item_update(&mut self, _update: ItemUpdate) {
|
fn on_item_update(&self, _update: &ItemUpdate) {
|
||||||
// Default implementation does nothing.
|
// Default implementation does nothing.
|
||||||
unimplemented!("Implement on_item_update method for SubscriptionListener.");
|
unimplemented!("Implement on_item_update method for SubscriptionListener.");
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user