summaryrefslogtreecommitdiff
path: root/azurelinuxagent/common/protocol/hostplugin.py
diff options
context:
space:
mode:
Diffstat (limited to 'azurelinuxagent/common/protocol/hostplugin.py')
-rw-r--r--azurelinuxagent/common/protocol/hostplugin.py124
1 files changed, 124 insertions, 0 deletions
diff --git a/azurelinuxagent/common/protocol/hostplugin.py b/azurelinuxagent/common/protocol/hostplugin.py
new file mode 100644
index 0000000..6569604
--- /dev/null
+++ b/azurelinuxagent/common/protocol/hostplugin.py
@@ -0,0 +1,124 @@
+# Microsoft Azure Linux Agent
+#
+# Copyright 2014 Microsoft Corporation
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+# Requires Python 2.4+ and Openssl 1.0+
+#
+
+from azurelinuxagent.common.protocol.wire import *
+from azurelinuxagent.common.utils import textutil
+
+HOST_PLUGIN_PORT = 32526
+URI_FORMAT_GET_API_VERSIONS = "http://{0}:{1}/versions"
+URI_FORMAT_PUT_VM_STATUS = "http://{0}:{1}/status"
+URI_FORMAT_PUT_LOG = "http://{0}:{1}/vmAgentLog"
+API_VERSION = "2015-09-01"
+
+
+class HostPluginProtocol(object):
+ def __init__(self, endpoint):
+ if endpoint is None:
+ raise ProtocolError("Host plugin endpoint not provided")
+ self.is_initialized = False
+ self.is_available = False
+ self.api_versions = None
+ self.endpoint = endpoint
+
+ def ensure_initialized(self):
+ if not self.is_initialized:
+ self.api_versions = self.get_api_versions()
+ self.is_available = API_VERSION in self.api_versions
+ self.is_initialized = True
+ return self.is_available
+
+ def get_api_versions(self):
+ url = URI_FORMAT_GET_API_VERSIONS.format(self.endpoint,
+ HOST_PLUGIN_PORT)
+ logger.info("getting API versions at [{0}]".format(url))
+ try:
+ response = restutil.http_get(url)
+ if response.status != httpclient.OK:
+ logger.error(
+ "get API versions returned status code [{0}]".format(
+ response.status))
+ return []
+ return response.read()
+ except HttpError as e:
+ logger.error("get API versions failed with [{0}]".format(e))
+ return []
+
+ def put_vm_status(self, status_blob, sas_url):
+ """
+ Try to upload the VM status via the host plugin /status channel
+ :param sas_url: the blob SAS url to pass to the host plugin
+ :type status_blob: StatusBlob
+ """
+ if not self.ensure_initialized():
+ logger.error("host plugin channel is not available")
+ return
+ if status_blob is None or status_blob.vm_status is None:
+ logger.error("no status data was provided")
+ return
+ url = URI_FORMAT_PUT_VM_STATUS.format(self.endpoint, HOST_PLUGIN_PORT)
+ status = textutil.b64encode(status_blob.vm_status)
+ headers = {"x-ms-version": API_VERSION}
+ blob_headers = [{'headerName': 'x-ms-version',
+ 'headerValue': status_blob.__storage_version__},
+ {'headerName': 'x-ms-blob-type',
+ 'headerValue': status_blob.type}]
+ data = json.dumps({'requestUri': sas_url, 'headers': blob_headers,
+ 'content': status}, sort_keys=True)
+ logger.info("put VM status at [{0}]".format(url))
+ try:
+ response = restutil.http_put(url, data, headers)
+ if response.status != httpclient.OK:
+ logger.error("put VM status returned status code [{0}]".format(
+ response.status))
+ except HttpError as e:
+ logger.error("put VM status failed with [{0}]".format(e))
+
+ def put_vm_log(self, content, container_id, deployment_id):
+ """
+ Try to upload the given content to the host plugin
+ :param deployment_id: the deployment id, which is obtained from the
+ goal state (tenant name)
+ :param container_id: the container id, which is obtained from the
+ goal state
+ :param content: the binary content of the zip file to upload
+ :return:
+ """
+ if not self.ensure_initialized():
+ logger.error("host plugin channel is not available")
+ return
+ if content is None or container_id is None or deployment_id is None:
+ logger.error(
+ "invalid arguments passed: "
+ "[{0}], [{1}], [{2}]".format(
+ content,
+ container_id,
+ deployment_id))
+ return
+ url = URI_FORMAT_PUT_LOG.format(self.endpoint, HOST_PLUGIN_PORT)
+
+ headers = {"x-ms-vmagentlog-deploymentid": deployment_id,
+ "x-ms-vmagentlog-containerid": container_id}
+ logger.info("put VM log at [{0}]".format(url))
+ try:
+ response = restutil.http_put(url, content, headers)
+ if response.status != httpclient.OK:
+ logger.error("put log returned status code [{0}]".format(
+ response.status))
+ except HttpError as e:
+ logger.error("put log failed with [{0}]".format(e))