210 lines
6.4 KiB
Python
210 lines
6.4 KiB
Python
# Copyright 2010-2014 The Rust Project Developers. See the COPYRIGHT
|
|
# file at the top-level directory of this distribution and at
|
|
# http://rust-lang.org/COPYRIGHT.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
# http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
# <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
# option. This file may not be copied, modified, or distributed
|
|
# except according to those terms.
|
|
|
|
import sys
|
|
import fileinput
|
|
import subprocess
|
|
import re
|
|
import os
|
|
from licenseck import check_license
|
|
import snapshot
|
|
|
|
err = 0
|
|
cols = 100
|
|
cr_flag = "ignore-tidy-cr"
|
|
tab_flag = "ignore-tidy-tab"
|
|
linelength_flag = "ignore-tidy-linelength"
|
|
|
|
interesting_files = ['.rs', '.py', '.js', '.sh', '.c', '.h']
|
|
uninteresting_files = ['miniz.c', 'jquery', 'rust_android_dummy']
|
|
|
|
|
|
def report_error_name_no(name, no, s):
|
|
global err
|
|
print("%s:%d: %s" % (name, no, s))
|
|
err = 1
|
|
|
|
|
|
def report_err(s):
|
|
report_error_name_no(fileinput.filename(), fileinput.filelineno(), s)
|
|
|
|
|
|
def report_warn(s):
|
|
print("%s:%d: %s" % (fileinput.filename(),
|
|
fileinput.filelineno(),
|
|
s))
|
|
|
|
|
|
def do_license_check(name, contents):
|
|
if not check_license(name, contents):
|
|
report_error_name_no(name, 1, "incorrect license")
|
|
|
|
|
|
def update_counts(current_name):
|
|
global file_counts
|
|
global count_other_linted_files
|
|
|
|
_, ext = os.path.splitext(current_name)
|
|
|
|
if ext in interesting_files:
|
|
file_counts[ext] += 1
|
|
else:
|
|
count_other_linted_files += 1
|
|
|
|
|
|
def interesting_file(f):
|
|
if any(x in f for x in uninteresting_files):
|
|
return False
|
|
|
|
return any(os.path.splitext(f)[1] == ext for ext in interesting_files)
|
|
|
|
|
|
# Be careful to support Python 2.4, 2.6, and 3.x here!
|
|
config_proc = subprocess.Popen(["git", "config", "core.autocrlf"],
|
|
stdout=subprocess.PIPE)
|
|
result = config_proc.communicate()[0]
|
|
|
|
true = "true".encode('utf8')
|
|
autocrlf = result.strip() == true if result is not None else False
|
|
|
|
current_name = ""
|
|
current_contents = ""
|
|
check_tab = True
|
|
check_cr = True
|
|
check_linelength = True
|
|
|
|
if len(sys.argv) < 2:
|
|
print("usage: tidy.py <src-dir>")
|
|
sys.exit(1)
|
|
|
|
src_dir = sys.argv[1]
|
|
|
|
count_lines = 0
|
|
count_non_blank_lines = 0
|
|
count_other_linted_files = 0
|
|
|
|
file_counts = {ext: 0 for ext in interesting_files}
|
|
|
|
all_paths = set()
|
|
|
|
try:
|
|
for (dirpath, dirnames, filenames) in os.walk(src_dir):
|
|
# Skip some third-party directories
|
|
skippable_dirs = {
|
|
'src/jemalloc',
|
|
'src/llvm',
|
|
'src/gyp',
|
|
'src/libbacktrace',
|
|
'src/libuv',
|
|
'src/compiler-rt',
|
|
'src/rt/hoedown',
|
|
'src/rustllvm',
|
|
'src/rt/valgrind',
|
|
'src/rt/msvc',
|
|
'src/rust-installer'
|
|
}
|
|
|
|
if any(d in dirpath for d in skippable_dirs):
|
|
continue
|
|
|
|
file_names = [os.path.join(dirpath, f) for f in filenames
|
|
if interesting_file(f)
|
|
and not f.endswith("_gen.rs")
|
|
and not ".#" is f]
|
|
|
|
if not file_names:
|
|
continue
|
|
|
|
for line in fileinput.input(file_names,
|
|
openhook=fileinput.hook_encoded("utf-8")):
|
|
|
|
filename = fileinput.filename()
|
|
|
|
if "tidy.py" not in filename:
|
|
if "TODO" in line:
|
|
report_err("TODO is deprecated; use FIXME")
|
|
match = re.match(r'^.*/(\*|/!?)\s*XXX', line)
|
|
if match:
|
|
report_err("XXX is no longer necessary, use FIXME")
|
|
match = re.match(r'^.*//\s*(NOTE.*)$', line)
|
|
if match and "TRAVIS" not in os.environ:
|
|
m = match.group(1)
|
|
if "snap" in m.lower():
|
|
report_warn(match.group(1))
|
|
match = re.match(r'^.*//\s*SNAP\s+(\w+)', line)
|
|
if match:
|
|
hsh = match.group(1)
|
|
date, rev = snapshot.curr_snapshot_rev()
|
|
if not hsh.startswith(rev):
|
|
report_err("snapshot out of date (" + date
|
|
+ "): " + line)
|
|
else:
|
|
if "SNAP" in line:
|
|
report_warn("unmatched SNAP line: " + line)
|
|
|
|
if cr_flag in line:
|
|
check_cr = False
|
|
if tab_flag in line:
|
|
check_tab = False
|
|
if linelength_flag in line:
|
|
check_linelength = False
|
|
|
|
if check_tab and ('\t' in line and
|
|
"Makefile" not in filename):
|
|
report_err("tab character")
|
|
if check_cr and not autocrlf and '\r' in line:
|
|
report_err("CR character")
|
|
if line.endswith(" \n") or line.endswith("\t\n"):
|
|
report_err("trailing whitespace")
|
|
line_len = len(line)-2 if autocrlf else len(line)-1
|
|
|
|
if check_linelength and line_len > cols:
|
|
report_err("line longer than %d chars" % cols)
|
|
|
|
if fileinput.isfirstline():
|
|
# This happens at the end of each file except the last.
|
|
if current_name != "":
|
|
update_counts(current_name)
|
|
assert len(current_contents) > 0
|
|
do_license_check(current_name, current_contents)
|
|
|
|
current_name = filename
|
|
current_contents = ""
|
|
check_cr = True
|
|
check_tab = True
|
|
check_linelength = True
|
|
|
|
# Put a reasonable limit on the amount of header data we use for
|
|
# the licenseck
|
|
if len(current_contents) < 1000:
|
|
current_contents += line
|
|
|
|
count_lines += 1
|
|
if line.strip():
|
|
count_non_blank_lines += 1
|
|
|
|
if current_name != "":
|
|
update_counts(current_name)
|
|
assert len(current_contents) > 0
|
|
do_license_check(current_name, current_contents)
|
|
|
|
except UnicodeDecodeError as e:
|
|
report_err("UTF-8 decoding error " + str(e))
|
|
|
|
print
|
|
for ext in sorted(file_counts, key=file_counts.get, reverse=True):
|
|
print("* linted {} {} files".format(file_counts[ext], ext))
|
|
print("* linted {} other files".format(count_other_linted_files))
|
|
print("* total lines of code: {}".format(count_lines))
|
|
print("* total non-blank lines of code: {}".format(count_non_blank_lines))
|
|
print()
|
|
|
|
sys.exit(err)
|