use crate::ap::authenticator::Authenticator;
use crate::ap::event::*;
use crate::ap::remote_client::RemoteClient;
use crate::ap::{aid, Context, RsnCfg};
use anyhow::{ensure, format_err};
use ieee80211::MacAddr;
use log::error;
use std::sync::{Arc, Mutex};
use wlan_common::ie::rsn::rsne;
use wlan_common::ie::{intersect, SupportedRate};
use wlan_common::mac::{Aid, CapabilityInfo};
use wlan_common::timer::EventId;
use wlan_rsn::gtk::GtkProvider;
use wlan_rsn::nonce::NonceReader;
use wlan_rsn::rsna::{SecAssocStatus, SecAssocUpdate, UpdateSink};
use wlan_rsn::{NegotiatedProtection, ProtectionInfo};
use wlan_statemachine::*;
use {
fidl_fuchsia_wlan_common as fidl_common, fidl_fuchsia_wlan_ieee80211 as fidl_ieee80211,
fidl_fuchsia_wlan_mlme as fidl_mlme,
};
const ASSOCIATION_TIMEOUT_SECONDS: i64 = 300;
pub struct Authenticating;
impl Authenticating {
fn handle_auth_ind(
&self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
auth_type: fidl_mlme::AuthenticationTypes,
) -> Result<EventId, anyhow::Error> {
if auth_type != fidl_mlme::AuthenticationTypes::OpenSystem {
return Err(format_err!("unsupported authentication type: {:?}", auth_type));
}
let event = ClientEvent::AssociationTimeout;
let timeout_event_id = r_sta.schedule_at(
ctx,
zx::MonotonicInstant::after(zx::MonotonicDuration::from_seconds(
ASSOCIATION_TIMEOUT_SECONDS,
)),
event,
);
Ok(timeout_event_id)
}
}
fn new_authenticator_from_rsne(
device_addr: MacAddr,
client_addr: MacAddr,
s_rsne_bytes: &[u8],
a_rsn: &RsnCfg,
) -> Result<Box<dyn Authenticator>, anyhow::Error> {
let (_, s_rsne) =
rsne::from_bytes(s_rsne_bytes).map_err(|e| format_err!("failed to parse RSNE: {:?}", e))?;
ensure!(s_rsne.is_valid_subset_of(&a_rsn.rsne)?, "incompatible client RSNE");
let nonce_reader = NonceReader::new(&device_addr)?;
let gtk_provider =
GtkProvider::new(NegotiatedProtection::from_rsne(&s_rsne)?.group_data, 1, 0)?;
Ok(Box::new(wlan_rsn::Authenticator::new_wpa2psk_ccmp128(
nonce_reader,
Arc::new(Mutex::new(gtk_provider)),
a_rsn.psk.clone(),
client_addr,
ProtectionInfo::Rsne(s_rsne),
device_addr,
ProtectionInfo::Rsne(a_rsn.rsne.clone()),
)?))
}
pub struct Authenticated {
timeout_event_id: EventId,
}
struct AssociationError {
error: anyhow::Error,
result_code: fidl_mlme::AssociateResultCode,
reason_code: fidl_ieee80211::ReasonCode,
}
struct Association {
aid: Aid,
capabilities: CapabilityInfo,
rates: Vec<SupportedRate>,
rsna_link_state: Option<RsnaLinkState>,
}
impl Authenticated {
#[allow(clippy::too_many_arguments, reason = "mass allow for https://fxbug.dev/381896734")]
fn handle_assoc_ind(
&self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
aid_map: &mut aid::Map,
ap_capabilities: CapabilityInfo,
client_capablities: u16,
ap_rates: &[SupportedRate],
client_rates: &[SupportedRate],
rsn_cfg: &Option<RsnCfg>,
s_rsne: Option<Vec<u8>>,
) -> Result<Association, AssociationError> {
let rsna_link_state = match (s_rsne.as_ref(), rsn_cfg) {
(Some(s_rsne_bytes), Some(a_rsn)) => {
let authenticator = new_authenticator_from_rsne(
ctx.device_info.sta_addr.into(),
r_sta.addr,
s_rsne_bytes,
a_rsn,
)
.map_err(|error| AssociationError {
error,
result_code: fidl_mlme::AssociateResultCode::RefusedCapabilitiesMismatch,
reason_code: fidl_ieee80211::ReasonCode::Ieee8021XAuthFailed,
})?;
Some(RsnaLinkState::new(authenticator))
}
(None, None) => None,
_ => {
return Err(AssociationError {
error: format_err!("unexpected RSN element: {:?}", s_rsne),
result_code: fidl_mlme::AssociateResultCode::RefusedCapabilitiesMismatch,
reason_code: fidl_ieee80211::ReasonCode::ReasonInvalidElement,
});
}
};
let aid = aid_map.assign_aid().map_err(|error| AssociationError {
error,
result_code: fidl_mlme::AssociateResultCode::RefusedReasonUnspecified,
reason_code: fidl_ieee80211::ReasonCode::UnspecifiedReason,
})?;
let (capabilities, rates) = if ctx.mac_sublayer_support.device.mac_implementation_type
== fidl_common::MacImplementationType::Softmac
{
let capabilities = CapabilityInfo(client_capablities & ap_capabilities.raw());
let rates = intersect::intersect_rates(
intersect::ApRates(ap_rates),
intersect::ClientRates(client_rates),
)
.map_err(|error| AssociationError {
error: format_err!(
"could not intersect rates ({:?} + {:?}): {:?}",
ap_rates,
client_rates,
error
),
result_code: match error {
intersect::IntersectRatesError::BasicRatesMismatch => {
fidl_mlme::AssociateResultCode::RefusedBasicRatesMismatch
}
intersect::IntersectRatesError::NoApRatesSupported => {
fidl_mlme::AssociateResultCode::RefusedCapabilitiesMismatch
}
},
reason_code: fidl_ieee80211::ReasonCode::ReasonInvalidElement,
})?;
(capabilities, rates)
} else {
(CapabilityInfo(client_capablities), client_rates.to_vec())
};
Ok(Association {
capabilities: capabilities
.with_privacy(rsna_link_state.is_some()),
aid,
rates,
rsna_link_state,
})
}
fn handle_association_timeout(
&self,
_unused_r_sta: &mut RemoteClient,
_unused_ctx: &mut Context,
timeout_event_id: EventId,
) -> Result<(), ()> {
if timeout_event_id != self.timeout_event_id {
return Ok(());
}
Err(())
}
}
#[derive(Debug)]
struct RsnaLinkState {
authenticator: Box<dyn Authenticator>,
last_key_frame: Option<eapol::KeyFrameBuf>,
request_attempts: usize,
request_timeout_event_id: Option<EventId>,
negotiation_timeout_event_id: Option<EventId>,
}
pub const RSNA_NEGOTIATION_REQUEST_MAX_ATTEMPTS: usize = 4;
pub const RSNA_NEGOTIATION_REQUEST_TIMEOUT_SECONDS: i64 = 1;
pub const RSNA_NEGOTIATION_TIMEOUT_SECONDS: i64 = 5;
impl RsnaLinkState {
fn new(authenticator: Box<dyn Authenticator>) -> Self {
Self {
authenticator,
last_key_frame: None,
request_attempts: 0,
request_timeout_event_id: None,
negotiation_timeout_event_id: None,
}
}
fn initiate_key_exchange(
&mut self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
) -> Result<(), anyhow::Error> {
let mut update_sink = vec![];
self.authenticator.reset();
self.authenticator.initiate(&mut update_sink)?;
self.process_authenticator_updates(r_sta, ctx, &update_sink);
if self.last_key_frame.is_none() {
return Err(format_err!("no key frame was produced on authenticator initiation"));
}
self.negotiation_timeout_event_id = Some(r_sta.schedule_at(
ctx,
zx::MonotonicInstant::after(zx::MonotonicDuration::from_seconds(
RSNA_NEGOTIATION_TIMEOUT_SECONDS,
)),
ClientEvent::RsnaTimeout(RsnaTimeout::Negotiation),
));
self.reschedule_request_timeout(r_sta, ctx);
Ok(())
}
fn reschedule_request_timeout(&mut self, r_sta: &mut RemoteClient, ctx: &mut Context) {
self.request_timeout_event_id = Some(r_sta.schedule_at(
ctx,
zx::MonotonicInstant::after(zx::MonotonicDuration::from_seconds(
RSNA_NEGOTIATION_REQUEST_TIMEOUT_SECONDS,
)),
ClientEvent::RsnaTimeout(RsnaTimeout::Request),
));
}
fn handle_rsna_timeout(
&mut self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
timeout_event_id: EventId,
timeout_type: RsnaTimeout,
) -> Result<(), RsnaNegotiationError> {
match timeout_type {
RsnaTimeout::Request => self.handle_rsna_request_timeout(r_sta, ctx, timeout_event_id),
RsnaTimeout::Negotiation => self.handle_rsna_negotiation_timeout(timeout_event_id),
}
}
fn handle_rsna_request_timeout(
&mut self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
timeout_event_id: EventId,
) -> Result<(), RsnaNegotiationError> {
if self.request_timeout_event_id != Some(timeout_event_id) {
return Ok(());
}
self.request_timeout_event_id = None;
self.request_attempts += 1;
if self.request_attempts >= RSNA_NEGOTIATION_REQUEST_MAX_ATTEMPTS {
return Err(RsnaNegotiationError::Timeout);
}
let frame = self
.last_key_frame
.as_ref()
.ok_or(RsnaNegotiationError::Error(format_err!("no key frame available to resend?")))?;
r_sta.send_eapol_req(ctx, frame.clone());
self.reschedule_request_timeout(r_sta, ctx);
Ok(())
}
fn handle_rsna_negotiation_timeout(
&mut self,
timeout_event_id: EventId,
) -> Result<(), RsnaNegotiationError> {
if self.negotiation_timeout_event_id != Some(timeout_event_id) {
return Ok(());
}
self.negotiation_timeout_event_id = None;
Err(RsnaNegotiationError::Timeout)
}
fn process_authenticator_updates(
&mut self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
update_sink: &UpdateSink,
) {
for update in update_sink {
match update {
SecAssocUpdate::TxEapolKeyFrame { frame, .. } => {
r_sta.send_eapol_req(ctx, frame.clone());
self.last_key_frame = Some(frame.clone());
}
SecAssocUpdate::Key(key) => r_sta.send_key(ctx, key),
SecAssocUpdate::Status(status) => {
if *status == SecAssocStatus::EssSaEstablished {
r_sta.send_set_controlled_port_req(
ctx,
fidl_mlme::ControlledPortState::Open,
);
self.last_key_frame = None;
self.request_timeout_event_id = None;
self.negotiation_timeout_event_id = None;
}
}
update => error!("Unhandled association update: {:?}", update),
}
}
}
fn handle_eapol_frame(
&mut self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
data: &[u8],
) -> Result<(), anyhow::Error> {
self.request_attempts = 0;
let authenticator = self.authenticator.as_mut();
let key_frame = eapol::KeyFrameRx::parse(
authenticator.get_negotiated_protection().mic_size as usize,
data,
)?;
let mut update_sink = vec![];
authenticator.on_eapol_frame(&mut update_sink, eapol::Frame::Key(key_frame))?;
self.process_authenticator_updates(r_sta, ctx, &update_sink);
Ok(())
}
fn handle_eapol_conf(
&mut self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
result: fidl_mlme::EapolResultCode,
) -> Result<(), anyhow::Error> {
let authenticator = self.authenticator.as_mut();
let mut update_sink = vec![];
authenticator.on_eapol_conf(&mut update_sink, result)?;
self.process_authenticator_updates(r_sta, ctx, &update_sink);
Ok(())
}
}
pub struct Associated {
aid: Aid,
rsna_link_state: Option<RsnaLinkState>,
}
enum RsnaNegotiationError {
Error(anyhow::Error),
Timeout,
}
impl Associated {
fn aid(&self) -> Aid {
self.aid
}
fn handle_rsna_timeout(
&mut self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
timeout_event_id: EventId,
timeout_type: RsnaTimeout,
) -> Result<(), RsnaNegotiationError> {
match self.rsna_link_state.as_mut() {
Some(rsna_link_state) => {
rsna_link_state.handle_rsna_timeout(r_sta, ctx, timeout_event_id, timeout_type)
}
None => Ok(()),
}
}
fn handle_eapol_ind(
&mut self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
data: &[u8],
) -> Result<(), anyhow::Error> {
match self.rsna_link_state.as_mut() {
Some(rsna_link_state) => rsna_link_state.handle_eapol_frame(r_sta, ctx, data),
None => Err(format_err!("received EAPoL indication without RSNA link state")),
}
}
fn handle_eapol_conf(
&mut self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
result: fidl_mlme::EapolResultCode,
) -> Result<(), anyhow::Error> {
match self.rsna_link_state.as_mut() {
Some(rsna_link_state) => rsna_link_state.handle_eapol_conf(r_sta, ctx, result),
None => Err(format_err!("received EAPoL confirm without RSNA link state")),
}
}
fn handle_disassoc_ind(
&self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
aid_map: &mut aid::Map,
) -> EventId {
aid_map.release_aid(self.aid);
let event = ClientEvent::AssociationTimeout;
r_sta.schedule_at(
ctx,
zx::MonotonicInstant::after(zx::MonotonicDuration::from_seconds(
ASSOCIATION_TIMEOUT_SECONDS,
)),
event,
)
}
}
statemachine!(
pub enum States,
() => Authenticating,
Authenticating => Authenticated,
Authenticated => Associated,
Associated => Authenticated,
Authenticated => Authenticating,
Associated => Authenticating,
);
impl States {
pub fn new_initial() -> States {
States::from(State::new(Authenticating))
}
pub fn aid(&self) -> Option<Aid> {
match self {
States::Associated(state) => Some(state.aid()),
_ => None,
}
}
pub fn authenticated(&self) -> bool {
!matches!(self, States::Authenticating(..))
}
pub fn handle_auth_ind(
self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
auth_type: fidl_mlme::AuthenticationTypes,
) -> States {
match self {
States::Authenticating(state) => match state.handle_auth_ind(r_sta, ctx, auth_type) {
Ok(timeout_event_id) => {
r_sta.send_authenticate_resp(ctx, fidl_mlme::AuthenticateResultCode::Success);
state.transition_to(Authenticated { timeout_event_id }).into()
}
Err(e) => {
error!("client {:02X?} MLME-AUTHENTICATE.indication: {}", r_sta.addr, e);
r_sta.send_authenticate_resp(ctx, fidl_mlme::AuthenticateResultCode::Refused);
state.into()
}
},
_ => {
r_sta.send_authenticate_resp(ctx, fidl_mlme::AuthenticateResultCode::Refused);
self
}
}
}
#[allow(clippy::too_many_arguments, reason = "mass allow for https://fxbug.dev/381896734")]
pub fn handle_assoc_ind(
self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
aid_map: &mut aid::Map,
ap_capabilities: CapabilityInfo,
client_capablities: u16,
ap_rates: &[SupportedRate],
client_rates: &[SupportedRate],
rsn_cfg: &Option<RsnCfg>,
s_rsne: Option<Vec<u8>>,
) -> States {
match self {
States::Authenticated(state) => {
match state.handle_assoc_ind(
r_sta,
ctx,
aid_map,
ap_capabilities,
client_capablities,
ap_rates,
client_rates,
rsn_cfg,
s_rsne,
) {
Ok(Association { aid, capabilities, rates, mut rsna_link_state }) => {
r_sta.send_associate_resp(
ctx,
fidl_mlme::AssociateResultCode::Success,
aid,
capabilities,
rates,
);
if let Some(rsna_link_state) = rsna_link_state.as_mut() {
if let Err(error) = rsna_link_state.initiate_key_exchange(r_sta, ctx) {
error!(
"client {:02X?} MLME-ASSOCIATE.indication (key exchange): {}",
r_sta.addr, error
);
r_sta.send_deauthenticate_req(
ctx,
fidl_ieee80211::ReasonCode::Ieee8021XAuthFailed,
);
return state.transition_to(Authenticating).into();
}
}
state.transition_to(Associated { aid, rsna_link_state }).into()
}
Err(AssociationError { error, result_code, reason_code }) => {
error!("client {:02X?} MLME-ASSOCIATE.indication: {}", r_sta.addr, error);
r_sta.send_associate_resp(ctx, result_code, 0, CapabilityInfo(0), vec![]);
r_sta.send_deauthenticate_req(ctx, reason_code);
state.transition_to(Authenticating).into()
}
}
}
_ => {
r_sta.send_associate_resp(
ctx,
fidl_mlme::AssociateResultCode::RefusedReasonUnspecified,
0,
CapabilityInfo(0),
vec![],
);
self
}
}
}
pub fn handle_disassoc_ind(
self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
aid_map: &mut aid::Map,
) -> States {
match self {
States::Associated(state) => {
let timeout_event_id = state.handle_disassoc_ind(r_sta, ctx, aid_map);
state.transition_to(Authenticated { timeout_event_id }).into()
}
_ => self,
}
}
pub fn handle_eapol_ind(
self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
data: &[u8],
) -> States {
match self {
States::Associated(mut state) => {
if let Err(e) = state.handle_eapol_ind(r_sta, ctx, data) {
error!("client {:02X?} EAPOL.indication: {}", r_sta.addr, e);
}
state.into()
}
_ => self,
}
}
pub fn handle_eapol_conf(
self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
result: fidl_mlme::EapolResultCode,
) -> States {
match self {
States::Associated(mut state) => {
if let Err(e) = state.handle_eapol_conf(r_sta, ctx, result) {
error!("client {:02X?} EAPOL.confirm: {}", r_sta.addr, e);
}
state.into()
}
_ => self,
}
}
pub fn handle_timeout(
self,
r_sta: &mut RemoteClient,
ctx: &mut Context,
timeout_event_id: EventId,
event: ClientEvent,
) -> States {
match event {
ClientEvent::AssociationTimeout => match self {
States::Authenticated(state) => {
match state.handle_association_timeout(r_sta, ctx, timeout_event_id) {
Ok(()) => state.into(),
Err(()) => {
r_sta.send_deauthenticate_req(
ctx,
fidl_ieee80211::ReasonCode::InvalidAuthentication,
);
state.transition_to(Authenticating).into()
}
}
}
States::Associated(state) => {
state.into()
}
_ => {
error!(
"client {:02X?} received AssociationTimeout in unexpected state; \
ignoring timeout",
r_sta.addr,
);
self
}
},
ClientEvent::RsnaTimeout(timeout_type) => match self {
States::Associated(state) => {
let (transition, mut state) = state.release_data();
match state.handle_rsna_timeout(r_sta, ctx, timeout_event_id, timeout_type) {
Ok(()) => transition.to(state).into(),
Err(e) => {
let reason_code = match e {
RsnaNegotiationError::Error(e) => {
error!(
"client {:02X?} RSNA negotiation error: {}",
r_sta.addr, e
);
fidl_ieee80211::ReasonCode::UnspecifiedReason
}
RsnaNegotiationError::Timeout => {
fidl_ieee80211::ReasonCode::FourwayHandshakeTimeout
}
};
r_sta.send_deauthenticate_req(ctx, reason_code);
transition.to(Authenticating).into()
}
}
}
_ => {
error!(
"client {:02X?} received RsnaTimeout in unexpected state; \
ignoring timeout",
r_sta.addr,
);
self
}
},
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::ap::create_rsn_cfg;
use crate::ap::test_utils::MockAuthenticator;
use crate::{test_utils, MlmeRequest, MlmeSink, MlmeStream};
use futures::channel::mpsc;
use ieee80211::{MacAddrBytes, Ssid};
use lazy_static::lazy_static;
use wlan_common::ie::rsn::akm::AKM_PSK;
use wlan_common::ie::rsn::cipher::{CIPHER_CCMP_128, CIPHER_GCMP_256};
use wlan_common::ie::rsn::rsne::Rsne;
use wlan_common::test_utils::fake_features::fake_mac_sublayer_support;
use wlan_common::{assert_variant, timer};
use wlan_rsn::key::exchange::Key;
lazy_static! {
static ref AP_ADDR: MacAddr = [6u8; 6].into();
static ref CLIENT_ADDR: MacAddr = [7u8; 6].into();
}
fn make_remote_client() -> RemoteClient {
RemoteClient::new(*CLIENT_ADDR)
}
fn make_env() -> (Context, MlmeStream, timer::EventStream<Event>) {
let device_info = test_utils::fake_device_info(*AP_ADDR);
let mac_sublayer_support = fake_mac_sublayer_support();
let (mlme_sink, mlme_stream) = mpsc::unbounded();
let (timer, time_stream) = timer::create_timer();
let ctx = Context {
device_info,
mac_sublayer_support,
mlme_sink: MlmeSink::new(mlme_sink),
timer,
};
(ctx, mlme_stream, time_stream)
}
#[test]
fn authenticating_goes_to_authenticated() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, mut time_stream) = make_env();
let state = States::from(State::new(Authenticating));
let state =
state.handle_auth_ind(&mut r_sta, &mut ctx, fidl_mlme::AuthenticationTypes::OpenSystem);
let (_, Authenticated { timeout_event_id }) = match state {
States::Authenticated(state) => state.release_data(),
_ => panic!("unexpected state"),
};
let (_, timed_event) = time_stream.try_next().unwrap().expect("expected timed event");
assert_eq!(timed_event.id, timeout_event_id);
assert_variant!(timed_event.event, Event::Client { addr, event } => {
assert_eq!(addr, *CLIENT_ADDR);
assert_variant!(event, ClientEvent::AssociationTimeout);
});
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AuthResponse(fidl_mlme::AuthenticateResponse {
peer_sta_address,
result_code,
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(result_code, fidl_mlme::AuthenticateResultCode::Success);
});
}
#[test]
fn authenticating_stays_authenticating_with_unsupported_authentication_type() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state = States::from(State::new(Authenticating));
let state =
state.handle_auth_ind(&mut r_sta, &mut ctx, fidl_mlme::AuthenticationTypes::SharedKey);
let (_, Authenticating) = match state {
States::Authenticating(state) => state.release_data(),
_ => panic!("unexpected state"),
};
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AuthResponse(fidl_mlme::AuthenticateResponse {
peer_sta_address,
result_code,
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(result_code, fidl_mlme::AuthenticateResultCode::Refused);
});
}
#[test]
fn authenticating_refuses_association() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state = States::from(State::new(Authenticating));
let mut aid_map = aid::Map::default();
let state = state.handle_assoc_ind(
&mut r_sta,
&mut ctx,
&mut aid_map,
CapabilityInfo(0).with_short_preamble(true),
CapabilityInfo(0).with_short_preamble(true).raw(),
&[SupportedRate(0b11111000)][..],
&[SupportedRate(0b11111000)][..],
&None,
None,
);
let (_, Authenticating) = match state {
States::Authenticating(state) => state.release_data(),
_ => panic!("unexpected state"),
};
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AssocResponse(fidl_mlme::AssociateResponse {
peer_sta_address,
association_id,
result_code,
capability_info,
rates,
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(association_id, 0);
assert_eq!(result_code, fidl_mlme::AssociateResultCode::RefusedReasonUnspecified);
assert_eq!(capability_info, 0);
assert_eq!(rates, Vec::<u8>::new());
});
}
#[test]
fn authenticated_refuses_authentication() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States =
State::new(Authenticating).transition_to(Authenticated { timeout_event_id: 1 }).into();
let state =
state.handle_auth_ind(&mut r_sta, &mut ctx, fidl_mlme::AuthenticationTypes::SharedKey);
let (_, Authenticated { .. }) = match state {
States::Authenticated(state) => state.release_data(),
_ => panic!("unexpected state"),
};
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AuthResponse(fidl_mlme::AuthenticateResponse {
peer_sta_address,
result_code,
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(result_code, fidl_mlme::AuthenticateResultCode::Refused);
});
}
#[test]
fn authenticated_deauthenticates_on_timeout() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States =
State::new(Authenticating).transition_to(Authenticated { timeout_event_id: 1 }).into();
let state = state.handle_timeout(&mut r_sta, &mut ctx, 1, ClientEvent::AssociationTimeout);
let (_, Authenticating) = match state {
States::Authenticating(state) => state.release_data(),
_ => panic!("unexpected state"),
};
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::Deauthenticate(fidl_mlme::DeauthenticateRequest {
peer_sta_address,
reason_code,
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(reason_code, fidl_ieee80211::ReasonCode::InvalidAuthentication);
});
}
#[test]
fn authenticated_does_not_send_deauthentication_on_wrong_timeout() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States =
State::new(Authenticating).transition_to(Authenticated { timeout_event_id: 1 }).into();
let state = state.handle_timeout(&mut r_sta, &mut ctx, 2, ClientEvent::AssociationTimeout);
let (_, Authenticated { .. }) = match state {
States::Authenticated(state) => state.release_data(),
_ => panic!("unexpected state"),
};
assert_variant!(mlme_stream.try_next(), Err(_));
}
#[test]
fn authenticated_goes_to_associated_no_rsn() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States =
State::new(Authenticating).transition_to(Authenticated { timeout_event_id: 1 }).into();
let mut aid_map = aid::Map::default();
let state = state.handle_assoc_ind(
&mut r_sta,
&mut ctx,
&mut aid_map,
CapabilityInfo(0).with_short_preamble(true),
CapabilityInfo(0).with_short_preamble(true).raw(),
&[SupportedRate(0b11111000)][..],
&[SupportedRate(0b11111000)][..],
&None,
None,
);
let (_, Associated { rsna_link_state, aid }) = match state {
States::Associated(state) => state.release_data(),
_ => panic!("unexpected_state"),
};
assert_variant!(rsna_link_state, None);
assert_eq!(aid, 1);
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AssocResponse(fidl_mlme::AssociateResponse {
peer_sta_address,
result_code,
capability_info,
rates,
..
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(result_code, fidl_mlme::AssociateResultCode::Success);
assert_eq!(capability_info, CapabilityInfo(0).with_short_preamble(true).raw());
assert_eq!(rates, vec![0b11111000]);
});
}
#[test]
fn authenticated_goes_to_associated_no_rsn_differing_cap() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States =
State::new(Authenticating).transition_to(Authenticated { timeout_event_id: 1 }).into();
let mut aid_map = aid::Map::default();
let state = state.handle_assoc_ind(
&mut r_sta,
&mut ctx,
&mut aid_map,
CapabilityInfo(0).with_short_preamble(true).with_spectrum_mgmt(true),
CapabilityInfo(0)
.with_short_preamble(true)
.with_spectrum_mgmt(true)
.with_radio_measurement(true)
.raw(),
&[SupportedRate(0b11111000)][..],
&[SupportedRate(0b11111000)][..],
&None,
None,
);
let (_, Associated { rsna_link_state, aid }) = match state {
States::Associated(state) => state.release_data(),
_ => panic!("unexpected_state"),
};
assert_variant!(rsna_link_state, None);
assert_eq!(aid, 1);
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AssocResponse(fidl_mlme::AssociateResponse {
peer_sta_address,
result_code,
capability_info,
rates,
..
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(result_code, fidl_mlme::AssociateResultCode::Success);
assert_eq!(
capability_info,
CapabilityInfo(0).with_short_preamble(true).with_spectrum_mgmt(true).raw(),
);
assert_eq!(rates, vec![0b11111000]);
});
}
#[test]
fn authenticated_goes_to_associated_differing_nonbasic_rates() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States =
State::new(Authenticating).transition_to(Authenticated { timeout_event_id: 1 }).into();
let mut aid_map = aid::Map::default();
let _next_state = state.handle_assoc_ind(
&mut r_sta,
&mut ctx,
&mut aid_map,
CapabilityInfo(0).with_short_preamble(true),
CapabilityInfo(0).with_short_preamble(true).raw(),
&[SupportedRate(0b11111000), SupportedRate(0b01111001)][..],
&[SupportedRate(0b11111000), SupportedRate(0b01111010)][..],
&None,
None,
);
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AssocResponse(fidl_mlme::AssociateResponse {
capability_info,
rates,
..
}) => {
assert_eq!(capability_info, CapabilityInfo(0).with_short_preamble(true).raw());
assert_eq!(rates, vec![0b11111000]);
});
}
#[test]
fn authenticated_goes_to_associated_fullmac() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
ctx.mac_sublayer_support = fake_mac_sublayer_support();
ctx.mac_sublayer_support.device.mac_implementation_type =
fidl_common::MacImplementationType::Fullmac;
let state: States =
State::new(Authenticating).transition_to(Authenticated { timeout_event_id: 1 }).into();
let mut aid_map = aid::Map::default();
let _next_state = state.handle_assoc_ind(
&mut r_sta,
&mut ctx,
&mut aid_map,
CapabilityInfo(0).with_short_preamble(true),
CapabilityInfo(0).with_short_preamble(true).raw(),
&[][..],
&[SupportedRate(0b11111000), SupportedRate(0b01111010)][..],
&None,
None,
);
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AssocResponse(fidl_mlme::AssociateResponse {
capability_info,
rates,
..
}) => {
assert_eq!(capability_info, CapabilityInfo(0).with_short_preamble(true).raw());
assert_eq!(rates, vec![0b11111000, 0b01111010]);
});
}
#[test]
fn authenticated_goes_to_associated_differing_basic_rates() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States =
State::new(Authenticating).transition_to(Authenticated { timeout_event_id: 1 }).into();
let mut aid_map = aid::Map::default();
let _next_state = state.handle_assoc_ind(
&mut r_sta,
&mut ctx,
&mut aid_map,
CapabilityInfo(0).with_short_preamble(true),
CapabilityInfo(0).with_short_preamble(true).raw(),
&[SupportedRate(0b11111001)][..],
&[SupportedRate(0b11111000)][..],
&None,
None,
);
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AssocResponse(fidl_mlme::AssociateResponse {
peer_sta_address,
result_code,
..
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(result_code, fidl_mlme::AssociateResultCode::RefusedBasicRatesMismatch);
});
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::Deauthenticate(fidl_mlme::DeauthenticateRequest {
peer_sta_address,
reason_code,
..
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(reason_code, fidl_ieee80211::ReasonCode::ReasonInvalidElement);
});
}
#[test]
fn authenticated_goes_to_associated_no_ap_rates() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States =
State::new(Authenticating).transition_to(Authenticated { timeout_event_id: 1 }).into();
let mut aid_map = aid::Map::default();
let _next_state = state.handle_assoc_ind(
&mut r_sta,
&mut ctx,
&mut aid_map,
CapabilityInfo(0).with_short_preamble(true),
CapabilityInfo(0).with_short_preamble(true).raw(),
&[SupportedRate(0b01111000)][..],
&[][..],
&None,
None,
);
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AssocResponse(fidl_mlme::AssociateResponse {
peer_sta_address,
result_code,
..
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(result_code, fidl_mlme::AssociateResultCode::RefusedCapabilitiesMismatch);
});
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::Deauthenticate(fidl_mlme::DeauthenticateRequest {
peer_sta_address,
reason_code,
..
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(reason_code, fidl_ieee80211::ReasonCode::ReasonInvalidElement);
});
}
#[test]
fn authenticated_goes_to_authenticating_out_of_aids() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States =
State::new(Authenticating).transition_to(Authenticated { timeout_event_id: 1 }).into();
let mut aid_map = aid::Map::default();
while aid_map.assign_aid().is_ok() {
}
let state = state.handle_assoc_ind(
&mut r_sta,
&mut ctx,
&mut aid_map,
CapabilityInfo(0).with_short_preamble(true),
CapabilityInfo(0).with_short_preamble(true).raw(),
&[SupportedRate(0b11111000)][..],
&[SupportedRate(0b11111000)][..],
&None,
None,
);
let (_, Authenticating) = match state {
States::Authenticating(state) => state.release_data(),
_ => panic!("unexpected state"),
};
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AssocResponse(fidl_mlme::AssociateResponse {
peer_sta_address,
result_code,
..
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(result_code, fidl_mlme::AssociateResultCode::RefusedReasonUnspecified);
});
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::Deauthenticate(fidl_mlme::DeauthenticateRequest {
peer_sta_address,
reason_code,
..
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(reason_code, fidl_ieee80211::ReasonCode::UnspecifiedReason);
});
}
#[test]
fn authenticated_goes_to_authenticating_with_bogus_rsn_ind() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States =
State::new(Authenticating).transition_to(Authenticated { timeout_event_id: 1 }).into();
let s_rsne = Rsne::wpa2_rsne();
let mut s_rsne_vec = Vec::with_capacity(s_rsne.len());
s_rsne.write_into(&mut s_rsne_vec).expect("error writing RSNE");
let mut aid_map = aid::Map::default();
let state = state.handle_assoc_ind(
&mut r_sta,
&mut ctx,
&mut aid_map,
CapabilityInfo(0).with_short_preamble(true),
CapabilityInfo(0).with_short_preamble(true).raw(),
&[SupportedRate(0b11111000)][..],
&[SupportedRate(0b11111000)][..],
&None,
Some(s_rsne_vec),
);
let (_, Authenticating) = match state {
States::Authenticating(state) => state.release_data(),
_ => panic!("unexpected state"),
};
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AssocResponse(fidl_mlme::AssociateResponse {
peer_sta_address,
result_code,
..
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(result_code, fidl_mlme::AssociateResultCode::RefusedCapabilitiesMismatch);
});
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::Deauthenticate(fidl_mlme::DeauthenticateRequest {
peer_sta_address,
reason_code,
..
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(reason_code, fidl_ieee80211::ReasonCode::ReasonInvalidElement);
});
}
#[test]
fn authenticated_goes_to_authenticating_with_incompatible_rsn() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States =
State::new(Authenticating).transition_to(Authenticated { timeout_event_id: 1 }).into();
let mut rsn_cfg =
create_rsn_cfg(&Ssid::try_from("coolnet").unwrap(), b"password").unwrap().unwrap();
rsn_cfg.rsne = Rsne {
group_data_cipher_suite: Some(CIPHER_GCMP_256),
pairwise_cipher_suites: vec![CIPHER_CCMP_128],
akm_suites: vec![AKM_PSK],
..Default::default()
};
let s_rsne = Rsne {
group_data_cipher_suite: Some(CIPHER_CCMP_128),
pairwise_cipher_suites: vec![CIPHER_CCMP_128],
akm_suites: vec![AKM_PSK],
..Default::default()
};
let mut s_rsne_vec = Vec::with_capacity(s_rsne.len());
s_rsne.write_into(&mut s_rsne_vec).expect("error writing RSNE");
let mut aid_map = aid::Map::default();
let state = state.handle_assoc_ind(
&mut r_sta,
&mut ctx,
&mut aid_map,
CapabilityInfo(0).with_short_preamble(true),
CapabilityInfo(0).with_short_preamble(true).raw(),
&[SupportedRate(0b11111000)][..],
&[SupportedRate(0b11111000)][..],
&Some(rsn_cfg),
Some(s_rsne_vec),
);
let (_, Authenticating) = match state {
States::Authenticating(state) => state.release_data(),
_ => panic!("unexpected state"),
};
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AssocResponse(fidl_mlme::AssociateResponse {
peer_sta_address,
result_code,
..
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(result_code, fidl_mlme::AssociateResultCode::RefusedCapabilitiesMismatch);
});
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::Deauthenticate(fidl_mlme::DeauthenticateRequest {
peer_sta_address,
reason_code,
..
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(reason_code, fidl_ieee80211::ReasonCode::Ieee8021XAuthFailed);
});
}
#[test]
fn authenticated_goes_to_associated_rsn() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States =
State::new(Authenticating).transition_to(Authenticated { timeout_event_id: 1 }).into();
let rsn_cfg =
create_rsn_cfg(&Ssid::try_from("coolnet").unwrap(), b"password").unwrap().unwrap();
let mut s_rsne_vec = Vec::with_capacity(rsn_cfg.rsne.len());
rsn_cfg.rsne.write_into(&mut s_rsne_vec).expect("error writing RSNE");
let mut aid_map = aid::Map::default();
let state = state.handle_assoc_ind(
&mut r_sta,
&mut ctx,
&mut aid_map,
CapabilityInfo(0).with_short_preamble(true),
CapabilityInfo(0).with_short_preamble(true).raw(),
&[SupportedRate(0b11111000)][..],
&[SupportedRate(0b11111000)][..],
&Some(rsn_cfg),
Some(s_rsne_vec),
);
let (_, Associated { rsna_link_state, aid }) = match state {
States::Associated(state) => state.release_data(),
_ => panic!("unexpected_state"),
};
assert_eq!(aid, 1);
assert_variant!(rsna_link_state, Some(_));
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AssocResponse(fidl_mlme::AssociateResponse {
peer_sta_address,
result_code,
capability_info,
rates,
..
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(result_code, fidl_mlme::AssociateResultCode::Success);
assert_eq!(capability_info, CapabilityInfo(0).with_short_preamble(true).with_privacy(true).raw());
assert_eq!(rates, vec![0b11111000]);
});
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::Eapol(fidl_mlme::EapolRequest { .. }));
}
#[test]
fn authenticated_goes_to_associated_rsn_different_cap() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States =
State::new(Authenticating).transition_to(Authenticated { timeout_event_id: 1 }).into();
let rsn_cfg =
create_rsn_cfg(&Ssid::try_from("coolnet").unwrap(), b"password").unwrap().unwrap();
let mut s_rsne_vec = Vec::with_capacity(rsn_cfg.rsne.len());
rsn_cfg.rsne.write_into(&mut s_rsne_vec).expect("error writing RSNE");
let mut aid_map = aid::Map::default();
let state = state.handle_assoc_ind(
&mut r_sta,
&mut ctx,
&mut aid_map,
CapabilityInfo(0)
.with_short_preamble(true)
.with_spectrum_mgmt(true)
.with_radio_measurement(true),
CapabilityInfo(0).with_short_preamble(true).with_spectrum_mgmt(true).raw(),
&[SupportedRate(0b11111000)][..],
&[SupportedRate(0b11111000)][..],
&Some(rsn_cfg),
Some(s_rsne_vec),
);
let (_, Associated { rsna_link_state, aid }) = match state {
States::Associated(state) => state.release_data(),
_ => panic!("unexpected_state"),
};
assert_eq!(aid, 1);
assert_variant!(rsna_link_state, Some(_));
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::AssocResponse(fidl_mlme::AssociateResponse {
peer_sta_address,
result_code,
capability_info,
rates,
..
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(result_code, fidl_mlme::AssociateResultCode::Success);
assert_eq!(
capability_info,
CapabilityInfo(0)
.with_short_preamble(true)
.with_spectrum_mgmt(true)
.with_privacy(true)
.raw());
assert_eq!(rates, vec![0b11111000]);
});
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::Eapol(fidl_mlme::EapolRequest { .. }));
}
#[test]
fn associated_goes_to_authenticated() {
let mut r_sta = make_remote_client();
let (mut ctx, _, mut time_stream) = make_env();
let mut aid_map = aid::Map::default();
let aid = aid_map.assign_aid().unwrap();
let state: States = State::new(Authenticating)
.transition_to(Authenticated { timeout_event_id: 1 })
.transition_to(Associated { aid, rsna_link_state: None })
.into();
let state = state.handle_disassoc_ind(&mut r_sta, &mut ctx, &mut aid_map);
let (_, Authenticated { timeout_event_id }) = match state {
States::Authenticated(state) => state.release_data(),
_ => panic!("unexpected state"),
};
assert_eq!(aid, aid_map.assign_aid().unwrap());
let (_, timed_event) = time_stream.try_next().unwrap().expect("expected timed event");
assert_eq!(timed_event.id, timeout_event_id);
assert_variant!(timed_event.event, Event::Client { addr, event } => {
assert_eq!(addr, *CLIENT_ADDR);
assert_variant!(event, ClientEvent::AssociationTimeout);
});
}
#[test]
fn associated_ignores_rsna_negotiation_timeout_without_rsna_link_state() {
let mut r_sta = make_remote_client();
let (mut ctx, _, mut time_stream) = make_env();
let state: States = State::new(Authenticating)
.transition_to(Authenticated { timeout_event_id: 1 })
.transition_to(Associated { aid: 1, rsna_link_state: None })
.into();
let state = state.handle_timeout(
&mut r_sta,
&mut ctx,
1,
ClientEvent::RsnaTimeout(RsnaTimeout::Negotiation),
);
let (_, Associated { .. }) = match state {
States::Associated(state) => state.release_data(),
_ => panic!("unexpected_state"),
};
assert_variant!(time_stream.try_next(), Err(_));
}
#[test]
fn associated_ignores_rsna_request_timeout_without_rsna_link_state() {
let mut r_sta = make_remote_client();
let (mut ctx, _, mut time_stream) = make_env();
let state: States = State::new(Authenticating)
.transition_to(Authenticated { timeout_event_id: 1 })
.transition_to(Associated { aid: 1, rsna_link_state: None })
.into();
let state = state.handle_timeout(
&mut r_sta,
&mut ctx,
1,
ClientEvent::RsnaTimeout(RsnaTimeout::Request),
);
let (_, Associated { .. }) = match state {
States::Associated(state) => state.release_data(),
_ => panic!("unexpected_state"),
};
assert_variant!(time_stream.try_next(), Err(_));
}
#[test]
fn associated_handles_rsna_request_timeout() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, mut time_stream) = make_env();
let rsn_cfg =
create_rsn_cfg(&Ssid::try_from("coolnet").unwrap(), b"password").unwrap().unwrap();
let s_rsne = Rsne::wpa2_rsne();
let mut s_rsne_vec = Vec::with_capacity(s_rsne.len());
s_rsne.write_into(&mut s_rsne_vec).expect("error writing RSNE");
let state: States = State::new(Authenticating)
.transition_to(Authenticated { timeout_event_id: 1 })
.transition_to(Associated {
aid: 1,
rsna_link_state: Some(RsnaLinkState {
request_attempts: 0,
last_key_frame: Some(test_utils::eapol_key_frame()),
request_timeout_event_id: Some(1),
negotiation_timeout_event_id: Some(2),
authenticator: new_authenticator_from_rsne(
*AP_ADDR,
*CLIENT_ADDR,
&s_rsne_vec[..],
&rsn_cfg,
)
.unwrap(),
}),
})
.into();
let state = state.handle_timeout(
&mut r_sta,
&mut ctx,
1,
ClientEvent::RsnaTimeout(RsnaTimeout::Request),
);
let (_, Associated { rsna_link_state, .. }) = match state {
States::Associated(state) => state.release_data(),
_ => panic!("unexpected_state"),
};
assert_eq!(rsna_link_state.as_ref().unwrap().request_attempts, 1);
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::Eapol(fidl_mlme::EapolRequest { .. }));
let (_, timed_event) = time_stream.try_next().unwrap().expect("expected timed event");
assert_eq!(
timed_event.id,
rsna_link_state.as_ref().unwrap().request_timeout_event_id.unwrap()
);
assert_variant!(timed_event.event, Event::Client { addr, event } => {
assert_eq!(addr, *CLIENT_ADDR);
assert_variant!(event, ClientEvent::RsnaTimeout(RsnaTimeout::Request));
});
}
#[test]
fn associated_handles_rsna_negotiation_timeout() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let rsn_cfg =
create_rsn_cfg(&Ssid::try_from("coolnet").unwrap(), b"password").unwrap().unwrap();
let s_rsne = Rsne::wpa2_rsne();
let mut s_rsne_vec = Vec::with_capacity(s_rsne.len());
s_rsne.write_into(&mut s_rsne_vec).expect("error writing RSNE");
let state: States = State::new(Authenticating)
.transition_to(Authenticated { timeout_event_id: 1 })
.transition_to(Associated {
aid: 1,
rsna_link_state: Some(RsnaLinkState {
request_attempts: 3,
last_key_frame: Some(test_utils::eapol_key_frame()),
request_timeout_event_id: Some(1),
negotiation_timeout_event_id: Some(2),
authenticator: new_authenticator_from_rsne(
*AP_ADDR,
*CLIENT_ADDR,
&s_rsne_vec[..],
&rsn_cfg,
)
.unwrap(),
}),
})
.into();
let state = state.handle_timeout(
&mut r_sta,
&mut ctx,
2,
ClientEvent::RsnaTimeout(RsnaTimeout::Negotiation),
);
let (_, Authenticating) = match state {
States::Authenticating(state) => state.release_data(),
_ => panic!("unexpected_state"),
};
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::Deauthenticate(fidl_mlme::DeauthenticateRequest {
peer_sta_address,
reason_code,
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(reason_code, fidl_ieee80211::ReasonCode::FourwayHandshakeTimeout);
});
}
#[test]
fn associated_handles_rsna_key_frame_resets_request_attempts() {
let mut r_sta = make_remote_client();
let (mut ctx, _, _) = make_env();
let rsn_cfg =
create_rsn_cfg(&Ssid::try_from("coolnet").unwrap(), b"password").unwrap().unwrap();
let s_rsne = Rsne::wpa2_rsne();
let mut s_rsne_vec = Vec::with_capacity(s_rsne.len());
s_rsne.write_into(&mut s_rsne_vec).expect("error writing RSNE");
let state: States = State::new(Authenticating)
.transition_to(Authenticated { timeout_event_id: 1 })
.transition_to(Associated {
aid: 1,
rsna_link_state: Some(RsnaLinkState {
request_attempts: 3,
last_key_frame: Some(test_utils::eapol_key_frame()),
request_timeout_event_id: Some(1),
negotiation_timeout_event_id: Some(2),
authenticator: new_authenticator_from_rsne(
*AP_ADDR,
*CLIENT_ADDR,
&s_rsne_vec[..],
&rsn_cfg,
)
.unwrap(),
}),
})
.into();
let state = state.handle_eapol_ind(
&mut r_sta,
&mut ctx,
&Vec::<u8>::from(test_utils::eapol_key_frame())[..],
);
let (_, Associated { rsna_link_state, .. }) = match state {
States::Associated(state) => state.release_data(),
_ => panic!("unexpected_state"),
};
assert_eq!(rsna_link_state.as_ref().unwrap().request_attempts, 0);
}
#[test]
fn associated_handles_rsna_request_timeout_last_attempt() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let rsn_cfg =
create_rsn_cfg(&Ssid::try_from("coolnet").unwrap(), b"password").unwrap().unwrap();
let s_rsne = Rsne::wpa2_rsne();
let mut s_rsne_vec = Vec::with_capacity(s_rsne.len());
s_rsne.write_into(&mut s_rsne_vec).expect("error writing RSNE");
let state: States = State::new(Authenticating)
.transition_to(Authenticated { timeout_event_id: 1 })
.transition_to(Associated {
aid: 1,
rsna_link_state: Some(RsnaLinkState {
request_attempts: 3,
last_key_frame: Some(test_utils::eapol_key_frame()),
request_timeout_event_id: Some(1),
negotiation_timeout_event_id: Some(2),
authenticator: new_authenticator_from_rsne(
*AP_ADDR,
*CLIENT_ADDR,
&s_rsne_vec[..],
&rsn_cfg,
)
.unwrap(),
}),
})
.into();
let state = state.handle_timeout(
&mut r_sta,
&mut ctx,
1,
ClientEvent::RsnaTimeout(RsnaTimeout::Request),
);
let (_, Authenticating) = match state {
States::Authenticating(state) => state.release_data(),
_ => panic!("unexpected state"),
};
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::Deauthenticate(fidl_mlme::DeauthenticateRequest {
peer_sta_address,
reason_code,
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(reason_code, fidl_ieee80211::ReasonCode::FourwayHandshakeTimeout);
});
}
#[test]
fn associated_handles_eapol_key_frame() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States = State::new(Authenticating)
.transition_to(Authenticated { timeout_event_id: 1 })
.transition_to(Associated {
aid: 1,
rsna_link_state: Some(RsnaLinkState {
request_attempts: 0,
last_key_frame: Some(test_utils::eapol_key_frame()),
request_timeout_event_id: Some(1),
negotiation_timeout_event_id: Some(2),
authenticator: Box::new(MockAuthenticator::new(
Arc::new(Mutex::new(vec![])),
Arc::new(Mutex::new(vec![SecAssocUpdate::TxEapolKeyFrame {
frame: test_utils::eapol_key_frame(),
expect_response: false,
}])),
)),
}),
})
.into();
let _next_state = state.handle_eapol_ind(
&mut r_sta,
&mut ctx,
&Vec::<u8>::from(test_utils::eapol_key_frame())[..],
);
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::Eapol(fidl_mlme::EapolRequest {
src_addr,
dst_addr,
data,
}) => {
assert_eq!(&src_addr, AP_ADDR.as_array());
assert_eq!(&dst_addr, CLIENT_ADDR.as_array());
assert_eq!(data, Vec::<u8>::from(test_utils::eapol_key_frame()));
});
}
#[test]
fn associated_handles_eapol_conf() {
let mut r_sta = make_remote_client();
let (mut ctx, _mlme_stream, _) = make_env();
let state: States = State::new(Authenticating)
.transition_to(Authenticated { timeout_event_id: 1 })
.transition_to(Associated {
aid: 1,
rsna_link_state: Some(RsnaLinkState {
request_attempts: 0,
last_key_frame: Some(test_utils::eapol_key_frame()),
request_timeout_event_id: Some(1),
negotiation_timeout_event_id: Some(2),
authenticator: Box::new(MockAuthenticator::new(
Arc::new(Mutex::new(vec![])),
Arc::new(Mutex::new(vec![SecAssocUpdate::TxEapolKeyFrame {
frame: test_utils::eapol_key_frame(),
expect_response: false,
}])),
)),
}),
})
.into();
let state =
state.handle_eapol_conf(&mut r_sta, &mut ctx, fidl_mlme::EapolResultCode::Success);
match state {
States::Associated(_) => (),
_ => panic!("Eapol conf should leave us in Associated"),
}
}
#[test]
fn associated_handles_eapol_key() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States = State::new(Authenticating)
.transition_to(Authenticated { timeout_event_id: 1 })
.transition_to(Associated {
aid: 1,
rsna_link_state: Some(RsnaLinkState {
request_attempts: 0,
last_key_frame: Some(test_utils::eapol_key_frame()),
request_timeout_event_id: Some(1),
negotiation_timeout_event_id: Some(2),
authenticator: Box::new(MockAuthenticator::new(
Arc::new(Mutex::new(vec![])),
Arc::new(Mutex::new(vec![SecAssocUpdate::Key(
Key::Ptk(test_utils::ptk()),
)])),
)),
}),
})
.into();
let _next_state = state.handle_eapol_ind(
&mut r_sta,
&mut ctx,
&Vec::<u8>::from(test_utils::eapol_key_frame())[..],
);
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::SetKeys(fidl_mlme::SetKeysRequest { keylist }) => {
assert_eq!(keylist.len(), 1);
let k = keylist.first().expect("expect key descriptor");
assert_eq!(k.key, vec![0xCCu8; test_utils::cipher().tk_bytes().unwrap() as usize]);
assert_eq!(k.key_id, 0);
assert_eq!(k.key_type, fidl_mlme::KeyType::Pairwise);
assert_eq!(&k.address, CLIENT_ADDR.as_array());
assert_eq!(k.rsc, 0);
assert_eq!(k.cipher_suite_oui, [0x00, 0x0F, 0xAC]);
assert_eq!(k.cipher_suite_type, fidl_ieee80211::CipherSuiteType::from_primitive_allow_unknown(4));
});
}
#[test]
fn associated_handles_esssa_established() {
let mut r_sta = make_remote_client();
let (mut ctx, mut mlme_stream, _) = make_env();
let state: States = State::new(Authenticating)
.transition_to(Authenticated { timeout_event_id: 1 })
.transition_to(Associated {
aid: 1,
rsna_link_state: Some(RsnaLinkState {
request_attempts: 0,
last_key_frame: Some(test_utils::eapol_key_frame()),
request_timeout_event_id: Some(1),
negotiation_timeout_event_id: Some(2),
authenticator: Box::new(MockAuthenticator::new(
Arc::new(Mutex::new(vec![])),
Arc::new(Mutex::new(vec![SecAssocUpdate::Status(
SecAssocStatus::EssSaEstablished,
)])),
)),
}),
})
.into();
let state = state.handle_eapol_ind(
&mut r_sta,
&mut ctx,
&Vec::<u8>::from(test_utils::eapol_key_frame())[..],
);
let (_, Associated { rsna_link_state, .. }) = match state {
States::Associated(state) => state.release_data(),
_ => panic!("unexpected_state"),
};
assert_variant!(&rsna_link_state.as_ref().unwrap().last_key_frame, None);
assert_variant!(&rsna_link_state.as_ref().unwrap().request_timeout_event_id, None);
assert_variant!(&rsna_link_state.as_ref().unwrap().negotiation_timeout_event_id, None);
let mlme_event = mlme_stream.try_next().unwrap().expect("expected mlme event");
assert_variant!(mlme_event, MlmeRequest::SetCtrlPort(fidl_mlme::SetControlledPortRequest {
peer_sta_address,
state,
}) => {
assert_eq!(&peer_sta_address, CLIENT_ADDR.as_array());
assert_eq!(state, fidl_mlme::ControlledPortState::Open);
});
}
}