9821cc4510
* Address issues raised in PR comments * Add isort tool to code_quality fab task * Applied isort tool to changed files * Refactored devops dependencies install method * Refactored application path munging * Made bitmessagecli executable * Minor bug fixes to toolchain * Removed stale donation address * Added absolute_imports to give hints to pylint
328 lines
11 KiB
Python
328 lines
11 KiB
Python
# pylint: disable=not-context-manager
|
|
"""
|
|
Fabric tasks for PyBitmessage devops operations.
|
|
|
|
Note that where tasks declare params to be bools, they use coerce_bool() and so will accept any commandline (string)
|
|
representation of true or false that coerce_bool() understands.
|
|
"""
|
|
|
|
import os
|
|
import sys
|
|
|
|
from fabric.api import cd, hide, run, settings, sudo, task
|
|
from fabric.contrib.project import rsync_project
|
|
from fabvenv import virtualenv
|
|
|
|
import fabfile.app_path # pylint: disable=unused-import
|
|
|
|
from version import softwareVersion
|
|
|
|
from fabfile.lib import (
|
|
PROJECT_ROOT, VENV_ROOT, autopep8, coerce_bool, default_hosts, filelist_from_git, flatten,
|
|
get_filtered_flake8_output, get_filtered_pycodestyle_output, get_filtered_pylint_output, isort
|
|
)
|
|
|
|
|
|
def get_tool_results(file_list):
|
|
"""Take a list of files and resuln the results of applying the tools"""
|
|
|
|
results = []
|
|
for path_to_file in file_list:
|
|
result = {}
|
|
result['pycodestyle_violations'] = get_filtered_pycodestyle_output(path_to_file)
|
|
result['flake8_violations'] = get_filtered_flake8_output(path_to_file)
|
|
result['pylint_violations'] = get_filtered_pylint_output(path_to_file)
|
|
result['path_to_file'] = path_to_file
|
|
result['total_violations'] = sum([
|
|
len(result['pycodestyle_violations']),
|
|
len(result['flake8_violations']),
|
|
len(result['pylint_violations']),
|
|
])
|
|
results.append(result)
|
|
return results
|
|
|
|
|
|
def print_results(results, top, verbose, details):
|
|
"""Print an item with the appropriate verbosity / detail"""
|
|
|
|
if verbose and results:
|
|
print ''.join(
|
|
[
|
|
os.linesep,
|
|
'total pycodestyle flake8 pylint path_to_file',
|
|
os.linesep,
|
|
]
|
|
)
|
|
|
|
for item in sort_and_slice(results, top):
|
|
|
|
if verbose:
|
|
line = "{0} {1} {2} {3} {4}".format(
|
|
item['total_violations'],
|
|
len(item['pycodestyle_violations']),
|
|
len(item['flake8_violations']),
|
|
len(item['pylint_violations']),
|
|
item['path_to_file'],
|
|
)
|
|
else:
|
|
line = item['path_to_file']
|
|
print line
|
|
|
|
if details:
|
|
print
|
|
print "pycodestyle:"
|
|
for detail in flatten(item['pycodestyle_violations']):
|
|
print detail
|
|
print
|
|
|
|
print "flake8:"
|
|
for detail in flatten(item['flake8_violations']):
|
|
print detail
|
|
print
|
|
|
|
print "pylint:"
|
|
for detail in flatten(item['pylint_violations']):
|
|
print detail
|
|
print
|
|
|
|
|
|
def sort_and_slice(results, top):
|
|
"""Sort dictionary items by the `total_violations` key and honour top"""
|
|
|
|
returnables = []
|
|
for item in sorted(
|
|
results,
|
|
reverse=True,
|
|
key=lambda x: x['total_violations']
|
|
)[:top]:
|
|
returnables.append(item)
|
|
return returnables
|
|
|
|
|
|
def generate_file_list(filename):
|
|
"""Return an unfiltered list of absolute paths to the files to act on"""
|
|
|
|
with hide('warnings', 'running', 'stdout'):
|
|
with virtualenv(VENV_ROOT):
|
|
|
|
if filename:
|
|
filename = os.path.abspath(filename)
|
|
if not os.path.exists(filename):
|
|
print "Bad filename {}, specify a Python file".format(filename)
|
|
sys.exit(1)
|
|
else:
|
|
file_list = [filename]
|
|
|
|
else:
|
|
with cd(PROJECT_ROOT):
|
|
file_list = [
|
|
os.path.abspath(i.rstrip('\r'))
|
|
for i in run('find . -name "*.py"').split(os.linesep)
|
|
]
|
|
|
|
return file_list
|
|
|
|
|
|
def create_dependency_graphs():
|
|
"""
|
|
To better understand the relationship between methods, dependency graphs can be drawn between functions and
|
|
methods.
|
|
|
|
Since the resulting images are large and could be out of date on the next commit, storing them in the repo is
|
|
pointless. Instead, it makes sense to build a dependency graph for a particular, documented version of the code.
|
|
|
|
.. todo:: Consider saving a hash of the intermediate dotty file so unnecessary image generation can be avoided.
|
|
|
|
"""
|
|
with virtualenv(VENV_ROOT):
|
|
with hide('running', 'stdout'):
|
|
|
|
# .. todo:: consider a better place to put this, use a particular commit
|
|
with cd(PROJECT_ROOT):
|
|
with settings(warn_only=True):
|
|
if run('stat pyan').return_code:
|
|
run('git clone https://github.com/davidfraser/pyan.git')
|
|
with cd(os.path.join(PROJECT_ROOT, 'pyan')):
|
|
run('git checkout pre-python3')
|
|
|
|
# .. todo:: Use better settings. This is MVP to make a diagram
|
|
with cd(PROJECT_ROOT):
|
|
file_list = run("find . -type f -name '*.py' ! -path './src/.eggs/*'").split('\r\n')
|
|
for cmd in [
|
|
'neato -Goverlap=false -Tpng > deps-neato.png',
|
|
'sfdp -Goverlap=false -Tpng > deps-sfdp.png',
|
|
'dot -Goverlap=false -Tpng > deps-dot.png',
|
|
]:
|
|
pyan_cmd = './pyan/pyan.py {} --dot'.format(' '.join(file_list))
|
|
sed_cmd = r"sed s'/http\-old/http_old/'g" # dot doesn't like dashes
|
|
run('|'.join([pyan_cmd, sed_cmd, cmd]))
|
|
|
|
run('mv *.png docs/_build/html/_static/')
|
|
|
|
|
|
@task
|
|
@default_hosts(['localhost'])
|
|
def code_quality(verbose=True, details=False, fix=False, filename=None, top=10, rev=None):
|
|
"""
|
|
Check code quality.
|
|
|
|
By default this command will analyse each Python file in the project with a variety of tools and display the count
|
|
or details of the violations discovered, sorted by most violations first.
|
|
|
|
Default usage:
|
|
|
|
$ fab -H localhost code_quality
|
|
|
|
:param rev: If not None, act on files changed since this commit. 'cached/staged' and 'working' have special meaning
|
|
:type rev: str or None, default None
|
|
:param top: Display / fix only the top N violating files, a value of 0 will display / fix all files
|
|
:type top: int, default 10
|
|
:param verbose: Display a header and the counts, without this you just get the filenames in order
|
|
:type verbose: bool, default True
|
|
:param details: Display the violations one per line after the count / file summary
|
|
:type details: bool, default False
|
|
:param fix: Run autopep8 aggressively on the displayed file(s)
|
|
:type fix: bool, default False
|
|
:param filename: Don't test/fix the top N, just the specified file
|
|
:type filename: string, valid path to a file, default all files in the project
|
|
:return: None, exit status equals total number of violations
|
|
:rtype: None
|
|
|
|
Intended to be temporary until we have improved code quality and have safeguards to maintain it in place.
|
|
|
|
"""
|
|
# pylint: disable=too-many-arguments
|
|
|
|
verbose = coerce_bool(verbose)
|
|
details = coerce_bool(details)
|
|
fix = coerce_bool(fix)
|
|
top = int(top) or -1
|
|
|
|
file_list = generate_file_list(filename) if not rev else filelist_from_git(rev)
|
|
results = get_tool_results(file_list)
|
|
|
|
if fix:
|
|
if filename:
|
|
sorted_and_sliced = [{'path_to_file': os.path.abspath(filename)}]
|
|
else:
|
|
sorted_and_sliced = sort_and_slice(results, top)
|
|
|
|
for item in sorted_and_sliced:
|
|
path_to_file = item['path_to_file']
|
|
print 'Applying automatic fixes to {}'.format(path_to_file)
|
|
isort(path_to_file)
|
|
autopep8(path_to_file)
|
|
# Recalculate results after autopep8 to surprise the user the least
|
|
results = get_tool_results(file_list)
|
|
|
|
print_results(results, top, verbose, details)
|
|
sys.exit(sum([item['total_violations'] for item in results]))
|
|
|
|
|
|
@task
|
|
@default_hosts(['localhost'])
|
|
def test():
|
|
"""Run tests on the code"""
|
|
|
|
with cd(PROJECT_ROOT):
|
|
with virtualenv(VENV_ROOT):
|
|
|
|
run('pip uninstall -y pybitmessage')
|
|
run('python setup.py install')
|
|
|
|
run('pybitmessage -t')
|
|
run('python setup.py test')
|
|
|
|
|
|
@task
|
|
@default_hosts(['localhost'])
|
|
def build_docs(dep_graph=False, apidoc=True):
|
|
"""
|
|
Build the documentation locally.
|
|
|
|
:param dep_graph: Build the dependency graphs
|
|
:type dep_graph: Bool, default False
|
|
:param apidoc: Build the automatically generated rst files from the source code
|
|
:type apidoc: Bool, default True
|
|
|
|
Default usage:
|
|
|
|
$ fab -H localhost build_docs
|
|
|
|
Implementation:
|
|
|
|
First, a dependency graph is generated and converted into an image that is referenced in the development page.
|
|
|
|
Next, the sphinx-apidoc command is (usually) run which searches the code. As part of this it loads the modules and
|
|
if this has side-effects then they will be evident. Any documentation strings that make use of Python documentation
|
|
conventions (like parameter specification) or the Restructured Text (RsT) syntax will be extracted.
|
|
|
|
Next, the `make html` command is run to generate HTML output. Other formats (epub, pdf) are available.
|
|
|
|
.. todo:: support other languages
|
|
|
|
"""
|
|
|
|
apidoc = coerce_bool(apidoc)
|
|
|
|
if coerce_bool(dep_graph):
|
|
create_dependency_graphs()
|
|
|
|
with virtualenv(VENV_ROOT):
|
|
with hide('running'):
|
|
|
|
apidoc_result = 0
|
|
if apidoc:
|
|
run('mkdir -p {}'.format(os.path.join(PROJECT_ROOT, 'docs', 'autodoc')))
|
|
with cd(os.path.join(PROJECT_ROOT, 'docs', 'autodoc')):
|
|
with settings(warn_only=True):
|
|
run('rm *.rst')
|
|
with cd(os.path.join(PROJECT_ROOT, 'docs')):
|
|
apidoc_result = run('sphinx-apidoc -o autodoc ..').return_code
|
|
|
|
with cd(os.path.join(PROJECT_ROOT, 'docs')):
|
|
make_result = run('make html').return_code
|
|
return_code = apidoc_result + make_result
|
|
|
|
sys.exit(return_code)
|
|
|
|
|
|
@task
|
|
@default_hosts(['localhost'])
|
|
def push_docs(path=None):
|
|
"""
|
|
Upload the generated docs to a public server.
|
|
|
|
Default usage:
|
|
|
|
$ fab -H localhost push_docs
|
|
|
|
.. todo:: support other languages
|
|
.. todo:: integrate with configuration management data to get web root and webserver restart command
|
|
|
|
"""
|
|
|
|
# Making assumptions
|
|
WEB_ROOT = path if path is not None else os.path.join('var', 'www', 'html', 'pybitmessage', 'en', 'latest')
|
|
VERSION_ROOT = os.path.join(os.path.dirname(WEB_ROOT), softwareVersion)
|
|
|
|
rsync_project(
|
|
remote_dir=VERSION_ROOT,
|
|
local_dir=os.path.join(PROJECT_ROOT, 'docs', '_build', 'html')
|
|
)
|
|
result = run('ln -sf {0} {1}'.format(WEB_ROOT, VERSION_ROOT))
|
|
if result.return_code:
|
|
print 'Linking the new release failed'
|
|
|
|
# More assumptions
|
|
sudo('systemctl restart apache2')
|
|
|
|
|
|
@task
|
|
@default_hosts(['localhost'])
|
|
def clean():
|
|
"""Clean up files generated by fabric commands."""
|
|
with hide('running', 'stdout'):
|
|
with cd(PROJECT_ROOT):
|
|
run(r"find . -name '*.pyc' -exec rm '{}' \;")
|