deb-heat/heat/tests/test_swift_client.py
Steve Baker 7470a5e919 Deployment signal_transport: TEMP_URL_SIGNAL
This change allows swift TempURLs to be used to signal heat from
server deployments.

When signal_transport: TEMP_URL_SIGNAL is specified, the following will happen:
* On creation a swift object will be created with an associated TempURL for PUTs
* The PUT URL is used for the deploy_signal_id input value, along with the new
  deploy_signal_verb input being set to PUT
* In the deployment resource check_complete, the swift object is polled for
  new signal contents, and signal() is called when the swift object contents change

By specifying deployment signal_transport: TEMP_URL_SIGNAL and
server software_config_transport: POLL_TEMP_URL it should now be possible to use
config/deployment resources with only a keystone v2 API (or with a standalone heat)

Implements-Blueprint: software-config-swift-signal

Change-Id: I1dfba248dcfc90c3d872ba35f0aa935cca5c5606
2015-02-25 13:48:52 +13:00

139 lines
5.1 KiB
Python

#
# 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.
import datetime
import mock
from oslo_utils import timeutils
import pytz
from testtools import matchers
from heat.engine.clients.os import swift
from heat.tests import common
from heat.tests import utils
class SwiftClientPluginTestCase(common.HeatTestCase):
def setUp(self):
super(SwiftClientPluginTestCase, self).setUp()
self.swift_client = mock.Mock()
self.context = utils.dummy_context()
self.context.tenant_id = "demo"
c = self.context.clients
self.swift_plugin = c.client_plugin('swift')
self.swift_plugin._client = self.swift_client
class SwiftUtilsTests(SwiftClientPluginTestCase):
def test_is_valid_temp_url_path(self):
valids = [
"/v1/AUTH_demo/c/o",
"/v1/AUTH_demo/c/o/",
"/v1/TEST_demo/c/o",
"/v1/AUTH_demo/c/pseudo_folder/o",
]
for url in valids:
self.assertTrue(self.swift_plugin.is_valid_temp_url_path(url))
invalids = [
"/v2/AUTH_demo/c/o",
"/v1/AUTH_demo/c//",
"/v1/AUTH_demo/c/",
"/AUTH_demo/c//",
"//AUTH_demo/c/o",
"//v1/AUTH_demo/c/o",
"/v1/AUTH_demo/o",
"/v1/AUTH_demo//o",
"/v1/AUTH_d3mo//o",
"/v1//c/o",
"/v1/c/o",
]
for url in invalids:
self.assertFalse(self.swift_plugin.is_valid_temp_url_path(url))
def test_get_temp_url(self):
self.swift_client.url = ("http://fake-host.com:8080/v1/"
"AUTH_demo")
self.swift_client.head_account = mock.Mock(return_value={
'x-account-meta-temp-url-key': '123456'})
self.swift_client.post_account = mock.Mock()
container_name = '1234' # from stack.id
stack_name = 'test'
handle_name = 'foo'
obj_name = '%s-%s' % (stack_name, handle_name)
url = self.swift_plugin.get_temp_url(container_name, obj_name)
self.assertFalse(self.swift_client.post_account.called)
regexp = ("http://fake-host.com:8080/v1/AUTH_demo/%s"
"/%s\?temp_url_sig=[0-9a-f]{40}&"
"temp_url_expires=[0-9]{10}" %
(container_name, obj_name))
self.assertThat(url, matchers.MatchesRegex(regexp))
timeout = int(url.split('=')[-1])
self.assertTrue(timeout < swift.MAX_EPOCH)
def test_get_temp_url_no_account_key(self):
self.swift_client.url = ("http://fake-host.com:8080/v1/"
"AUTH_demo")
head_account = {}
def post_account(data):
head_account.update(data)
self.swift_client.head_account = mock.Mock(return_value=head_account)
self.swift_client.post_account = post_account
container_name = '1234' # from stack.id
stack_name = 'test'
handle_name = 'foo'
obj_name = '%s-%s' % (stack_name, handle_name)
self.assertNotIn('x-account-meta-temp-url-key', head_account)
self.swift_plugin.get_temp_url(container_name, obj_name)
self.assertIn('x-account-meta-temp-url-key', head_account)
def test_get_signal_url(self):
self.swift_client.url = ("http://fake-host.com:8080/v1/"
"AUTH_demo")
self.swift_client.head_account = mock.Mock(return_value={
'x-account-meta-temp-url-key': '123456'})
self.swift_client.post_account = mock.Mock()
container_name = '1234' # from stack.id
stack_name = 'test'
handle_name = 'foo'
obj_name = '%s-%s' % (stack_name, handle_name)
url = self.swift_plugin.get_signal_url(container_name, obj_name)
self.assertTrue(self.swift_client.put_container.called)
self.assertTrue(self.swift_client.put_object.called)
regexp = ("http://fake-host.com:8080/v1/AUTH_demo/%s"
"/%s\?temp_url_sig=[0-9a-f]{40}&"
"temp_url_expires=[0-9]{10}" %
(container_name, obj_name))
self.assertThat(url, matchers.MatchesRegex(regexp))
def test_parse_last_modified(self):
self.assertIsNone(self.swift_plugin.parse_last_modified(None))
now = datetime.datetime(
2015, 2, 5, 1, 4, 40, 0, pytz.timezone('GMT'))
now_naive = datetime.datetime(
2015, 2, 5, 1, 4, 40, 0)
last_modified = timeutils.strtime(now, '%a, %d %b %Y %H:%M:%S %Z')
self.assertEqual('Thu, 05 Feb 2015 01:04:40 GMT', last_modified)
self.assertEqual(
now_naive,
self.swift_plugin.parse_last_modified(last_modified))