1# Copyright 2015 The Chromium OS Authors. All rights reserved.
2# Use of this source code is governed by a BSD-style license that can be
3# found in the LICENSE file.
4
5import logging, json
6
7from autotest_lib.client.bin import test, utils
8from autotest_lib.client.common_lib import error
9from autotest_lib.client.cros.networking.chrome_testing \
10        import chrome_networking_test_context as cntc
11from autotest_lib.client.cros.networking.chrome_testing \
12        import chrome_networking_test_api as cnta
13from autotest_lib.client.cros.networking.chrome_testing import test_utils
14from autotest_lib.client.cros.power import sys_power
15from collections import namedtuple
16
17NetworkInfo = namedtuple('NetworkInfo', ['name', 'guid', 'connectionState',
18                                         'networkType'])
19
20class network_ChromeCellularEndToEnd(test.test):
21    """
22    Tests the following UI functionality with chrome.networkingPrivate APIs:
23        1. Tests that the device auto connects to cellular network.
24        3. Tests that the device prefers ethernet over cellular network.
25        4. Tests that the enabling and disabling of cellular modem works.
26
27    """
28    version = 1
29
30
31    def _extract_network_info(self, networks_found):
32        """Extract the needed information from the list of networks.
33
34        @param networks_found: Networks found via api.
35        @return Formatted list of available cellular networks.
36
37        """
38        formatted_network_list = []
39
40        for network in networks_found:
41          network = NetworkInfo(name=network['Name'],
42                  guid=network['GUID'],
43                  connectionState=network.get('ConnectionState', 'none'),
44                  networkType=network['Type'])
45          formatted_network_list.append(network)
46
47        return formatted_network_list
48
49
50    def _set_autoconnect(self, service, value=True):
51        """Turn on autoconnect for cellular network.
52
53        @param service: Cellular service dictionary
54        @value: Set / unset autoconnect
55
56        """
57        logging.debug('_set_autoconnect')
58        properties = json.dumps({'Cellular': {'AutoConnect': value}})
59        set_properties = test_utils.call_test_function_check_success(
60                self._chrome_testing,
61                'setProperties',
62                ('"' + service['GUID'] + '"', properties))
63        self.chrome_net.scan_for_networks()
64
65
66    def _is_cellular_network_connected(self, service):
67        """Check if device is connected to cellular network.
68
69        @param service: Cellular service dict
70        @return True if connected to cellular, else False
71
72        """
73        network_properties = self.chrome_net._chrome_testing.call_test_function(
74                test_utils.LONG_TIMEOUT,
75                'getNetworkInfo',
76                ('"' + service['GUID'] + '"'))
77
78        logging.debug('Network properties: %s', network_properties)
79
80        if network_properties['status'] == 'chrome-test-call-status-failure':
81            raise error.TestFail('getNetworkInfo did not return with status '
82                                 'SUCCESS: %s' % network_properties['error'])
83
84        if network_properties['result']['ConnectionState'] == 'Connected':
85            return True
86
87        return False
88
89
90    def _cellular_service(self):
91        """Find cellular service.
92
93        @return Cellular service dict
94
95        """
96        cell_networks = self.chrome_net._chrome_testing.find_cellular_networks()
97        for service in cell_networks:
98            if service['GUID']:
99                return service
100
101        return None
102
103
104    def _find_cellular_service(self):
105        """Find and return cellular service if available.
106
107        @return Cellular service
108
109        """
110        utils.poll_for_condition(lambda: self._cellular_service() is not None,
111                                 exception=error.TestFail('No cell service.'),
112                                 sleep_interval=1,
113                                 timeout=60)
114        return self._cellular_service()
115
116
117    def _connect_to_cellular_network(self, service):
118        """Connect to cellular network.
119
120        @param service: Cellular service dict
121
122        """
123        logging.debug('_connect_to_cellular_network')
124        if service is None:
125            raise error.TestFail('GUID not available for cellular network.')
126        self.chrome_net.connect_to_network(service)
127        self.chrome_net.scan_for_networks()
128
129
130    def _autoconnect_cellular(self):
131        """Verify that the DUT is able to autoconnect to cellular network."""
132        logging.debug('_autoconnect_cellular')
133        service = self._find_cellular_service()
134        logging.debug('Cellular service: %s', service)
135
136        if service['ConnectionState'] == 'NotConnected':
137            self._connect_to_cellular_network(service)
138
139        self._set_autoconnect(service)
140
141        logging.debug('Suspend and resume device')
142        sys_power.do_suspend(20)
143        service = self._find_cellular_service()
144
145        utils.poll_for_condition(
146                lambda: self._is_cellular_network_connected(service),
147                exception=error.TestFail('Network not connected after suspend '
148                                         'and resume.'),
149                sleep_interval=1,
150                timeout=60)
151        logging.debug('Autoconnect works after suspend/resume.')
152
153
154    def _get_networks(self, network_type='All'):
155        """Get available networks with getNetworks api.
156
157        @param network_type: Type of network, defaults to All
158        @return List of networks found
159
160        """
161        logging.debug('_get_networks')
162        properties = json.dumps({'networkType': network_type,
163                                 'visible': True,
164                                 'limit': 2})
165        network_list = self.chrome_net._chrome_testing.call_test_function(
166                test_utils.LONG_TIMEOUT,
167                'getNetworks',
168                (properties))
169        return network_list
170
171
172    def _ethernet_preferred_over_cellular(self):
173        """Verify that the DUT prefers ethernet over cellular connection."""
174        logging.debug('_ethernet_preferred_over_cellular')
175        self.chrome_net.disable_network_device(self.chrome_net.WIFI_DEVICE)
176        network_list = self._get_networks()
177
178        if not len(network_list['result']) > 1:
179            logging.debug('Available networks: %s', network_list)
180            raise error.TestFail('Not enough networks available to check '
181                                 'network preference. Need minimum 2 networks '
182                                 'to do a successfull comparision.')
183
184        formatted_network_list = self._extract_network_info(
185                network_list['result'])
186
187        logging.debug('Available network list: %s', formatted_network_list)
188
189        if not formatted_network_list[0].networkType == 'Ethernet':
190            raise error.TestFail('Ethernet is not preferred.')
191
192        if not formatted_network_list[1].networkType == 'Cellular':
193            raise error.TestFail('Cellular is not available to determine '
194                                 'network preference.')
195
196        if (not formatted_network_list[0].connectionState == 'Connected' or
197            not formatted_network_list[1].connectionState == 'Connected'):
198            raise error.TestFail('Ethernet and Cellular should both be '
199                                 'connected to successfully determine '
200                                 'network preference.')
201
202        logging.debug('Ethernet is preferred over cellular.')
203
204
205    def _enable_disable_network_check(
206            self, original_enabled_networks, new_enabled_networks):
207        """Tests enabling and disabling of Cellular.
208
209        @param original_enabled_networks: Original list of network devices that
210                were enabled when the test started.
211        @param new_enabled_networks: New list of network devices that are
212                enabled now.
213        @raises error.TestFail if Cellular state is not toggled.
214
215        """
216        # Make sure we leave the Cellular modem in enabled state before
217        # ending the test.
218        logging.debug('_enable_disable_network_check')
219        self.chrome_net.enable_network_device(self.chrome_net.CELLULAR)
220
221        if self.chrome_net.CELLULAR in original_enabled_networks:
222            if self.chrome_net.CELLULAR in new_enabled_networks:
223                raise error.TestFail('Cellular was not disabled.')
224            elif self.chrome_net.CELLULAR not in new_enabled_networks:
225                logging.info('Cellular was successfully disabled.')
226
227        if self.chrome_net.CELLULAR not in original_enabled_networks:
228            if self.chrome_net.CELLULAR not in new_enabled_networks:
229                raise error.TestFail('Cellular was not enabled.')
230            elif self.chrome_net.CELLULAR in new_enabled_networks:
231                logging.info('Cellular was successfully enabled.')
232
233
234    def _enable_disable_cellular(self):
235        """Verify that the test is able to enable and disable Cellular."""
236        logging.debug('_enable_disable_cellular')
237        original_enabled_networks = self.chrome_net.get_enabled_devices()
238        if self.chrome_net.CELLULAR in original_enabled_networks:
239            self.chrome_net.disable_network_device(self.chrome_net.CELLULAR)
240        else:
241            self.chrome_net.enable_network_device(self.chrome_net.CELLULAR)
242        new_enabled_networks = self.chrome_net.get_enabled_devices()
243        self._enable_disable_network_check(
244                original_enabled_networks, new_enabled_networks)
245
246
247    def run_once(self, test):
248        """Runs the test.
249
250        @param test: Set by the server test control file depending on the test
251                that is being run.
252
253        """
254        with cntc.ChromeNetworkingTestContext() as testing_context:
255            self._chrome_testing = testing_context
256            self.chrome_net = cnta.ChromeNetworkProvider(testing_context)
257            enabled_devices = self.chrome_net.get_enabled_devices()
258
259            logging.debug('Enabled devices: %s', enabled_devices)
260            if (self.chrome_net.CELLULAR not in enabled_devices):
261                self.chrome_net.enable_network_device(
262                    self.chrome_net.CELLULAR)
263            self.chrome_net.scan_for_networks()
264
265            if test == 'autoconnectCellular':
266                self._autoconnect_cellular()
267            elif test == 'ethernetPreferred':
268                self._ethernet_preferred_over_cellular()
269            elif test == 'enableDisableCellular':
270                self._enable_disable_cellular()
271