• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1#!/usr/bin/env python
2#
3# Copyright 2016 - The Android Open Source Project
4#
5# Licensed under the Apache License, Version 2.0 (the "License");
6# you may not use this file except in compliance with the License.
7# You may obtain a copy of the License at
8#
9#     http://www.apache.org/licenses/LICENSE-2.0
10#
11# Unless required by applicable law or agreed to in writing, software
12# distributed under the License is distributed on an "AS IS" BASIS,
13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14# See the License for the specific language governing permissions and
15# limitations under the License.
16
17"""Public Device Driver APIs.
18
19This module provides public device driver APIs that can be called
20as a Python library.
21
22TODO: The following APIs have not been implemented
23  - RebootAVD(ip):
24  - RegisterSshPubKey(username, key):
25  - UnregisterSshPubKey(username, key):
26  - CleanupStaleImages():
27  - CleanupStaleDevices():
28"""
29
30from __future__ import print_function
31import logging
32import os
33
34from acloud import errors
35from acloud.public import avd
36from acloud.public import report
37from acloud.public.actions import common_operations
38from acloud.internal import constants
39from acloud.internal.lib import auth
40from acloud.internal.lib import android_build_client
41from acloud.internal.lib import android_compute_client
42from acloud.internal.lib import gstorage_client
43from acloud.internal.lib import utils
44from acloud.internal.lib.adb_tools import AdbTools
45
46
47logger = logging.getLogger(__name__)
48
49MAX_BATCH_CLEANUP_COUNT = 100
50
51_SSH_USER = "root"
52
53
54# pylint: disable=invalid-name
55class AndroidVirtualDevicePool():
56    """A class that manages a pool of devices."""
57
58    def __init__(self, cfg, devices=None):
59        self._devices = devices or []
60        self._cfg = cfg
61        credentials = auth.CreateCredentials(cfg)
62        self._build_client = android_build_client.AndroidBuildClient(
63            credentials)
64        self._storage_client = gstorage_client.StorageClient(credentials)
65        self._compute_client = android_compute_client.AndroidComputeClient(
66            cfg, credentials)
67
68    @utils.TimeExecute("Creating GCE image")
69    def _CreateGceImageWithBuildInfo(self, build_target, build_id):
70        """Creates a Gce image using build from Launch Control.
71
72        Clone avd-system.tar.gz of a build to a cache storage bucket
73        using launch control api. And then create a Gce image.
74
75        Args:
76            build_target: Target name, e.g. "aosp_cf_x86_64_phone-userdebug"
77            build_id: Build id, a string, e.g. "2263051", "P2804227"
78
79        Returns:
80            String, name of the Gce image that has been created.
81        """
82        logger.info("Creating a new gce image using build: build_id %s, "
83                    "build_target %s", build_id, build_target)
84        disk_image_id = utils.GenerateUniqueName(
85            suffix=self._cfg.disk_image_name)
86        self._build_client.CopyTo(
87            build_target,
88            build_id,
89            artifact_name=self._cfg.disk_image_name,
90            destination_bucket=self._cfg.storage_bucket_name,
91            destination_path=disk_image_id)
92        disk_image_url = self._storage_client.GetUrl(
93            self._cfg.storage_bucket_name, disk_image_id)
94        try:
95            image_name = self._compute_client.GenerateImageName(build_target,
96                                                                build_id)
97            self._compute_client.CreateImage(image_name=image_name,
98                                             source_uri=disk_image_url)
99        finally:
100            self._storage_client.Delete(self._cfg.storage_bucket_name,
101                                        disk_image_id)
102        return image_name
103
104    @utils.TimeExecute("Creating GCE image")
105    def _CreateGceImageWithLocalFile(self, local_disk_image):
106        """Create a Gce image with a local image file.
107
108        The local disk image can be either a tar.gz file or a
109        raw vmlinux image.
110        e.g.  /tmp/avd-system.tar.gz or /tmp/android_system_disk_syslinux.img
111        If a raw vmlinux image is provided, it will be archived into a tar.gz file.
112
113        The final tar.gz file will be uploaded to a cache bucket in storage.
114
115        Args:
116            local_disk_image: string, path to a local disk image,
117
118        Returns:
119            String, name of the Gce image that has been created.
120
121        Raises:
122            DriverError: if a file with an unexpected extension is given.
123        """
124        logger.info("Creating a new gce image from a local file %s",
125                    local_disk_image)
126        with utils.TempDir() as tempdir:
127            if local_disk_image.endswith(self._cfg.disk_raw_image_extension):
128                dest_tar_file = os.path.join(tempdir,
129                                             self._cfg.disk_image_name)
130                utils.MakeTarFile(
131                    src_dict={local_disk_image: self._cfg.disk_raw_image_name},
132                    dest=dest_tar_file)
133                local_disk_image = dest_tar_file
134            elif not local_disk_image.endswith(self._cfg.disk_image_extension):
135                raise errors.DriverError(
136                    "Wrong local_disk_image type, must be a *%s file or *%s file"
137                    % (self._cfg.disk_raw_image_extension,
138                       self._cfg.disk_image_extension))
139
140            disk_image_id = utils.GenerateUniqueName(
141                suffix=self._cfg.disk_image_name)
142            self._storage_client.Upload(
143                local_src=local_disk_image,
144                bucket_name=self._cfg.storage_bucket_name,
145                object_name=disk_image_id,
146                mime_type=self._cfg.disk_image_mime_type)
147        disk_image_url = self._storage_client.GetUrl(
148            self._cfg.storage_bucket_name, disk_image_id)
149        try:
150            image_name = self._compute_client.GenerateImageName()
151            self._compute_client.CreateImage(image_name=image_name,
152                                             source_uri=disk_image_url)
153        finally:
154            self._storage_client.Delete(self._cfg.storage_bucket_name,
155                                        disk_image_id)
156        return image_name
157
158    # pylint: disable=too-many-locals
159    def CreateDevices(self,
160                      num,
161                      build_target=None,
162                      build_id=None,
163                      gce_image=None,
164                      local_disk_image=None,
165                      cleanup=True,
166                      extra_data_disk_size_gb=None,
167                      precreated_data_image=None,
168                      avd_spec=None,
169                      extra_scopes=None):
170        """Creates |num| devices for given build_target and build_id.
171
172        - If gce_image is provided, will use it to create an instance.
173        - If local_disk_image is provided, will upload it to a temporary
174          caching storage bucket which is defined by user as |storage_bucket_name|
175          And then create an gce image with it; and then create an instance.
176        - If build_target and build_id are provided, will clone the disk image
177          via launch control to the temporary caching storage bucket.
178          And then create an gce image with it; and then create an instance.
179
180        Args:
181            num: Number of devices to create.
182            build_target: Target name, e.g. "aosp_cf_x86_64_phone-userdebug"
183            build_id: Build id, a string, e.g. "2263051", "P2804227"
184            gce_image: string, if given, will use this image
185                       instead of creating a new one.
186                       implies cleanup=False.
187            local_disk_image: string, path to a local disk image, e.g.
188                              /tmp/avd-system.tar.gz
189            cleanup: boolean, if True clean up compute engine image after creating
190                     the instance.
191            extra_data_disk_size_gb: Integer, size of extra disk, or None.
192            precreated_data_image: A string, the image to use for the extra disk.
193            avd_spec: AVDSpec object for pass hw_property.
194            extra_scopes: A list of extra scopes given to the new instance.
195
196        Raises:
197            errors.DriverError: If no source is specified for image creation.
198        """
199        if gce_image:
200            # GCE image is provided, we can directly move to instance creation.
201            logger.info("Using existing gce image %s", gce_image)
202            image_name = gce_image
203            cleanup = False
204        elif local_disk_image:
205            image_name = self._CreateGceImageWithLocalFile(local_disk_image)
206        elif build_target and build_id:
207            image_name = self._CreateGceImageWithBuildInfo(build_target,
208                                                           build_id)
209        else:
210            raise errors.DriverError(
211                "Invalid image source, must specify one of the following: gce_image, "
212                "local_disk_image, or build_target and build id.")
213
214        # Create GCE instances.
215        try:
216            for _ in range(num):
217                instance = self._compute_client.GenerateInstanceName(
218                    build_target, build_id)
219                extra_disk_name = None
220                if extra_data_disk_size_gb > 0:
221                    extra_disk_name = self._compute_client.GetDataDiskName(
222                        instance)
223                    self._compute_client.CreateDisk(extra_disk_name,
224                                                    precreated_data_image,
225                                                    extra_data_disk_size_gb,
226                                                    disk_type=avd_spec.disk_type)
227                self._compute_client.CreateInstance(
228                    instance=instance,
229                    image_name=image_name,
230                    extra_disk_name=extra_disk_name,
231                    avd_spec=avd_spec,
232                    extra_scopes=extra_scopes)
233                ip = self._compute_client.GetInstanceIP(instance)
234                self.devices.append(avd.AndroidVirtualDevice(
235                    ip=ip, instance_name=instance))
236        finally:
237            if cleanup:
238                self._compute_client.DeleteImage(image_name)
239
240    def DeleteDevices(self):
241        """Deletes devices.
242
243        Returns:
244            A tuple, (deleted, failed, error_msgs)
245            deleted: A list of names of instances that have been deleted.
246            faild: A list of names of instances that we fail to delete.
247            error_msgs: A list of failure messages.
248        """
249        instance_names = [device.instance_name for device in self._devices]
250        return self._compute_client.DeleteInstances(instance_names,
251                                                    self._cfg.zone)
252
253    @utils.TimeExecute("Waiting for AVD to boot")
254    def WaitForBoot(self):
255        """Waits for all devices to boot up.
256
257        Returns:
258            A dictionary that contains all the failures.
259            The key is the name of the instance that fails to boot,
260            the value is an errors.DeviceBoottError object.
261        """
262        failures = {}
263        for device in self._devices:
264            try:
265                self._compute_client.WaitForBoot(device.instance_name)
266            except errors.DeviceBootError as e:
267                failures[device.instance_name] = e
268        return failures
269
270    @property
271    def devices(self):
272        """Returns a list of devices in the pool.
273
274        Returns:
275            A list of devices in the pool.
276        """
277        return self._devices
278
279
280def AddDeletionResultToReport(report_obj, deleted, failed, error_msgs,
281                              resource_name):
282    """Adds deletion result to a Report object.
283
284    This function will add the following to report.data.
285      "deleted": [
286         {"name": "resource_name", "type": "resource_name"},
287       ],
288      "failed": [
289         {"name": "resource_name", "type": "resource_name"},
290       ],
291    This function will append error_msgs to report.errors.
292
293    Args:
294        report_obj: A Report object.
295        deleted: A list of names of the resources that have been deleted.
296        failed: A list of names of the resources that we fail to delete.
297        error_msgs: A list of error message strings to be added to the report.
298        resource_name: A string, representing the name of the resource.
299    """
300    for name in deleted:
301        report_obj.AddData(key="deleted",
302                           value={"name": name,
303                                  "type": resource_name})
304    for name in failed:
305        report_obj.AddData(key="failed",
306                           value={"name": name,
307                                  "type": resource_name})
308    report_obj.AddErrors(error_msgs)
309    if failed or error_msgs:
310        report_obj.SetStatus(report.Status.FAIL)
311
312
313def _FetchSerialLogsFromDevices(compute_client, instance_names, output_file,
314                                port):
315    """Fetch serial logs from a port for a list of devices to a local file.
316
317    Args:
318        compute_client: An object of android_compute_client.AndroidComputeClient
319        instance_names: A list of instance names.
320        output_file: A path to a file ending with "tar.gz"
321        port: The number of serial port to read from, 0 for serial output, 1 for
322              logcat.
323    """
324    with utils.TempDir() as tempdir:
325        src_dict = {}
326        for instance_name in instance_names:
327            serial_log = compute_client.GetSerialPortOutput(
328                instance=instance_name, port=port)
329            file_name = "%s.log" % instance_name
330            file_path = os.path.join(tempdir, file_name)
331            src_dict[file_path] = file_name
332            with open(file_path, "w") as f:
333                f.write(serial_log.encode("utf-8"))
334        utils.MakeTarFile(src_dict, output_file)
335
336
337# pylint: disable=too-many-locals
338def CreateGCETypeAVD(cfg,
339                     build_target=None,
340                     build_id=None,
341                     num=1,
342                     gce_image=None,
343                     local_disk_image=None,
344                     cleanup=True,
345                     serial_log_file=None,
346                     autoconnect=False,
347                     report_internal_ip=False,
348                     avd_spec=None):
349    """Creates one or multiple gce android devices.
350
351    Args:
352        cfg: An AcloudConfig instance.
353        build_target: Target name, e.g. "aosp_cf_x86_64_phone-userdebug"
354        build_id: Build id, a string, e.g. "2263051", "P2804227"
355        num: Number of devices to create.
356        gce_image: string, if given, will use this gce image
357                   instead of creating a new one.
358                   implies cleanup=False.
359        local_disk_image: string, path to a local disk image, e.g.
360                          /tmp/avd-system.tar.gz
361        cleanup: boolean, if True clean up compute engine image and
362                 disk image in storage after creating the instance.
363        serial_log_file: A path to a file where serial output should
364                         be saved to.
365        autoconnect: Create ssh tunnel(s) and adb connect after device creation.
366        report_internal_ip: Boolean to report the internal ip instead of
367                            external ip.
368        avd_spec: AVDSpec object for pass hw_property.
369
370    Returns:
371        A Report instance.
372    """
373    r = report.Report(command="create")
374    credentials = auth.CreateCredentials(cfg)
375    compute_client = android_compute_client.AndroidComputeClient(cfg,
376                                                                 credentials)
377    try:
378        common_operations.CreateSshKeyPairIfNecessary(cfg)
379        device_pool = AndroidVirtualDevicePool(cfg)
380        device_pool.CreateDevices(
381            num,
382            build_target,
383            build_id,
384            gce_image,
385            local_disk_image,
386            cleanup,
387            extra_data_disk_size_gb=cfg.extra_data_disk_size_gb,
388            precreated_data_image=cfg.precreated_data_image_map.get(
389                cfg.extra_data_disk_size_gb),
390            avd_spec=avd_spec,
391            extra_scopes=cfg.extra_scopes)
392        failures = device_pool.WaitForBoot()
393        # Write result to report.
394        for device in device_pool.devices:
395            ip = (device.ip.internal if report_internal_ip
396                  else device.ip.external)
397            device_dict = {
398                "ip": ip,
399                "instance_name": device.instance_name
400            }
401            if autoconnect:
402                forwarded_ports = utils.AutoConnect(
403                    ip_addr=ip,
404                    rsa_key_file=cfg.ssh_private_key_path,
405                    target_vnc_port=constants.GCE_VNC_PORT,
406                    target_adb_port=constants.GCE_ADB_PORT,
407                    ssh_user=_SSH_USER,
408                    client_adb_port=avd_spec.client_adb_port,
409                    extra_args_ssh_tunnel=cfg.extra_args_ssh_tunnel)
410                device_dict[constants.VNC_PORT] = forwarded_ports.vnc_port
411                device_dict[constants.ADB_PORT] = forwarded_ports.adb_port
412                if avd_spec.unlock_screen:
413                    AdbTools(forwarded_ports.adb_port).AutoUnlockScreen()
414            if device.instance_name in failures:
415                r.AddData(key="devices_failing_boot", value=device_dict)
416                r.AddError(str(failures[device.instance_name]))
417            else:
418                r.AddData(key="devices", value=device_dict)
419        if failures:
420            r.SetStatus(report.Status.BOOT_FAIL)
421        else:
422            r.SetStatus(report.Status.SUCCESS)
423
424    except errors.DriverError as e:
425        r.AddError(str(e))
426        r.SetStatus(report.Status.FAIL)
427    finally:
428         # Let's do our best to obtain the serial log, even though this
429         # could fail in case of failed boots.
430        if serial_log_file:
431            instance_names=[d.instance_name for d in device_pool.devices]
432            try:
433                _FetchSerialLogsFromDevices(
434                    compute_client,
435                    instance_names=instance_names,
436                    port=constants.DEFAULT_SERIAL_PORT,
437                    output_file=serial_log_file)
438            except Exception as log_err:
439                logging.warning("Failed to obtain serial logs from %s", ", ".join(instance_names))
440    return r
441
442
443def DeleteAndroidVirtualDevices(cfg, instance_names, default_report=None):
444    """Deletes android devices.
445
446    Args:
447        cfg: An AcloudConfig instance.
448        instance_names: A list of names of the instances to delete.
449        default_report: A initialized Report instance.
450
451    Returns:
452        A Report instance.
453    """
454    # delete, failed, error_msgs are used to record result.
455    deleted = []
456    failed = []
457    error_msgs = []
458
459    r = default_report if default_report else report.Report(command="delete")
460    credentials = auth.CreateCredentials(cfg)
461    compute_client = android_compute_client.AndroidComputeClient(cfg,
462                                                                 credentials)
463    zone_instances = compute_client.GetZonesByInstances(instance_names)
464
465    try:
466        for zone, instances in zone_instances.items():
467            deleted_ins, failed_ins, error_ins = compute_client.DeleteInstances(
468                instances, zone)
469            deleted.extend(deleted_ins)
470            failed.extend(failed_ins)
471            error_msgs.extend(error_ins)
472        AddDeletionResultToReport(
473            r, deleted,
474            failed, error_msgs,
475            resource_name="instance")
476        if r.status == report.Status.UNKNOWN:
477            r.SetStatus(report.Status.SUCCESS)
478    except errors.DriverError as e:
479        r.AddError(str(e))
480        r.SetStatus(report.Status.FAIL)
481    return r
482
483
484def CheckAccess(cfg):
485    """Check if user has access.
486
487    Args:
488         cfg: An AcloudConfig instance.
489    """
490    credentials = auth.CreateCredentials(cfg)
491    compute_client = android_compute_client.AndroidComputeClient(
492        cfg, credentials)
493    logger.info("Checking if user has access to project %s", cfg.project)
494    if not compute_client.CheckAccess():
495        logger.error("User does not have access to project %s", cfg.project)
496        # Print here so that command line user can see it.
497        print("Looks like you do not have access to %s. " % cfg.project)
498        if cfg.project in cfg.no_project_access_msg_map:
499            print(cfg.no_project_access_msg_map[cfg.project])
500