2015-03-26 08:00:20 +01:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
|
|
|
|
|
|
|
|
# Copyright 2014-2015 Florian Bruhin (The Compiler) <mail@qutebrowser.org>
|
|
|
|
#
|
|
|
|
# This file is part of qutebrowser.
|
|
|
|
#
|
|
|
|
# qutebrowser is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# qutebrowser is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with qutebrowser. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
"""Various small code checkers."""
|
|
|
|
|
|
|
|
import os
|
2015-03-31 22:12:38 +02:00
|
|
|
import re
|
2015-03-26 08:00:20 +01:00
|
|
|
import sys
|
|
|
|
import os.path
|
|
|
|
import argparse
|
|
|
|
import subprocess
|
|
|
|
import tokenize
|
|
|
|
import traceback
|
2015-03-31 22:12:38 +02:00
|
|
|
import collections
|
2015-03-26 08:00:20 +01:00
|
|
|
|
2015-06-28 22:31:30 +02:00
|
|
|
sys.path.insert(0, os.path.join(os.path.dirname(__file__), os.pardir,
|
|
|
|
os.pardir))
|
2015-03-26 08:00:20 +01:00
|
|
|
|
|
|
|
from scripts import utils
|
|
|
|
|
|
|
|
|
2015-06-29 17:44:36 +02:00
|
|
|
def _get_files(only_py=False):
|
2015-03-31 22:12:38 +02:00
|
|
|
"""Iterate over all python files and yield filenames."""
|
2015-06-05 18:18:29 +02:00
|
|
|
for (dirpath, _dirnames, filenames) in os.walk('.'):
|
|
|
|
parts = dirpath.split(os.sep)
|
|
|
|
if len(parts) >= 2 and parts[1].startswith('.'):
|
|
|
|
# ignore hidden dirs
|
|
|
|
continue
|
2015-06-29 17:44:36 +02:00
|
|
|
|
|
|
|
if only_py:
|
|
|
|
endings = {'.py'}
|
|
|
|
else:
|
|
|
|
endings = {'.py', '.asciidoc', '.js'}
|
|
|
|
files = (e for e in filenames if os.path.splitext(e)[1] in endings)
|
|
|
|
for name in files:
|
2015-03-31 22:12:38 +02:00
|
|
|
yield os.path.join(dirpath, name)
|
|
|
|
|
|
|
|
|
2015-03-26 08:00:20 +01:00
|
|
|
def check_git():
|
2015-03-31 20:49:29 +02:00
|
|
|
"""Check for uncommitted git files.."""
|
2015-03-26 08:00:20 +01:00
|
|
|
if not os.path.isdir(".git"):
|
|
|
|
print("No .git dir, ignoring")
|
|
|
|
print()
|
|
|
|
return False
|
|
|
|
untracked = []
|
|
|
|
gitst = subprocess.check_output(['git', 'status', '--porcelain'])
|
|
|
|
gitst = gitst.decode('UTF-8').strip()
|
|
|
|
for line in gitst.splitlines():
|
|
|
|
s, name = line.split(maxsplit=1)
|
|
|
|
if s == '??' and name != '.venv/':
|
|
|
|
untracked.append(name)
|
|
|
|
status = True
|
|
|
|
if untracked:
|
|
|
|
status = False
|
|
|
|
utils.print_col("Untracked files:", 'red')
|
|
|
|
print('\n'.join(untracked))
|
|
|
|
print()
|
|
|
|
return status
|
|
|
|
|
|
|
|
|
2015-06-05 18:18:29 +02:00
|
|
|
def check_spelling():
|
2015-03-31 22:12:38 +02:00
|
|
|
"""Check commonly misspelled words."""
|
|
|
|
# Words which I often misspell
|
2015-06-07 01:11:25 +02:00
|
|
|
words = {'[Bb]ehaviour', '[Qq]uitted', 'Ll]ikelyhood', '[Ss]ucessfully',
|
|
|
|
'[Oo]ccur[^r .]', '[Ss]eperator', '[Ee]xplicitely', '[Rr]esetted',
|
|
|
|
'[Aa]uxillary', '[Aa]ccidentaly', '[Aa]mbigious', '[Ll]oosly',
|
|
|
|
'[Ii]nitialis', '[Cc]onvienence', '[Ss]imiliar', '[Uu]ncommited',
|
|
|
|
'[Rr]eproducable'}
|
2015-03-31 22:12:38 +02:00
|
|
|
|
|
|
|
# Words which look better when splitted, but might need some fine tuning.
|
2015-06-29 17:45:56 +02:00
|
|
|
words |= {'[Ww]ebelements', '[Mm]ouseevent', '[Kk]eysequence',
|
|
|
|
'[Nn]ormalmode', '[Ee]ventloops', '[Ss]izehint',
|
|
|
|
'[Ss]tatemachine', '[Mm]etaobject', '[Ll]ogrecord',
|
|
|
|
'[Ff]iletype'}
|
2015-03-31 22:12:38 +02:00
|
|
|
|
|
|
|
seen = collections.defaultdict(list)
|
|
|
|
try:
|
|
|
|
ok = True
|
2015-06-29 17:44:36 +02:00
|
|
|
for fn in _get_files():
|
2015-03-31 22:12:38 +02:00
|
|
|
with tokenize.open(fn) as f:
|
2015-06-28 22:31:30 +02:00
|
|
|
if fn == os.path.join('.', 'scripts', 'dev', 'misc_checks.py'):
|
2015-03-31 22:12:38 +02:00
|
|
|
continue
|
|
|
|
for line in f:
|
|
|
|
for w in words:
|
2015-06-07 01:11:25 +02:00
|
|
|
if re.search(w, line) and fn not in seen[w]:
|
|
|
|
print('Found "{}" in {}!'.format(w, fn))
|
2015-03-31 22:12:38 +02:00
|
|
|
seen[w].append(fn)
|
2015-04-05 19:37:56 +02:00
|
|
|
ok = False
|
2015-03-31 22:12:38 +02:00
|
|
|
print()
|
|
|
|
return ok
|
|
|
|
except Exception:
|
|
|
|
traceback.print_exc()
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
2015-06-05 18:18:29 +02:00
|
|
|
def check_vcs_conflict():
|
2015-03-26 08:00:20 +01:00
|
|
|
"""Check VCS conflict markers."""
|
|
|
|
try:
|
|
|
|
ok = True
|
2015-06-29 17:44:36 +02:00
|
|
|
for fn in _get_files(only_py=True):
|
2015-03-31 22:12:38 +02:00
|
|
|
with tokenize.open(fn) as f:
|
|
|
|
for line in f:
|
|
|
|
if any(line.startswith(c * 7) for c in '<>=|'):
|
|
|
|
print("Found conflict marker in {}".format(fn))
|
|
|
|
ok = False
|
2015-03-26 08:00:20 +01:00
|
|
|
print()
|
|
|
|
return ok
|
|
|
|
except Exception:
|
|
|
|
traceback.print_exc()
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
"""Main entry point."""
|
|
|
|
parser = argparse.ArgumentParser()
|
2015-03-31 22:12:38 +02:00
|
|
|
parser.add_argument('checker', choices=('git', 'vcs', 'spelling'),
|
2015-03-26 08:00:20 +01:00
|
|
|
help="Which checker to run.")
|
|
|
|
args = parser.parse_args()
|
|
|
|
if args.checker == 'git':
|
|
|
|
ok = check_git()
|
|
|
|
elif args.checker == 'vcs':
|
2015-06-05 18:18:29 +02:00
|
|
|
ok = check_vcs_conflict()
|
2015-03-31 22:12:38 +02:00
|
|
|
elif args.checker == 'spelling':
|
2015-06-05 18:18:29 +02:00
|
|
|
ok = check_spelling()
|
|
|
|
return 0 if ok else 1
|
2015-03-26 08:00:20 +01:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
sys.exit(main())
|