use std::{ffi::c_void, time::Duration};
use mssf_com::{
FabricClient::{
IFabricGetApplicationListResult2, IFabricGetDeployedServiceReplicaDetailResult,
IFabricGetNodeListResult2, IFabricGetPartitionListResult2,
IFabricGetPartitionLoadInformationResult, IFabricGetReplicaListResult2,
IFabricQueryClient10,
},
FabricTypes::{
FABRIC_APPLICATION_QUERY_DESCRIPTION,
FABRIC_DEPLOYED_SERVICE_REPLICA_DETAIL_QUERY_DESCRIPTION, FABRIC_NODE_QUERY_DESCRIPTION,
FABRIC_NODE_QUERY_DESCRIPTION_EX1, FABRIC_NODE_QUERY_DESCRIPTION_EX2,
FABRIC_NODE_QUERY_DESCRIPTION_EX3, FABRIC_PARTITION_LOAD_INFORMATION_QUERY_DESCRIPTION,
FABRIC_SERVICE_PARTITION_QUERY_DESCRIPTION, FABRIC_SERVICE_QUERY_DESCRIPTION,
FABRIC_SERVICE_REPLICA_QUERY_DESCRIPTION,
},
};
use crate::types::{
DeployedServiceReplicaDetailQueryDescription, DeployedServiceReplicaDetailQueryResult,
GetPartitionLoadInformationResult, NodeListResult, NodeQueryDescription,
PartitionLoadInformationQueryDescription, ServicePartitionList,
ServicePartitionQueryDescription, ServiceReplicaList, ServiceReplicaQueryDescription,
};
use crate::{
runtime::executor::BoxedCancelToken,
sync::{FabricReceiver, fabric_begin_end_proxy},
types::ServiceQueryDescription,
};
#[derive(Debug, Clone)]
pub struct QueryClient {
com: IFabricQueryClient10,
}
impl QueryClient {
pub fn get_node_list_internal(
&self,
query_description: &FABRIC_NODE_QUERY_DESCRIPTION,
timeout_milliseconds: u32,
cancellation_token: Option<BoxedCancelToken>,
) -> FabricReceiver<crate::WinResult<IFabricGetNodeListResult2>> {
let com1 = &self.com;
let com2 = self.com.clone();
fabric_begin_end_proxy(
move |callback| unsafe {
com1.BeginGetNodeList(query_description, timeout_milliseconds, callback)
},
move |ctx| unsafe { com2.EndGetNodeList2(ctx) },
cancellation_token,
)
}
pub fn get_application_list_internal(
&self,
query_description: &FABRIC_APPLICATION_QUERY_DESCRIPTION,
timeout_milliseconds: u32,
cancellation_token: Option<BoxedCancelToken>,
) -> FabricReceiver<crate::WinResult<IFabricGetApplicationListResult2>> {
let com1 = &self.com;
let com2 = self.com.clone();
fabric_begin_end_proxy(
move |callback| unsafe {
com1.BeginGetApplicationList(query_description, timeout_milliseconds, callback)
},
move |ctx| unsafe { com2.EndGetApplicationList2(ctx) },
cancellation_token,
)
}
fn get_service_list_internal(
&self,
desc: &FABRIC_SERVICE_QUERY_DESCRIPTION,
timeout_milliseconds: u32,
cancellation_token: Option<BoxedCancelToken>,
) -> FabricReceiver<crate::WinResult<mssf_com::FabricClient::IFabricGetServiceListResult2>>
{
let com1 = &self.com;
let com2 = self.com.clone();
fabric_begin_end_proxy(
move |callback| unsafe {
com1.BeginGetServiceList(desc, timeout_milliseconds, callback)
},
move |ctx| unsafe { com2.EndGetServiceList2(ctx) },
cancellation_token,
)
}
fn get_partition_list_internal(
&self,
desc: &FABRIC_SERVICE_PARTITION_QUERY_DESCRIPTION,
timeout_milliseconds: u32,
cancellation_token: Option<BoxedCancelToken>,
) -> FabricReceiver<crate::WinResult<IFabricGetPartitionListResult2>> {
let com1 = &self.com;
let com2 = self.com.clone();
fabric_begin_end_proxy(
move |callback| unsafe {
com1.BeginGetPartitionList(desc, timeout_milliseconds, callback)
},
move |ctx| unsafe { com2.EndGetPartitionList2(ctx) },
cancellation_token,
)
}
fn get_replica_list_internal(
&self,
desc: &FABRIC_SERVICE_REPLICA_QUERY_DESCRIPTION,
timeout_milliseconds: u32,
cancellation_token: Option<BoxedCancelToken>,
) -> FabricReceiver<crate::WinResult<IFabricGetReplicaListResult2>> {
let com1 = &self.com;
let com2 = self.com.clone();
fabric_begin_end_proxy(
move |callback| unsafe {
com1.BeginGetReplicaList(desc, timeout_milliseconds, callback)
},
move |ctx| unsafe { com2.EndGetReplicaList2(ctx) },
cancellation_token,
)
}
fn get_partition_load_information_internal(
&self,
desc: &FABRIC_PARTITION_LOAD_INFORMATION_QUERY_DESCRIPTION,
timeout_milliseconds: u32,
cancellation_token: Option<BoxedCancelToken>,
) -> FabricReceiver<crate::WinResult<IFabricGetPartitionLoadInformationResult>> {
let com1 = &self.com;
let com2 = self.com.clone();
fabric_begin_end_proxy(
move |callback| unsafe {
com1.BeginGetPartitionLoadInformation(desc, timeout_milliseconds, callback)
},
move |ctx| unsafe { com2.EndGetPartitionLoadInformation(ctx) },
cancellation_token,
)
}
fn get_deployed_replica_detail_internal(
&self,
desc: &FABRIC_DEPLOYED_SERVICE_REPLICA_DETAIL_QUERY_DESCRIPTION,
timeout_milliseconds: u32,
cancellation_token: Option<BoxedCancelToken>,
) -> FabricReceiver<crate::WinResult<IFabricGetDeployedServiceReplicaDetailResult>> {
let com1 = &self.com;
let com2 = self.com.clone();
fabric_begin_end_proxy(
move |callback| unsafe {
com1.BeginGetDeployedReplicaDetail(desc, timeout_milliseconds, callback)
},
move |ctx| unsafe { com2.EndGetDeployedReplicaDetail(ctx) },
cancellation_token,
)
}
}
impl From<IFabricQueryClient10> for QueryClient {
fn from(com: IFabricQueryClient10) -> Self {
Self { com }
}
}
impl From<QueryClient> for IFabricQueryClient10 {
fn from(value: QueryClient) -> Self {
value.com
}
}
impl QueryClient {
pub async fn get_node_list(
&self,
desc: &NodeQueryDescription,
timeout: Duration,
cancellation_token: Option<BoxedCancelToken>,
) -> crate::Result<NodeListResult> {
let com = {
let ex3 = FABRIC_NODE_QUERY_DESCRIPTION_EX3 {
MaxResults: desc.paged_query.max_results.unwrap_or(0),
Reserved: std::ptr::null_mut(),
};
let ex2 = FABRIC_NODE_QUERY_DESCRIPTION_EX2 {
NodeStatusFilter: desc.node_status_filter.bits(),
Reserved: std::ptr::addr_of!(ex3) as *mut c_void,
};
let ex1 = FABRIC_NODE_QUERY_DESCRIPTION_EX1 {
ContinuationToken: desc.paged_query.continuation_token.as_ref().into(),
Reserved: std::ptr::addr_of!(ex2) as *mut c_void,
};
let arg = FABRIC_NODE_QUERY_DESCRIPTION {
NodeNameFilter: desc.node_name_filter.as_ref().into(),
Reserved: std::ptr::addr_of!(ex1) as *mut c_void,
};
self.get_node_list_internal(
&arg,
timeout.as_millis().try_into().unwrap(),
cancellation_token,
)
}
.await??;
Ok(NodeListResult::from(&com))
}
pub async fn get_application_list(
&self,
desc: &crate::types::ApplicationQueryDescription,
timeout: Duration,
cancellation_token: Option<BoxedCancelToken>,
) -> crate::Result<crate::types::ApplicationListResult> {
let com = {
let (mut base, mut ex1, mut ex2, mut ex3, ex4) = desc.get_raw_parts();
base.Reserved = std::ptr::addr_of!(ex1) as *mut c_void;
ex1.Reserved = std::ptr::addr_of!(ex2) as *mut c_void;
ex2.Reserved = std::ptr::addr_of!(ex3) as *mut c_void;
ex3.Reserved = std::ptr::addr_of!(ex4) as *mut c_void;
self.get_application_list_internal(
&base,
timeout.as_millis().try_into().unwrap(),
cancellation_token,
)
}
.await??;
Ok(crate::types::ApplicationListResult::from(&com))
}
pub async fn get_service_list(
&self,
desc: &ServiceQueryDescription,
timeout: Duration,
cancellation_token: Option<BoxedCancelToken>,
) -> crate::Result<crate::types::ServiceListResult> {
let com = {
let (mut base, mut ex1, mut ex2, ex3) = desc.get_raw_parts();
base.Reserved = &ex1 as *const _ as *mut c_void;
ex1.Reserved = &ex2 as *const _ as *mut c_void;
ex2.Reserved = &ex3 as *const _ as *mut c_void;
self.get_service_list_internal(&base, timeout.as_millis() as u32, cancellation_token)
}
.await??;
Ok(crate::types::ServiceListResult::from(&com))
}
pub async fn get_partition_list(
&self,
desc: &ServicePartitionQueryDescription,
timeout: Duration,
cancellation_token: Option<BoxedCancelToken>,
) -> crate::Result<ServicePartitionList> {
let com = {
let raw: FABRIC_SERVICE_PARTITION_QUERY_DESCRIPTION = desc.into();
let mili = timeout.as_millis() as u32;
self.get_partition_list_internal(&raw, mili, cancellation_token)
}
.await??;
Ok(ServicePartitionList::from(&com))
}
pub async fn get_replica_list(
&self,
desc: &ServiceReplicaQueryDescription,
timeout: Duration,
cancellation_token: Option<BoxedCancelToken>,
) -> crate::Result<ServiceReplicaList> {
let com = {
let raw: FABRIC_SERVICE_REPLICA_QUERY_DESCRIPTION = desc.into();
let mili = timeout.as_millis() as u32;
self.get_replica_list_internal(&raw, mili, cancellation_token)
}
.await??;
Ok(ServiceReplicaList::from(&com))
}
pub async fn get_partition_load_information(
&self,
desc: &PartitionLoadInformationQueryDescription,
timeout: Duration,
cancellation_token: Option<BoxedCancelToken>,
) -> crate::Result<GetPartitionLoadInformationResult> {
let com = {
let raw: FABRIC_PARTITION_LOAD_INFORMATION_QUERY_DESCRIPTION = desc.into();
let timeout_ms = timeout.as_micros() as u32;
self.get_partition_load_information_internal(&raw, timeout_ms, cancellation_token)
}
.await??;
Ok(GetPartitionLoadInformationResult::from(&com))
}
pub async fn get_deployed_replica_detail(
&self,
desc: &DeployedServiceReplicaDetailQueryDescription,
timeout: Duration,
cancellation_token: Option<BoxedCancelToken>,
) -> crate::Result<DeployedServiceReplicaDetailQueryResult> {
let com = {
let raw: FABRIC_DEPLOYED_SERVICE_REPLICA_DETAIL_QUERY_DESCRIPTION = desc.into();
let timeout_ms = timeout.as_micros() as u32;
self.get_deployed_replica_detail_internal(&raw, timeout_ms, cancellation_token)
}
.await??;
Ok(DeployedServiceReplicaDetailQueryResult::new(com))
}
}