| # 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 html.parser |
| import ssl |
| from urllib import parse |
| from urllib import request |
| |
| from tempest.common import utils |
| from tempest import config |
| from tempest.lib import decorators |
| from tempest import test |
| |
| CONF = config.CONF |
| |
| |
| class HorizonHTMLParser(html.parser.HTMLParser): |
| csrf_token = None |
| region = None |
| login = None |
| |
| def _find_name(self, attrs, name): |
| for attrpair in attrs: |
| if attrpair[0] == 'name' and attrpair[1] == name: |
| return True |
| return False |
| |
| def _find_value(self, attrs): |
| for attrpair in attrs: |
| if attrpair[0] == 'value': |
| return attrpair[1] |
| return None |
| |
| def _find_attr_value(self, attrs, attr_name): |
| for attrpair in attrs: |
| if attrpair[0] == attr_name: |
| return attrpair[1] |
| return None |
| |
| def handle_starttag(self, tag, attrs): |
| if tag == 'input': |
| if self._find_name(attrs, 'csrfmiddlewaretoken'): |
| self.csrf_token = self._find_value(attrs) |
| if self._find_name(attrs, 'region'): |
| self.region = self._find_value(attrs) |
| if tag == 'form': |
| self.login = self._find_attr_value(attrs, 'action') |
| |
| |
| class TestDashboardBasicOps(test.BaseTestCase): |
| |
| """The test suite for dashboard basic operations |
| |
| This is a basic scenario test: |
| * checks that the login page is available |
| * logs in as a regular user |
| * checks that the user home page loads without error |
| """ |
| opener = None |
| |
| credentials = ['primary'] |
| |
| @classmethod |
| def skip_checks(cls): |
| super(TestDashboardBasicOps, cls).skip_checks() |
| if not CONF.service_available.horizon: |
| raise cls.skipException("Horizon support is required") |
| |
| @classmethod |
| def setup_credentials(cls): |
| cls.set_network_resources() |
| super(TestDashboardBasicOps, cls).setup_credentials() |
| |
| def check_login_page(self): |
| response = self._get_opener().open(CONF.dashboard.dashboard_url).read() |
| self.assertIn("id_username", response.decode("utf-8")) |
| |
| def user_login(self, username, password): |
| response = self._get_opener().open(CONF.dashboard.dashboard_url).read() |
| |
| # Grab the CSRF token and default region |
| parser = HorizonHTMLParser() |
| parser.feed(response.decode("utf-8")) |
| |
| # construct login url for dashboard, discovery accommodates non-/ web |
| # root for dashboard |
| login_url = parse.urljoin(CONF.dashboard.dashboard_url, parser.login) |
| |
| # Prepare login form request |
| req = request.Request(login_url) |
| req.add_header('Content-type', 'application/x-www-form-urlencoded') |
| req.add_header('Referer', CONF.dashboard.dashboard_url) |
| |
| # Pass the default domain name regardless of the auth version in order |
| # to test the scenario of when horizon is running with keystone v3 |
| params = {'username': username, |
| 'password': password, |
| 'region': parser.region, |
| 'domain': CONF.auth.default_credentials_domain_name, |
| 'csrfmiddlewaretoken': parser.csrf_token} |
| self._get_opener().open(req, parse.urlencode(params).encode()) |
| |
| def check_home_page(self): |
| response = self._get_opener().open(CONF.dashboard.dashboard_url).read() |
| self.assertIn('Overview', response.decode("utf-8")) |
| |
| def _get_opener(self): |
| if not self.opener: |
| if (CONF.dashboard.disable_ssl_certificate_validation and |
| self._ssl_default_context_supported()): |
| ctx = ssl.create_default_context() |
| ctx.check_hostname = False |
| ctx.verify_mode = ssl.CERT_NONE |
| self.opener = request.build_opener( |
| request.HTTPSHandler(context=ctx), |
| request.HTTPCookieProcessor()) |
| else: |
| self.opener = request.build_opener( |
| request.HTTPCookieProcessor()) |
| return self.opener |
| |
| def _ssl_default_context_supported(self): |
| return (hasattr(ssl, 'create_default_context')) |
| |
| @decorators.attr(type='smoke') |
| @decorators.idempotent_id('4f8851b1-0e69-482b-b63b-84c6e76f6c80') |
| @utils.services('dashboard') |
| def test_basic_scenario(self): |
| creds = self.os_primary.credentials |
| self.check_login_page() |
| self.user_login(creds.username, creds.password) |
| self.check_home_page() |