1 # Copyright (c) 2012 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.
5 """Presubmit script for Chromium JS resources.
7 See chrome/browser/PRESUBMIT.py
13 class JSChecker(object):
14 def __init__(self
, input_api
, output_api
, file_filter
=None):
15 self
.input_api
= input_api
16 self
.output_api
= output_api
17 self
.file_filter
= file_filter
19 def RegexCheck(self
, line_number
, line
, regex
, message
):
20 return regex_check
.RegexCheck(
21 self
.input_api
.re
, line_number
, line
, regex
, message
)
23 def ChromeSendCheck(self
, i
, line
):
24 """Checks for a particular misuse of 'chrome.send'."""
25 return self
.RegexCheck(i
, line
, r
"chrome\.send\('[^']+'\s*(, \[\])\)",
26 'Passing an empty array to chrome.send is unnecessary')
28 def ConstCheck(self
, i
, line
):
29 """Check for use of the 'const' keyword."""
30 if self
.input_api
.re
.search(r
'\*\s+@const', line
):
31 # Probably a JsDoc line
34 return self
.RegexCheck(i
, line
, r
'(?:^|\s|\()(const)\s',
35 'Use /** @const */ var varName; instead of const varName;')
37 def EndJsDocCommentCheck(self
, i
, line
):
38 msg
= 'End JSDoc comments with */ instead of **/'
40 return self
.RegexCheck(i
, line
, regex
, msg
)
41 return _check(r
'^\s*(\*\*/)\s*$') or _check(r
'/\*\* @[a-zA-Z]+.* (\*\*/)')
43 def ExtraDotInGenericCheck(self
, i
, line
):
44 return self
.RegexCheck(i
, line
, r
"((?:Array|Object|Promise)\.<)",
45 "Don't use a dot after generics (Object.<T> should be Object<T>).")
47 def GetElementByIdCheck(self
, i
, line
):
48 """Checks for use of 'document.getElementById' instead of '$'."""
49 return self
.RegexCheck(i
, line
, r
"(document\.getElementById)\('",
50 "Use $('id'), from chrome://resources/js/util.js, instead of "
51 "document.getElementById('id')")
53 def InheritDocCheck(self
, i
, line
):
54 """Checks for use of '@inheritDoc' instead of '@override'."""
55 return self
.RegexCheck(i
, line
, r
"\* (@inheritDoc)",
56 "@inheritDoc is deprecated, use @override instead")
58 def WrapperTypeCheck(self
, i
, line
):
59 """Check for wrappers (new String()) instead of builtins (string)."""
60 return self
.RegexCheck(i
, line
,
61 r
"(?:/\*)?\*.*?@(?:param|return|type) ?" # /** @param/@return/@type
62 r
"{[^}]*\b(String|Boolean|Number)\b[^}]*}", # {(Boolean|Number|String)}
63 "Don't use wrapper types (i.e. new String() or @type {String})")
65 def VarNameCheck(self
, i
, line
):
66 """See the style guide. http://goo.gl/uKir6"""
67 return self
.RegexCheck(i
, line
,
68 r
"var (?!g_\w+)([a-z]*[_$][\w_$]*)(?<! \$)",
69 "Please use var namesLikeThis <http://goo.gl/uKir6>")
71 def _GetErrorHighlight(self
, start
, length
):
72 """Takes a start position and a length, and produces a row of '^'s to
73 highlight the corresponding part of a string.
75 return start
* ' ' + length
* '^'
77 def _MakeErrorOrWarning(self
, error_text
, filename
):
78 """Takes a few lines of text indicating a style violation and turns it into
79 a PresubmitError (if |filename| is in a directory where we've already
80 taken out all the style guide violations) or a PresubmitPromptWarning
81 (if it's in a directory where we haven't done that yet).
83 # TODO(tbreisacher): Once we've cleaned up the style nits in all of
84 # resources/ we can get rid of this function.
85 path
= self
.input_api
.os_path
86 resources
= path
.join(self
.input_api
.PresubmitLocalPath(), 'resources')
88 path
.join(resources
, 'bookmark_manager'),
89 path
.join(resources
, 'extensions'),
90 path
.join(resources
, 'file_manager'),
91 path
.join(resources
, 'help'),
92 path
.join(resources
, 'history'),
93 path
.join(resources
, 'memory_internals'),
94 path
.join(resources
, 'net_export'),
95 path
.join(resources
, 'net_internals'),
96 path
.join(resources
, 'network_action_predictor'),
97 path
.join(resources
, 'ntp4'),
98 path
.join(resources
, 'options'),
99 path
.join(resources
, 'password_manager_internals'),
100 path
.join(resources
, 'print_preview'),
101 path
.join(resources
, 'profiler'),
102 path
.join(resources
, 'sync_promo'),
103 path
.join(resources
, 'tracing'),
104 path
.join(resources
, 'uber'),
106 if filename
.startswith(dirs
):
107 return self
.output_api
.PresubmitError(error_text
)
109 return self
.output_api
.PresubmitPromptWarning(error_text
)
111 def ClosureLint(self
, file_to_lint
, source
=None):
112 """Lints |file_to_lint| and returns the errors."""
117 old_filters
= warnings
.filters
120 closure_linter_path
= self
.input_api
.os_path
.join(
121 self
.input_api
.change
.RepositoryRoot(),
124 gflags_path
= self
.input_api
.os_path
.join(
125 self
.input_api
.change
.RepositoryRoot(),
129 sys
.path
.insert(0, closure_linter_path
)
130 sys
.path
.insert(0, gflags_path
)
132 warnings
.filterwarnings('ignore', category
=DeprecationWarning)
134 from closure_linter
import errors
, runner
135 from closure_linter
.common
import errorhandler
140 warnings
.filters
= old_filters
142 class ErrorHandlerImpl(errorhandler
.ErrorHandler
):
143 """Filters out errors that don't apply to Chromium JavaScript code."""
145 def __init__(self
, re
):
149 def HandleFile(self
, filename
, first_token
):
150 self
._filename
= filename
152 def HandleError(self
, error
):
153 if (self
._valid
(error
)):
154 error
.filename
= self
._filename
155 self
._errors
.append(error
)
161 return bool(self
._errors
)
163 def _valid(self
, error
):
164 """Check whether an error is valid. Most errors are valid, with a few
165 exceptions which are listed here.
168 is_grit_statement
= bool(
169 self
.re
.search("</?(include|if)", error
.token
.line
))
171 # Ignore missing spaces before "(" until Promise#catch issue is solved.
172 # http://crbug.com/338301
173 if (error
.code
== errors
.MISSING_SPACE
and error
.token
.string
== '(' and
174 'catch(' in error
.token
.line
):
177 # Ignore "}.bind(" errors. http://crbug.com/397697
178 if (error
.code
== errors
.MISSING_SEMICOLON_AFTER_FUNCTION
and
179 '}.bind(' in error
.token
.line
):
182 return not is_grit_statement
and error
.code
not in [
183 errors
.COMMA_AT_END_OF_LITERAL
,
184 errors
.JSDOC_ILLEGAL_QUESTION_WITH_PIPE
,
185 errors
.LINE_TOO_LONG
,
186 errors
.MISSING_JSDOC_TAG_THIS
,
189 # Whitelist Polymer-specific JsDoc tags.
190 gflags
.FLAGS
.custom_jsdoc_tags
= ('group', 'element', 'attribute',
191 'default', 'polymerBehavior')
192 error_handler
= ErrorHandlerImpl(self
.input_api
.re
)
193 runner
.Run(file_to_lint
, error_handler
, source
=source
)
194 return error_handler
.GetErrors()
197 """Check for violations of the Chromium JavaScript style guide. See
198 http://chromium.org/developers/web-development-style-guide#TOC-JavaScript
202 affected_files
= self
.input_api
.change
.AffectedFiles(
203 file_filter
=self
.file_filter
,
204 include_deletes
=False)
205 affected_js_files
= filter(lambda f
: f
.LocalPath().endswith('.js'),
207 for f
in affected_js_files
:
210 # Check for the following:
211 # * document.getElementById()
212 # * the 'const' keyword
213 # * Passing an empty array to 'chrome.send()'
214 for i
, line
in enumerate(f
.NewContents(), start
=1):
215 error_lines
+= filter(None, [
216 self
.ChromeSendCheck(i
, line
),
217 self
.ConstCheck(i
, line
),
218 self
.GetElementByIdCheck(i
, line
),
219 self
.EndJsDocCommentCheck(i
, line
),
220 self
.ExtraDotInGenericCheck(i
, line
),
221 self
.InheritDocCheck(i
, line
),
222 self
.WrapperTypeCheck(i
, line
),
223 self
.VarNameCheck(i
, line
),
226 # Use closure linter to check for several different errors.
227 lint_errors
= self
.ClosureLint(self
.input_api
.os_path
.join(
228 self
.input_api
.change
.RepositoryRoot(), f
.LocalPath()))
230 for error
in lint_errors
:
231 highlight
= self
._GetErrorHighlight
(
232 error
.token
.start_index
, error
.token
.length
)
233 error_msg
= ' line %d: E%04d: %s\n%s\n%s' % (
234 error
.token
.line_number
,
237 error
.token
.line
.rstrip(),
239 error_lines
.append(error_msg
)
243 'Found JavaScript style violations in %s:' %
244 f
.LocalPath()] + error_lines
245 results
.append(self
._MakeErrorOrWarning
(
246 '\n'.join(error_lines
), f
.AbsoluteLocalPath()))
249 results
.append(self
.output_api
.PresubmitNotifyResult(
250 'See the JavaScript style guide at '
251 'http://www.chromium.org/developers/web-development-style-guide'