Home | History | Annotate | Download | only in testing
      1 # Copyright 2014 The Chromium Authors. All rights reserved.
      2 # Use of this source code is governed by a BSD-style license that can be
      3 # found in the LICENSE file.
      4 
      5 import logging
      6 import os
      7 import unittest
      8 
      9 from telemetry.core import discover
     10 from telemetry.internal.browser import browser_credentials
     11 from telemetry import page
     12 from telemetry import story as story_module
     13 from telemetry.wpr import archive_info
     14 
     15 
     16 class StorySetSmokeTest(unittest.TestCase):
     17 
     18   def setUp(self):
     19     # Make sure the added failure message is appended to the default failure
     20     # message.
     21     self.longMessage = True
     22 
     23   def GetAllStorySetClasses(self, story_sets_dir, top_level_dir):
     24     # We can't test page sets that aren't directly constructible since we
     25     # don't know what arguments to put for the constructor.
     26     return discover.DiscoverClasses(story_sets_dir, top_level_dir,
     27                                     story_module.StorySet,
     28                                     directly_constructable=True).values()
     29 
     30   def CheckArchive(self, story_set):
     31     """Verify that all URLs of pages in story_set have an associated archive."""
     32     # TODO: Eventually these should be fatal.
     33     if not story_set.archive_data_file:
     34       logging.warning('Skipping %s: no archive data file', story_set.file_path)
     35       return
     36 
     37     logging.info('Testing %s', story_set.file_path)
     38 
     39     archive_data_file_path = os.path.join(story_set.base_dir,
     40                                           story_set.archive_data_file)
     41     self.assertTrue(os.path.exists(archive_data_file_path),
     42                     msg='Archive data file not found for %s' %
     43                     story_set.file_path)
     44 
     45     wpr_archive_info = archive_info.WprArchiveInfo.FromFile(
     46         archive_data_file_path, story_set.bucket)
     47     for story in story_set.stories:
     48       if isinstance(story, page.Page) and story.url.startswith('http'):
     49         self.assertTrue(wpr_archive_info.WprFilePathForStory(story),
     50                         msg='No archive found for %s in %s' % (
     51                             story.url, story_set.archive_data_file))
     52 
     53   def CheckCredentials(self, story_set):
     54     """Verify that all pages in story_set use proper credentials"""
     55     for story in story_set.stories:
     56       if not isinstance(story, page.Page):
     57         continue
     58       credentials = browser_credentials.BrowserCredentials()
     59       if story.credentials_path:
     60         credentials.credentials_path = (
     61             os.path.join(story.base_dir, story.credentials_path))
     62       fail_message = ('page %s of %s has invalid credentials %s' %
     63                       (story.url, story_set.file_path, story.credentials))
     64       if story.credentials:
     65         try:
     66           self.assertTrue(credentials.CanLogin(story.credentials), fail_message)
     67         except browser_credentials.CredentialsError:
     68           self.fail(fail_message)
     69 
     70   def CheckAttributes(self, story_set):
     71     """Verify that story_set and its stories base attributes have the right
     72        types.
     73     """
     74     self.CheckAttributesOfStorySetBasicAttributes(story_set)
     75     for story in story_set.stories:
     76       self.CheckAttributesOfStoryBasicAttributes(story)
     77 
     78   def CheckAttributesOfStorySetBasicAttributes(self, story_set):
     79     if story_set.base_dir is not None:
     80       self.assertTrue(
     81           isinstance(story_set.base_dir, str),
     82           msg='story_set %\'s base_dir must have type string')
     83 
     84     self.assertTrue(
     85         isinstance(story_set.archive_data_file, str),
     86         msg='story_set\'s archive_data_file path must have type string')
     87 
     88   def CheckAttributesOfStoryBasicAttributes(self, story):
     89     self.assertTrue(not hasattr(story, 'disabled'))
     90     self.assertTrue(
     91        isinstance(story.name, str),
     92        msg='story %s \'s name field must have type string' % story.display_name)
     93     self.assertTrue(
     94        isinstance(story.labels, set),
     95        msg='story %s \'s labels field must have type set' % story.display_name)
     96     for l in story.labels:
     97       self.assertTrue(
     98          isinstance(l, str),
     99          msg='label %s in story %s \'s labels must have type string'
    100          % (str(l), story.display_name))
    101     if not isinstance(story, page.Page):
    102       return
    103     self.assertTrue(
    104        # We use basestring instead of str because story's URL can be string of
    105        # unicode.
    106        isinstance(story.url, basestring),
    107        msg='page %s \'s url must have type string' % story.display_name)
    108     self.assertTrue(
    109         isinstance(story.startup_url, str),
    110         msg=('page %s \'s startup_url field must have type string'
    111             % story.display_name))
    112     self.assertIsInstance(
    113         story.make_javascript_deterministic, bool,
    114         msg='page %s \'s make_javascript_deterministic must have type bool'
    115             % story.display_name)
    116 
    117   def CheckSharedStates(self, story_set):
    118     if not story_set.allow_mixed_story_states:
    119       shared_state_class = (
    120           story_set.stories[0].shared_state_class)
    121       for story in story_set:
    122         self.assertIs(
    123             shared_state_class,
    124             story.shared_state_class,
    125             msg='story %s\'s shared_state_class field is different '
    126             'from other story\'s shared_state_class whereas '
    127             'story set %s disallows having mixed states' %
    128             (story, story_set))
    129 
    130   def RunSmokeTest(self, story_sets_dir, top_level_dir):
    131     """Run smoke test on all story sets in story_sets_dir.
    132 
    133     Subclass of StorySetSmokeTest is supposed to call this in some test
    134     method to run smoke test.
    135     """
    136     story_sets = self.GetAllStorySetClasses(story_sets_dir, top_level_dir)
    137     for story_set_class in story_sets:
    138       story_set = story_set_class()
    139       logging.info('Testing %s', story_set.file_path)
    140       self.CheckArchive(story_set)
    141       self.CheckCredentials(story_set)
    142       self.CheckAttributes(story_set)
    143       self.CheckSharedStates(story_set)
    144