# Copyright 2015 The Chromium Authors. All rights reserved. # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. """Checks to use in PRESUBMIT.py for HTML style violations.""" import collections import difflib import re import bs4 from catapult_build import parse_html def RunChecks(input_api, output_api, excluded_paths=None): def ShouldCheck(affected_file): path = affected_file.LocalPath() if not path.endswith('.html'): return False if not excluded_paths: return True return not any(re.match(pattern, path) for pattern in excluded_paths) affected_files = input_api.AffectedFiles( file_filter=ShouldCheck, include_deletes=False) results = [] for f in affected_files: CheckAffectedFile(f, results, output_api) return results def CheckAffectedFile(affected_file, results, output_api): path = affected_file.LocalPath() soup = parse_html.BeautifulSoup('\n'.join(affected_file.NewContents())) for check in [CheckDoctype, CheckImportOrder]: check(path, soup, results, output_api) def CheckDoctype(path, soup, results, output_api): if _HasHtml5Declaration(soup): return error_text = 'Could not find "<!DOCTYPE html>" in %s.' % path results.append(output_api.PresubmitError(error_text)) def _HasHtml5Declaration(soup): for item in soup.contents: if isinstance(item, bs4.Doctype) and item.lower() == 'html': return True return False def CheckImportOrder(path, soup, results, output_api): grouped_hrefs = collections.defaultdict(list) # Link rel -> [link hrefs]. for link in soup.find_all('link'): if link.get('data-suppress-import-order') is not None: continue grouped_hrefs[','.join(link.get('rel'))].append(link.get('href')) for rel, actual_hrefs in grouped_hrefs.iteritems(): expected_hrefs = list(sorted(set(actual_hrefs))) if actual_hrefs != expected_hrefs: error_text = ( 'Invalid "%s" link sort order in %s:\n' % (rel, path) + ' ' + '\n '.join(difflib.ndiff(actual_hrefs, expected_hrefs)) + '\nIf this error is invalid, you can suppress it by adding a ' + '"data-suppress-import-order" attribute to the out-of-order <link> ' + 'element.') results.append(output_api.PresubmitError(error_text))