| // Copyright (c) 2013 The Chromium OS Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| // This file provides tests to verify that Callback80211Metrics sends UMA |
| // notifications for appropriate messages and doesn't send them for |
| // inappropriate messages. |
| |
| #include "shill/callback80211_metrics.h" |
| |
| #include <gmock/gmock.h> |
| #include <gtest/gtest.h> |
| |
| #include "shill/ieee80211.h" |
| #include "shill/mock_event_dispatcher.h" |
| #include "shill/mock_log.h" |
| #include "shill/mock_metrics.h" |
| #include "shill/nl80211_message.h" |
| #include "shill/refptr_types.h" |
| |
| using base::Bind; |
| using testing::_; |
| using testing::Test; |
| |
| namespace shill { |
| |
| namespace { |
| |
| // Unless otherwise specified, these data blocks have been collected by shill |
| // using NetlinkManager while, simultaneously (and manually) comparing shill |
| // output with that of the 'iw' code from which it was derived. The test |
| // strings represent the raw packet data coming from the kernel. The |
| // comments above each of these strings is the markup that 'iw' outputs for |
| // each of these packets. |
| |
| // These constants are consistent across the applicable packets, below. |
| |
| const uint16_t kNl80211FamilyId = 0x13; |
| const IEEE_80211::WiFiReasonCode kExpectedDisconnectReason = |
| IEEE_80211::kReasonCodePreviousAuthenticationInvalid; |
| |
| // NL80211_CMD_DISCONNECT message. |
| // wlan0 (phy #0): disconnected (by AP) reason: 2: Previous authentication no |
| // longer valid |
| |
| const unsigned char kDisconnectMessage[] = { |
| 0x30, 0x00, 0x00, 0x00, 0x13, 0x00, 0x00, 0x00, |
| 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, |
| 0x30, 0x01, 0x00, 0x00, 0x08, 0x00, 0x01, 0x00, |
| 0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x03, 0x00, |
| 0x04, 0x00, 0x00, 0x00, 0x06, 0x00, 0x36, 0x00, |
| 0x02, 0x00, 0x00, 0x00, 0x04, 0x00, 0x47, 0x00, |
| }; |
| |
| // NL80211_CMD_DISCONNECT message. |
| // Copied from kDisconnectMessage but with most of the payload removed. |
| |
| const unsigned char kEmptyDisconnectMessage[] = { |
| 0x1c, 0x00, 0x00, 0x00, 0x13, 0x00, 0x00, 0x00, |
| 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, |
| 0x30, 0x01, 0x00, 0x00, 0x08, 0x00, 0x01, 0x00, |
| 0x00, 0x00, 0x00, 0x00, |
| }; |
| |
| // NL80211_CMD_DEAUTHENTICATE message. |
| // wlan0 (phy #0): deauth c0:3f:0e:77:e8:7f -> ff:ff:ff:ff:ff:ff reason 2: |
| // Previous authentication no longer valid [frame: c0 00 00 00 ff ff ff ff |
| // ff ff c0 3f 0e 77 e8 7f c0 3f 0e 77 e8 7f c0 0e 02 00] |
| |
| const unsigned char kDeauthenticateMessage[] = { |
| 0x44, 0x00, 0x00, 0x00, 0x13, 0x00, 0x00, 0x00, |
| 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, |
| 0x27, 0x01, 0x00, 0x00, 0x08, 0x00, 0x01, 0x00, |
| 0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x03, 0x00, |
| 0x04, 0x00, 0x00, 0x00, 0x1e, 0x00, 0x33, 0x00, |
| 0xc0, 0x00, 0x00, 0x00, 0xff, 0xff, 0xff, 0xff, |
| 0xff, 0xff, 0xc0, 0x3f, 0x0e, 0x77, 0xe8, 0x7f, |
| 0xc0, 0x3f, 0x0e, 0x77, 0xe8, 0x7f, 0xc0, 0x0e, |
| 0x02, 0x00, 0x00, 0x00, |
| }; |
| |
| // NL80211_CMD_DEAUTHENTICATE message. |
| // Copied from kDeauthenticateMessage but with most of the payload |
| // removed. |
| |
| const unsigned char kEmptyDeauthenticateMessage[] = { |
| 0x1c, 0x00, 0x00, 0x00, 0x13, 0x00, 0x00, 0x00, |
| 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, |
| 0x27, 0x01, 0x00, 0x00, 0x08, 0x00, 0x01, 0x00, |
| 0x00, 0x00, 0x00, 0x00, |
| }; |
| |
| // NL80211_CMD_NEW_STATION message. |
| // kNewStationMessage is an nl80211 message that's not a deauthenticate or |
| // disconnect message. Used to make sure that only those nl80211 messages |
| // generate an UMA message. |
| // wlan0: new station c0:3f:0e:77:e8:7f |
| |
| const unsigned char kNewStationMessage[] = { |
| 0x34, 0x00, 0x00, 0x00, 0x13, 0x00, 0x00, 0x00, |
| 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, |
| 0x13, 0x01, 0x00, 0x00, 0x08, 0x00, 0x03, 0x00, |
| 0x04, 0x00, 0x00, 0x00, 0x0a, 0x00, 0x06, 0x00, |
| 0xc0, 0x3f, 0x0e, 0x77, 0xe8, 0x7f, 0x00, 0x00, |
| 0x08, 0x00, 0x2e, 0x00, 0x13, 0x01, 0x00, 0x00, |
| 0x04, 0x00, 0x15, 0x00, |
| }; |
| |
| // CTRL_CMD_GETFAMILY message. |
| // kGetFamilyMessage is not an nl80211 message. Used to make sure that |
| // non-nl80211 messages don't generate an UMA message. |
| // |
| // Extracted from net.log. It's just a non-nl80211 message (it's actually a |
| // message that's sent to the kernel rather than one received from the kernel |
| // but the code doesn't differentiate and this message was much shorter than the |
| // response). |
| |
| const unsigned char kGetFamilyMessage[] = { |
| 0x10, 0x00, 0x00, 0x00, 0x0c, 0x00, 0x02, 0x00, |
| 0x6e, 0x6c, 0x38, 0x30, 0x32, 0x31, 0x31, 0x00 |
| }; |
| |
| } // namespace |
| |
| class Callback80211MetricsTest : public Test { |
| public: |
| Callback80211MetricsTest() : |
| metrics_(&dispatcher_), callback_(&metrics_) { |
| message_factory_.AddFactoryMethod( |
| kNl80211FamilyId, Bind(&Nl80211Message::CreateMessage)); |
| Nl80211Message::SetMessageType(kNl80211FamilyId); |
| } |
| |
| protected: |
| MockEventDispatcher dispatcher_; |
| MockMetrics metrics_; |
| NetlinkMessageFactory message_factory_; |
| Callback80211Metrics callback_; |
| }; |
| |
| // Make sure that notifications happen for correctly formed messages. |
| TEST_F(Callback80211MetricsTest, DisconnectMessage) { |
| scoped_ptr<NetlinkMessage> netlink_message( |
| message_factory_.CreateMessage(const_cast<nlmsghdr *>( |
| reinterpret_cast<const nlmsghdr *>(kDisconnectMessage)))); |
| EXPECT_CALL(metrics_, Notify80211Disconnect(Metrics::kDisconnectedByAp, |
| kExpectedDisconnectReason)); |
| callback_.CollectDisconnectStatistics(*netlink_message); |
| } |
| |
| TEST_F(Callback80211MetricsTest, DeauthMessage) { |
| scoped_ptr<NetlinkMessage> netlink_message( |
| message_factory_.CreateMessage(const_cast<nlmsghdr *>( |
| reinterpret_cast<const nlmsghdr *>(kDeauthenticateMessage)))); |
| EXPECT_CALL(metrics_, Notify80211Disconnect(Metrics::kDisconnectedNotByAp, |
| kExpectedDisconnectReason)); |
| callback_.CollectDisconnectStatistics(*netlink_message); |
| } |
| |
| // Make sure there's no notification if there's no reason code in the message. |
| TEST_F(Callback80211MetricsTest, EmptyDisconnectMessage) { |
| scoped_ptr<NetlinkMessage> netlink_message( |
| message_factory_.CreateMessage(const_cast<nlmsghdr *>( |
| reinterpret_cast<const nlmsghdr *>( |
| kEmptyDisconnectMessage)))); |
| EXPECT_CALL(metrics_, Notify80211Disconnect(_, _)).Times(0); |
| callback_.CollectDisconnectStatistics(*netlink_message); |
| } |
| |
| TEST_F(Callback80211MetricsTest, EmptyDeauthMessage) { |
| scoped_ptr<NetlinkMessage> netlink_message( |
| message_factory_.CreateMessage(const_cast<nlmsghdr *>( |
| reinterpret_cast<const nlmsghdr *>( |
| kEmptyDeauthenticateMessage)))); |
| EXPECT_CALL(metrics_, Notify80211Disconnect(_, _)).Times(0); |
| callback_.CollectDisconnectStatistics(*netlink_message); |
| } |
| |
| // Make sure the callback doesn't notify anyone for message of the wrong type. |
| TEST_F(Callback80211MetricsTest, Nl80211NotDisconnectDeauthMessage) { |
| scoped_ptr<NetlinkMessage> netlink_message( |
| message_factory_.CreateMessage(const_cast<nlmsghdr *>( |
| reinterpret_cast<const nlmsghdr *>( |
| kNewStationMessage)))); |
| EXPECT_CALL(metrics_, Notify80211Disconnect(_, _)).Times(0); |
| callback_.CollectDisconnectStatistics(*netlink_message); |
| } |
| |
| TEST_F(Callback80211MetricsTest, NotNl80211Message) { |
| scoped_ptr<NetlinkMessage> netlink_message( |
| message_factory_.CreateMessage(const_cast<nlmsghdr *>( |
| reinterpret_cast<const nlmsghdr *>( |
| kGetFamilyMessage)))); |
| EXPECT_CALL(metrics_, Notify80211Disconnect(_, _)).Times(0); |
| callback_.CollectDisconnectStatistics(*netlink_message); |
| } |
| |
| } // namespace shill |