1# Copyright (c) 2013 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
6"""Top-level presubmit script for Skia.
7
8See http://dev.chromium.org/developers/how-tos/depottools/presubmit-scripts
9for more details about the presubmit API built into gcl.
10"""
11
12import collections
13import csv
14import fnmatch
15import os
16import re
17import subprocess
18import sys
19import traceback
20
21
22REVERT_CL_SUBJECT_PREFIX = 'Revert '
23
24SKIA_TREE_STATUS_URL = 'http://skia-tree-status.appspot.com'
25
26# Please add the complete email address here (and not just 'xyz@' or 'xyz').
27PUBLIC_API_OWNERS = (
28    'reed@chromium.org',
29    'reed@google.com',
30    'bsalomon@chromium.org',
31    'bsalomon@google.com',
32    'djsollen@chromium.org',
33    'djsollen@google.com',
34    'hcm@chromium.org',
35    'hcm@google.com',
36)
37
38AUTHORS_FILE_NAME = 'AUTHORS'
39
40DOCS_PREVIEW_URL = 'https://skia.org/?cl='
41GOLD_TRYBOT_URL = 'https://gold.skia.org/search?issue='
42
43SERVICE_ACCOUNT_SUFFIX = [
44    '@%s.iam.gserviceaccount.com' % project for project in [
45        'skia-buildbots.google.com', 'skia-swarming-bots', 'skia-public',
46        'skia-corp.google.com']]
47
48
49def _CheckChangeHasEol(input_api, output_api, source_file_filter=None):
50  """Checks that files end with atleast one \n (LF)."""
51  eof_files = []
52  for f in input_api.AffectedSourceFiles(source_file_filter):
53    contents = input_api.ReadFile(f, 'rb')
54    # Check that the file ends in atleast one newline character.
55    if len(contents) > 1 and contents[-1:] != '\n':
56      eof_files.append(f.LocalPath())
57
58  if eof_files:
59    return [output_api.PresubmitPromptWarning(
60      'These files should end in a newline character:',
61      items=eof_files)]
62  return []
63
64
65def _JsonChecks(input_api, output_api):
66  """Run checks on any modified json files."""
67  failing_files = []
68  for affected_file in input_api.AffectedFiles(None):
69    affected_file_path = affected_file.LocalPath()
70    is_json = affected_file_path.endswith('.json')
71    is_metadata = (affected_file_path.startswith('site/') and
72                   affected_file_path.endswith('/METADATA'))
73    if is_json or is_metadata:
74      try:
75        input_api.json.load(open(affected_file_path, 'r'))
76      except ValueError:
77        failing_files.append(affected_file_path)
78
79  results = []
80  if failing_files:
81    results.append(
82        output_api.PresubmitError(
83            'The following files contain invalid json:\n%s\n\n' %
84                '\n'.join(failing_files)))
85  return results
86
87
88def _IfDefChecks(input_api, output_api):
89  """Ensures if/ifdef are not before includes. See skbug/3362 for details."""
90  comment_block_start_pattern = re.compile('^\s*\/\*.*$')
91  comment_block_middle_pattern = re.compile('^\s+\*.*')
92  comment_block_end_pattern = re.compile('^\s+\*\/.*$')
93  single_line_comment_pattern = re.compile('^\s*//.*$')
94  def is_comment(line):
95    return (comment_block_start_pattern.match(line) or
96            comment_block_middle_pattern.match(line) or
97            comment_block_end_pattern.match(line) or
98            single_line_comment_pattern.match(line))
99
100  empty_line_pattern = re.compile('^\s*$')
101  def is_empty_line(line):
102    return empty_line_pattern.match(line)
103
104  failing_files = []
105  for affected_file in input_api.AffectedSourceFiles(None):
106    affected_file_path = affected_file.LocalPath()
107    if affected_file_path.endswith('.cpp') or affected_file_path.endswith('.h'):
108      f = open(affected_file_path)
109      for line in f.xreadlines():
110        if is_comment(line) or is_empty_line(line):
111          continue
112        # The below will be the first real line after comments and newlines.
113        if line.startswith('#if 0 '):
114          pass
115        elif line.startswith('#if ') or line.startswith('#ifdef '):
116          failing_files.append(affected_file_path)
117        break
118
119  results = []
120  if failing_files:
121    results.append(
122        output_api.PresubmitError(
123            'The following files have #if or #ifdef before includes:\n%s\n\n'
124            'See https://bug.skia.org/3362 for why this should be fixed.' %
125                '\n'.join(failing_files)))
126  return results
127
128
129def _CopyrightChecks(input_api, output_api, source_file_filter=None):
130  results = []
131  year_pattern = r'\d{4}'
132  year_range_pattern = r'%s(-%s)?' % (year_pattern, year_pattern)
133  years_pattern = r'%s(,%s)*,?' % (year_range_pattern, year_range_pattern)
134  copyright_pattern = (
135      r'Copyright (\([cC]\) )?%s \w+' % years_pattern)
136
137  for affected_file in input_api.AffectedSourceFiles(source_file_filter):
138    if 'third_party' in affected_file.LocalPath():
139      continue
140    contents = input_api.ReadFile(affected_file, 'rb')
141    if not re.search(copyright_pattern, contents):
142      results.append(output_api.PresubmitError(
143          '%s is missing a correct copyright header.' % affected_file))
144  return results
145
146
147def _ToolFlags(input_api, output_api):
148  """Make sure `{dm,nanobench}_flags.py test` passes if modified."""
149  results = []
150  sources = lambda x: ('dm_flags.py'        in x.LocalPath() or
151                       'nanobench_flags.py' in x.LocalPath())
152  for f in input_api.AffectedSourceFiles(sources):
153    if 0 != subprocess.call(['python', f.LocalPath(), 'test']):
154      results.append(output_api.PresubmitError('`python %s test` failed' % f))
155  return results
156
157
158def _InfraTests(input_api, output_api):
159  """Run the infra tests."""
160  results = []
161  if not any(f.LocalPath().startswith('infra')
162             for f in input_api.AffectedFiles()):
163    return results
164
165  cmd = ['python', os.path.join('infra', 'bots', 'infra_tests.py')]
166  try:
167    subprocess.check_output(cmd)
168  except subprocess.CalledProcessError as e:
169    results.append(output_api.PresubmitError(
170        '`%s` failed:\n%s' % (' '.join(cmd), e.output)))
171  return results
172
173
174def _CheckGNFormatted(input_api, output_api):
175  """Make sure any .gn files we're changing have been formatted."""
176  results = []
177  for f in input_api.AffectedFiles():
178    if (not f.LocalPath().endswith('.gn') and
179        not f.LocalPath().endswith('.gni')):
180      continue
181
182    gn = 'gn.bat' if 'win32' in sys.platform else 'gn'
183    cmd = [gn, 'format', '--dry-run', f.LocalPath()]
184    try:
185      subprocess.check_output(cmd)
186    except subprocess.CalledProcessError:
187      fix = 'gn format ' + f.LocalPath()
188      results.append(output_api.PresubmitError(
189          '`%s` failed, try\n\t%s' % (' '.join(cmd), fix)))
190  return results
191
192
193class _WarningsAsErrors():
194  def __init__(self, output_api):
195    self.output_api = output_api
196    self.old_warning = None
197  def __enter__(self):
198    self.old_warning = self.output_api.PresubmitPromptWarning
199    self.output_api.PresubmitPromptWarning = self.output_api.PresubmitError
200    return self.output_api
201  def __exit__(self, ex_type, ex_value, ex_traceback):
202    self.output_api.PresubmitPromptWarning = self.old_warning
203
204
205def _CommonChecks(input_api, output_api):
206  """Presubmit checks common to upload and commit."""
207  results = []
208  sources = lambda x: (x.LocalPath().endswith('.h') or
209                       x.LocalPath().endswith('.py') or
210                       x.LocalPath().endswith('.sh') or
211                       x.LocalPath().endswith('.m') or
212                       x.LocalPath().endswith('.mm') or
213                       x.LocalPath().endswith('.go') or
214                       x.LocalPath().endswith('.c') or
215                       x.LocalPath().endswith('.cc') or
216                       x.LocalPath().endswith('.cpp'))
217  results.extend(_CheckChangeHasEol(
218      input_api, output_api, source_file_filter=sources))
219  with _WarningsAsErrors(output_api):
220    results.extend(input_api.canned_checks.CheckChangeHasNoCR(
221        input_api, output_api, source_file_filter=sources))
222    results.extend(input_api.canned_checks.CheckChangeHasNoStrayWhitespace(
223        input_api, output_api, source_file_filter=sources))
224  results.extend(_JsonChecks(input_api, output_api))
225  results.extend(_IfDefChecks(input_api, output_api))
226  results.extend(_CopyrightChecks(input_api, output_api,
227                                  source_file_filter=sources))
228  results.extend(_ToolFlags(input_api, output_api))
229  return results
230
231
232def CheckChangeOnUpload(input_api, output_api):
233  """Presubmit checks for the change on upload.
234
235  The following are the presubmit checks:
236  * Check change has one and only one EOL.
237  """
238  results = []
239  results.extend(_CommonChecks(input_api, output_api))
240  # Run on upload, not commit, since the presubmit bot apparently doesn't have
241  # coverage or Go installed.
242  results.extend(_InfraTests(input_api, output_api))
243
244  results.extend(_CheckGNFormatted(input_api, output_api))
245  return results
246
247
248def _CheckTreeStatus(input_api, output_api, json_url):
249  """Check whether to allow commit.
250
251  Args:
252    input_api: input related apis.
253    output_api: output related apis.
254    json_url: url to download json style status.
255  """
256  tree_status_results = input_api.canned_checks.CheckTreeIsOpen(
257      input_api, output_api, json_url=json_url)
258  if not tree_status_results:
259    # Check for caution state only if tree is not closed.
260    connection = input_api.urllib2.urlopen(json_url)
261    status = input_api.json.loads(connection.read())
262    connection.close()
263    if ('caution' in status['message'].lower() and
264        os.isatty(sys.stdout.fileno())):
265      # Display a prompt only if we are in an interactive shell. Without this
266      # check the commit queue behaves incorrectly because it considers
267      # prompts to be failures.
268      short_text = 'Tree state is: ' + status['general_state']
269      long_text = status['message'] + '\n' + json_url
270      tree_status_results.append(
271          output_api.PresubmitPromptWarning(
272              message=short_text, long_text=long_text))
273  else:
274    # Tree status is closed. Put in message about contacting sheriff.
275    connection = input_api.urllib2.urlopen(
276        SKIA_TREE_STATUS_URL + '/current-sheriff')
277    sheriff_details = input_api.json.loads(connection.read())
278    if sheriff_details:
279      tree_status_results[0]._message += (
280          '\n\nPlease contact the current Skia sheriff (%s) if you are trying '
281          'to submit a build fix\nand do not know how to submit because the '
282          'tree is closed') % sheriff_details['username']
283  return tree_status_results
284
285
286class CodeReview(object):
287  """Abstracts which codereview tool is used for the specified issue."""
288
289  def __init__(self, input_api):
290    self._issue = input_api.change.issue
291    self._gerrit = input_api.gerrit
292
293  def GetOwnerEmail(self):
294    return self._gerrit.GetChangeOwner(self._issue)
295
296  def GetSubject(self):
297    return self._gerrit.GetChangeInfo(self._issue)['subject']
298
299  def GetDescription(self):
300    return self._gerrit.GetChangeDescription(self._issue)
301
302  def IsDryRun(self):
303    return self._gerrit.GetChangeInfo(
304        self._issue)['labels']['Commit-Queue'].get('value', 0) == 1
305
306  def GetReviewers(self):
307    code_review_label = (
308        self._gerrit.GetChangeInfo(self._issue)['labels']['Code-Review'])
309    return [r['email'] for r in code_review_label.get('all', [])]
310
311  def GetApprovers(self):
312    approvers = []
313    code_review_label = (
314        self._gerrit.GetChangeInfo(self._issue)['labels']['Code-Review'])
315    for m in code_review_label.get('all', []):
316      if m.get("value") == 1:
317        approvers.append(m["email"])
318    return approvers
319
320
321def _CheckOwnerIsInAuthorsFile(input_api, output_api):
322  results = []
323  if input_api.change.issue:
324    cr = CodeReview(input_api)
325
326    owner_email = cr.GetOwnerEmail()
327
328    # Service accounts don't need to be in AUTHORS.
329    for suffix in SERVICE_ACCOUNT_SUFFIX:
330      if owner_email.endswith(suffix):
331        return results
332
333    try:
334      authors_content = ''
335      for line in open(AUTHORS_FILE_NAME):
336        if not line.startswith('#'):
337          authors_content += line
338      email_fnmatches = re.findall('<(.*)>', authors_content)
339      for email_fnmatch in email_fnmatches:
340        if fnmatch.fnmatch(owner_email, email_fnmatch):
341          # Found a match, the user is in the AUTHORS file break out of the loop
342          break
343      else:
344        results.append(
345          output_api.PresubmitError(
346            'The email %s is not in Skia\'s AUTHORS file.\n'
347            'Issue owner, this CL must include an addition to the Skia AUTHORS '
348            'file.'
349            % owner_email))
350    except IOError:
351      # Do not fail if authors file cannot be found.
352      traceback.print_exc()
353      input_api.logging.error('AUTHORS file not found!')
354
355  return results
356
357
358def _CheckLGTMsForPublicAPI(input_api, output_api):
359  """Check LGTMs for public API changes.
360
361  For public API files make sure there is an LGTM from the list of owners in
362  PUBLIC_API_OWNERS.
363  """
364  results = []
365  requires_owner_check = False
366  for affected_file in input_api.AffectedFiles():
367    affected_file_path = affected_file.LocalPath()
368    file_path, file_ext = os.path.splitext(affected_file_path)
369    # We only care about files that end in .h and are under the top-level
370    # include dir, but not include/private.
371    if (file_ext == '.h' and
372        'include' == file_path.split(os.path.sep)[0] and
373        'private' not in file_path):
374      requires_owner_check = True
375
376  if not requires_owner_check:
377    return results
378
379  lgtm_from_owner = False
380  if input_api.change.issue:
381    cr = CodeReview(input_api)
382
383    if re.match(REVERT_CL_SUBJECT_PREFIX, cr.GetSubject(), re.I):
384      # It is a revert CL, ignore the public api owners check.
385      return results
386
387    if cr.IsDryRun():
388      # Ignore public api owners check for dry run CLs since they are not
389      # going to be committed.
390      return results
391
392    if input_api.gerrit:
393      for reviewer in cr.GetReviewers():
394        if reviewer in PUBLIC_API_OWNERS:
395          # If an owner is specified as an reviewer in Gerrit then ignore the
396          # public api owners check.
397          return results
398    else:
399      match = re.search(r'^TBR=(.*)$', cr.GetDescription(), re.M)
400      if match:
401        tbr_section = match.group(1).strip().split(' ')[0]
402        tbr_entries = tbr_section.split(',')
403        for owner in PUBLIC_API_OWNERS:
404          if owner in tbr_entries or owner.split('@')[0] in tbr_entries:
405            # If an owner is specified in the TBR= line then ignore the public
406            # api owners check.
407            return results
408
409    if cr.GetOwnerEmail() in PUBLIC_API_OWNERS:
410      # An owner created the CL that is an automatic LGTM.
411      lgtm_from_owner = True
412
413    for approver in cr.GetApprovers():
414      if approver in PUBLIC_API_OWNERS:
415        # Found an lgtm in a message from an owner.
416        lgtm_from_owner = True
417        break
418
419  if not lgtm_from_owner:
420    results.append(
421        output_api.PresubmitError(
422            "If this CL adds to or changes Skia's public API, you need an LGTM "
423            "from any of %s.  If this CL only removes from or doesn't change "
424            "Skia's public API, please add a short note to the CL saying so. "
425            "Add one of the owners as a reviewer to your CL as well as to the "
426            "TBR= line.  If you don't know if this CL affects Skia's public "
427            "API, treat it like it does." % str(PUBLIC_API_OWNERS)))
428  return results
429
430
431def _FooterExists(footers, key, value):
432  for k, v in footers:
433    if k == key and v == value:
434      return True
435  return False
436
437
438def PostUploadHook(cl, change, output_api):
439  """git cl upload will call this hook after the issue is created/modified.
440
441  This hook does the following:
442  * Adds a link to preview docs changes if there are any docs changes in the CL.
443  * Adds 'No-Try: true' if the CL contains only docs changes.
444  """
445
446  results = []
447  atleast_one_docs_change = False
448  all_docs_changes = True
449  for affected_file in change.AffectedFiles():
450    affected_file_path = affected_file.LocalPath()
451    file_path, _ = os.path.splitext(affected_file_path)
452    if 'site' == file_path.split(os.path.sep)[0]:
453      atleast_one_docs_change = True
454    else:
455      all_docs_changes = False
456    if atleast_one_docs_change and not all_docs_changes:
457      break
458
459  issue = cl.issue
460  if issue:
461    # Skip PostUploadHooks for all auto-commit service account bots. New
462    # patchsets (caused due to PostUploadHooks) invalidates the CQ+2 vote from
463    # the "--use-commit-queue" flag to "git cl upload".
464    for suffix in SERVICE_ACCOUNT_SUFFIX:
465      if cl.GetIssueOwner().endswith(suffix):
466        return results
467
468    original_description_lines, footers = cl.GetDescriptionFooters()
469    new_description_lines = list(original_description_lines)
470
471    # If the change includes only doc changes then add No-Try: true in the
472    # CL's description if it does not exist yet.
473    if all_docs_changes and not _FooterExists(footers, 'No-Try', 'true'):
474      new_description_lines.append('No-Try: true')
475      results.append(
476          output_api.PresubmitNotifyResult(
477              'This change has only doc changes. Automatically added '
478              '\'No-Try: true\' to the CL\'s description'))
479
480    # If there is atleast one docs change then add preview link in the CL's
481    # description if it does not already exist there.
482    docs_preview_link = '%s%s' % (DOCS_PREVIEW_URL, issue)
483    docs_preview_line = 'Docs-Preview: %s' % docs_preview_link
484    if (atleast_one_docs_change and
485        not _FooterExists(footers, 'Docs-Preview', docs_preview_link)):
486      # Automatically add a link to where the docs can be previewed.
487      new_description_lines.append(docs_preview_line)
488      results.append(
489          output_api.PresubmitNotifyResult(
490              'Automatically added a link to preview the docs changes to the '
491              'CL\'s description'))
492
493    # If the description has changed update it.
494    if new_description_lines != original_description_lines:
495      # Add a new line separating the new contents from the old contents.
496      new_description_lines.insert(len(original_description_lines), '')
497      cl.UpdateDescriptionFooters(new_description_lines, footers)
498
499    return results
500
501
502def CheckChangeOnCommit(input_api, output_api):
503  """Presubmit checks for the change on commit.
504
505  The following are the presubmit checks:
506  * Check change has one and only one EOL.
507  * Ensures that the Skia tree is open in
508    http://skia-tree-status.appspot.com/. Shows a warning if it is in 'Caution'
509    state and an error if it is in 'Closed' state.
510  """
511  results = []
512  results.extend(_CommonChecks(input_api, output_api))
513  results.extend(
514      _CheckTreeStatus(input_api, output_api, json_url=(
515          SKIA_TREE_STATUS_URL + '/banner-status?format=json')))
516  results.extend(_CheckLGTMsForPublicAPI(input_api, output_api))
517  results.extend(_CheckOwnerIsInAuthorsFile(input_api, output_api))
518  # Checks for the presence of 'DO NOT''SUBMIT' in CL description and in
519  # content of files.
520  results.extend(
521      input_api.canned_checks.CheckDoNotSubmit(input_api, output_api))
522  return results
523