forked from crosswalk-project/crosswalk
-
Notifications
You must be signed in to change notification settings - Fork 0
/
lint.py
executable file
·244 lines (220 loc) · 8.19 KB
/
lint.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
#!/usr/bin/env python
# Copyright (c) 2013 Intel Corporation. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
''' This script is used to lint changeset before code checkin
It get the changeset from the repo and base specified by
command line arguments. And run cpplint over the changeset.
'''
# TODO(wang16): Only show error for the lines do changed in the changeset
import os
import re
import sys
from utils import GitExe, GetCommandOutput, TryAddDepotToolsToPythonPath
PYTHON_EXTS = ['.py']
def find_depot_tools_in_path():
paths = os.getenv('PATH').split(os.path.pathsep)
for path in paths:
if os.path.basename(path) == 'depot_tools':
return path
return None
def repo_is_dirty():
return GetCommandOutput([GitExe(), 'diff', 'HEAD']).strip() != ''
def get_tracking_remote():
branch = [GitExe(), 'branch', '-vv', '-a']
# The output of git branch -vv will be in format
# * <branch> <hash> [<remote>: ahead <n>, behind <m>] <subject>
# <branch> <hash> <subject>
output = GetCommandOutput(branch)
branches = output.split('\n')
for branch in branches:
# we only need active branch first.
if not branch.startswith('*'):
continue
detail = \
branch[1:].strip().split(' ', 1)[1].strip().split(' ', 1)[1].strip()
if detail.startswith('['):
remote = detail[1:].split(']', 1)[0]
remote = remote.split(':', 1)[0].strip()
# verify that remotes/branch or branch is a real branch
# There is still chance that developer named his commit
# as [origin/branch], in this case
exists = [\
r_branch for r_branch in branches \
if r_branch.strip().startswith('remotes/'+remote) or \
r_branch.strip().startswith(remote)]
if len(exists) == 0:
remote = ''
else:
remote = ''
break
if remote == '':
if repo_is_dirty():
remote = 'HEAD'
else:
remote = 'HEAD~'
print 'Base is not specified, '\
'will use %s as comparasion base for linting' % remote
return remote
# return pyfiles, others
def get_change_file_list(base):
diff = [GitExe(), 'diff', '--name-only', base]
output = GetCommandOutput(diff)
changes = [line.strip() for line in output.strip().split('\n')]
pyfiles = []
others = []
# pylint: disable=W0612
for change in changes:
root, ext = os.path.splitext(change)
if ext.lower() in PYTHON_EXTS:
pyfiles.append(change)
else:
others.append(change)
return pyfiles, others
def do_cpp_lint(changeset, repo, args):
# Try to import cpplint from depot_tools first
try:
import cpplint
except ImportError:
TryAddDepotToolsToPythonPath()
try:
import cpplint
import cpplint_chromium
import gcl
except ImportError:
sys.stderr.write("Can't find cpplint, please add your depot_tools "\
"to PATH or PYTHONPATH\n")
raise
origin_error = cpplint.Error
def MyError(filename, linenum, category, confidence, message):
# Skip no header guard error for MSVC generated files.
if (filename.endswith('resource.h')):
sys.stdout.write('Ignored Error:\n %s(%s): %s [%s] [%d]\n' % (
filename, linenum, message, category, confidence))
# Skip no header guard error for ipc messages definition,
# because they will be included multiple times for different macros.
elif (filename.endswith('messages.h') and linenum == 0 and
category == 'build/header_guard'):
sys.stdout.write('Ignored Error:\n %s(%s): %s [%s] [%d]\n' % (
filename, linenum, message, category, confidence))
else:
origin_error(filename, linenum, category, confidence, message)
cpplint.Error = MyError
origin_FileInfo = cpplint.FileInfo
class MyFileInfo(origin_FileInfo):
def RepositoryName(self):
''' Origin FileInfo find the first .git and take it as project root,
it's not the case for xwalk, header in xwalk should have guard
relevant to root dir of chromium project, which is one level
upper of the origin output of RepositoryName.
'''
repo_name = origin_FileInfo.RepositoryName(self)
if repo == "xwalk" and not repo_name.startswith('xwalk'):
return 'xwalk/%s' % repo_name
else:
return repo_name
cpplint.FileInfo = MyFileInfo
print '_____ do cpp lint'
if len(changeset) == 0:
print 'changeset is empty except python files'
return
# Following code is referencing depot_tools/gcl.py: CMDlint
# Process cpplints arguments if any.
filenames = cpplint.ParseArguments(args + changeset)
white_list = gcl.GetCodeReviewSetting("LINT_REGEX")
if not white_list:
white_list = gcl.DEFAULT_LINT_REGEX
white_regex = re.compile(white_list)
black_list = gcl.GetCodeReviewSetting("LINT_IGNORE_REGEX")
if not black_list:
black_list = gcl.DEFAULT_LINT_IGNORE_REGEX
black_regex = re.compile(black_list)
extra_check_functions = [cpplint_chromium.CheckPointerDeclarationWhitespace]
# pylint: disable=W0212
cpplint_state = cpplint._cpplint_state
for filename in filenames:
if white_regex.match(filename):
if black_regex.match(filename):
print "Ignoring file %s" % filename
else:
cpplint.ProcessFile(filename, cpplint_state.verbose_level,
extra_check_functions)
else:
print "Skipping file %s" % filename
print "Total errors found: %d\n" % cpplint_state.error_count
def do_py_lint(changeset):
print '_____ do python lint'
if sys.platform.startswith('win'):
pylint_cmd = ['pylint.bat']
else:
pylint_cmd = ['pylint']
_has_import_error = False
for pyfile in changeset:
py_dir, py_name = os.path.split(os.path.abspath(pyfile))
previous_cwd = os.getcwd()
os.chdir(py_dir)
print 'pylint %s' % pyfile
try:
output = GetCommandOutput(pylint_cmd + [py_name]).strip()
if len(output) > 0:
print output
except Exception, e:
if not _has_import_error and \
'F0401:' in [error[:6] for error in str(e).splitlines()]:
_has_import_error = True
print e
os.chdir(previous_cwd)
if _has_import_error:
print 'You have error for python importing, please check your PYTHONPATH'
def do_lint(repo, base, args):
# dir structure should be src/xwalk for xwalk
# src/third_party/WebKit for blink
# src/ for chromium
# lint.py should be located in src/xwalk/tools/lint.py
_lint_py = os.path.abspath(__file__)
_dirs = _lint_py.split(os.path.sep)
src_root = os.path.sep.join(_dirs[:len(_dirs)-3])
if repo == 'xwalk':
base_repo = os.path.join(src_root, 'xwalk')
elif repo == 'chromium':
base_repo = src_root
elif repo == 'blink':
base_repo = os.path.join(src_root, 'third_party', 'WebKit')
else:
raise NotImplementedError('repo must in xwalk, blink and chromium')
previous_cwd = os.getcwd()
os.chdir(base_repo)
if base == None:
base = get_tracking_remote()
changes_py, changes_others = get_change_file_list(base)
do_cpp_lint(changes_others, repo, args)
do_py_lint(changes_py)
os.chdir(previous_cwd)
return 1
from optparse import OptionParser, BadOptionError
class PassThroughOptionParser(OptionParser):
def _process_long_opt(self, rargs, values):
try:
OptionParser._process_long_opt(self, rargs, values)
except BadOptionError, err:
self.largs.append(err.opt_str)
def _process_short_opts(self, rargs, values):
try:
OptionParser._process_short_opts(self, rargs, values)
except BadOptionError, err:
self.largs.append(err.opt_str)
def main():
option_parser = PassThroughOptionParser()
option_parser.add_option('--repo', default='xwalk',
help='The repo to do lint, should be in [xwalk, blink, chromium]\
xwalk by default')
option_parser.add_option('--base', default=None,
help='The base point to get change set. If not specified,' +
' it will choose:\r\n' +
' 1. Active branch\'s tracking branch if exist\n' +
' 2. HEAD if current repo is dirty\n' +
' 3. HEAD~ elsewise')
options, args = option_parser.parse_args()
sys.exit(do_lint(options.repo, options.base, args))
if __name__ == '__main__':
main()