blob: b86938e452dbe649783f67b187b910e401002478 [file] [log] [blame]
# vi: ts=4 expandtab
#
# Cloud-Init Datasource for VMware Guestinfo
#
# Copyright (c) 2018 VMware, Inc. All Rights Reserved.
#
# This product is licensed to you under the Apache 2.0 license (the "License").
# You may not use this product except in compliance with the Apache 2.0 License.
#
# This product may include a number of subcomponents with separate copyright
# notices and license terms. Your use of these subcomponents is subject to the
# terms and conditions of the subcomponent's license, as noted in the LICENSE
# file.
#
# Authors: Anish Swaminathan <anishs@vmware.com>
# Andrew Kutz <akutz@vmware.com>
#
import os
import base64
import zlib
import json
from cloudinit import log as logging
from cloudinit import sources
from cloudinit import util
from cloudinit import safeyaml
from distutils.spawn import find_executable
LOG = logging.getLogger(__name__)
# from cloud-init >= 20.3 subp is in its own module
try:
from cloudinit.subp import subp, ProcessExecutionError
except ImportError:
from cloudinit.util import subp, ProcessExecutionError
# This cloud-init datasource was designed for use with CentOS 7,
# which uses cloud-init 0.7.9. However, this datasource should
# work with any Linux distribution for which cloud-init is
# avaialble.
#
# The documentation for cloud-init 0.7.9's datasource is
# available at http://bit.ly/cloudinit-datasource-0-7-9. The
# current documentation for cloud-init is found at
# https://cloudinit.readthedocs.io/en/latest/.
#
# Setting the hostname:
# The hostname is set by way of the metadata key "local-hostname".
#
# Setting the instance ID:
# The instance ID may be set by way of the metadata key "instance-id".
# However, if this value is absent then then the instance ID is
# read from the file /sys/class/dmi/id/product_uuid.
#
# Configuring the network:
# The network is configured by setting the metadata key "network"
# with a value consistent with Network Config Versions 1 or 2,
# depending on the Linux distro's version of cloud-init:
#
# Network Config Version 1 - http://bit.ly/cloudinit-net-conf-v1
# Network Config Version 2 - http://bit.ly/cloudinit-net-conf-v2
#
# For example, CentOS 7's official cloud-init package is version
# 0.7.9 and does not support Network Config Version 2. However,
# this datasource still supports supplying Network Config Version 2
# data as long as the Linux distro's cloud-init package is new
# enough to parse the data.
#
# The metadata key "network.encoding" may be used to indicate the
# format of the metadata key "network". Valid encodings are base64
# and gzip+base64.
class DataSourceVMwareGuestInfo(sources.DataSource):
def __init__(self, sys_cfg, distro, paths, ud_proc=None):
sources.DataSource.__init__(self, sys_cfg, distro, paths, ud_proc)
self.vmtoolsd = find_executable("vmtoolsd")
if not self.vmtoolsd:
LOG.error("Failed to find vmtoolsd")
def get_data(self):
if not self.vmtoolsd:
LOG.error("vmtoolsd is required to fetch guestinfo value")
return False
# Get the JSON metadata. Can be plain-text, base64, or gzip+base64.
metadata = self._get_encoded_guestinfo_data('metadata')
if metadata:
try:
self.metadata = json.loads(metadata)
except:
self.metadata = safeyaml.load(metadata)
# Get the YAML userdata. Can be plain-text, base64, or gzip+base64.
self.userdata_raw = self._get_encoded_guestinfo_data('userdata')
# Get the YAML vendordata. Can be plain-text, base64, or gzip+base64.
self.vendordata_raw = self._get_encoded_guestinfo_data('vendordata')
return True
@property
def network_config(self):
# Pull the network configuration out of the metadata.
if self.metadata and 'network' in self.metadata:
data = self._get_encoded_metadata('network')
if data:
# Load the YAML-formatted network data into an object
# and return it.
net_config = safeyaml.load(data)
LOG.debug("Loaded network config: %s", net_config)
return net_config
return None
def get_instance_id(self):
# Pull the instance ID out of the metadata if present. Otherwise
# read the file /sys/class/dmi/id/product_uuid for the instance ID.
if self.metadata and 'instance-id' in self.metadata:
return self.metadata['instance-id']
with open('/sys/class/dmi/id/product_uuid', 'r') as id_file:
return str(id_file.read()).rstrip()
def _get_encoded_guestinfo_data(self, key):
data = self._get_guestinfo_value(key)
if not data:
return None
enc_type = self._get_guestinfo_value(key + '.encoding')
return self._get_encoded_data('guestinfo.' + key, enc_type, data)
def _get_encoded_metadata(self, key):
if not self.metadata or not key in self.metadata:
return None
data = self.metadata[key]
enc_type = self.metadata.get(key + '.encoding')
return self._get_encoded_data('metadata.' + key, enc_type, data)
def _get_encoded_data(self, key, enc_type, data):
'''
The _get_encoded_data would always return a str
----
In py 2.7:
json.loads method takes string as input
zlib.decompress takes and returns a string
base64.b64decode takes and returns a string
-----
In py 3.6 and newer:
json.loads method takes bytes or string as input
zlib.decompress takes and returns a bytes
base64.b64decode takes bytes or string and returns bytes
-----
In py > 3, < 3.6:
json.loads method takes string as input
zlib.decompress takes and returns a bytes
base64.b64decode takes bytes or string and returns bytes
-----
Given the above conditions the output from zlib.decompress and
base64.b64decode would be bytes with newer python and str in older
version. Thus we would covert the output to str before returning
'''
rawdata = self._get_encoded_data_raw(key, enc_type, data)
if type(rawdata) == bytes:
return rawdata.decode('utf-8')
return rawdata
def _get_encoded_data_raw(self, key, enc_type, data):
LOG.debug("Getting encoded data for key=%s, enc=%s", key, enc_type)
if enc_type == "gzip+base64" or enc_type == "gz+b64":
LOG.debug("Decoding %s format %s", enc_type, key)
return zlib.decompress(base64.b64decode(data), zlib.MAX_WBITS | 16)
elif enc_type == "base64" or enc_type == "b64":
LOG.debug("Decoding %s format %s", enc_type, key)
return base64.b64decode(data)
else:
LOG.debug("Plain-text data %s", key)
return data
def _get_guestinfo_value(self, key):
NOVAL = "No value found"
LOG.debug("Getting guestinfo value for key %s", key)
try:
(stdout, stderr) = subp([self.vmtoolsd, "--cmd", "info-get guestinfo." + key])
if stderr == NOVAL:
LOG.debug("No value found for key %s", key)
elif not stdout:
LOG.error("Failed to get guestinfo value for key %s", key)
else:
return stdout.rstrip()
except ProcessExecutionError as error:
if error.stderr == NOVAL:
LOG.debug("No value found for key %s", key)
else:
util.logexc(LOG,"Failed to get guestinfo value for key %s: %s", key, error)
except Exception:
util.logexc(LOG,"Unexpected error while trying to get guestinfo value for key %s", key)
return None
def get_datasource_list(depends):
"""
Return a list of data sources that match this set of dependencies
"""
return [DataSourceVMwareGuestInfo]