summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJonathan Sharpe <j.r.sharpe+github@gmail.com>2015-11-08 16:19:05 +0000
committerJonathan Sharpe <j.r.sharpe+github@gmail.com>2015-11-08 16:19:05 +0000
commit745c80aa301347cf1ec0ee4016b1683388aa7288 (patch)
treeb2649a444d40157dd7cc5be346a997396326d2ff
parent8a55a6238fd0cccb69cb076e598c3ca806b22911 (diff)
parentff586cd08ad4e226ea5d65eda8683afb5a5c9373 (diff)
downloadpycco-745c80aa301347cf1ec0ee4016b1683388aa7288.tar.gz
Merged latest changes from upstream/master
-rw-r--r--.gitignore10
-rw-r--r--.travis.yml12
-rw-r--r--README.md (renamed from README)7
-rw-r--r--pycco/compat.py4
-rw-r--r--pycco/main.py223
-rw-r--r--requirements.test.txt3
-rw-r--r--requirements.txt3
-rw-r--r--tests/__init__.py0
-rw-r--r--tests/test_pycco.py115
9 files changed, 288 insertions, 89 deletions
diff --git a/.gitignore b/.gitignore
index 07b20dd..e1c9655 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,5 +1,13 @@
+.coverage
*.pyc
/Pycco.egg-info
build/*
dist/*
-docs/* \ No newline at end of file
+docs/*
+/tags
+
+.cache
+.hypothesis
+.ropeproject
+
+.DS_Store
diff --git a/.travis.yml b/.travis.yml
new file mode 100644
index 0000000..62e7c6b
--- /dev/null
+++ b/.travis.yml
@@ -0,0 +1,12 @@
+language: python
+python:
+ - '2.7'
+ - '3.5'
+install:
+ - 'pip install -r requirements.txt'
+ - 'pip install -r requirements.test.txt'
+script:
+ - 'py.test --cov=pycco tests/'
+ - 'python -m pycco.main pycco/main.py'
+after_success:
+ - coveralls
diff --git a/README b/README.md
index 5343728..9cad492 100644
--- a/README
+++ b/README.md
@@ -1,3 +1,4 @@
+```
888888b.
888 Y88b
888 888
@@ -9,6 +10,7 @@
888
Y8b d88P
"Y88P"
+```
Pycco is a Python port of Docco: the original quick-and-dirty, hundred-line-
long, literate-programming-style documentation generator. For more information,
@@ -22,4 +24,7 @@ CoffeeScript (Original) - http://jashkenas.github.com/docco/
Ruby - http://rtomayko.github.com/rocco/
-Sh - http://rtomayko.github.com/shocco/ \ No newline at end of file
+Sh - http://rtomayko.github.com/shocco/
+
+[![Build Status](https://travis-ci.org/subsetpark/pycco.svg?branch=hypothesis)](https://travis-ci.org/subsetpark/pycco)
+[![Coverage Status](https://coveralls.io/repos/subsetpark/pycco/badge.svg?branch=hypothesis&service=github)](https://coveralls.io/github/subsetpark/pycco?branch=hypothesis)
diff --git a/pycco/compat.py b/pycco/compat.py
new file mode 100644
index 0000000..6660531
--- /dev/null
+++ b/pycco/compat.py
@@ -0,0 +1,4 @@
+try:
+ pycco_unichr = unichr
+except NameError:
+ pycco_unichr = chr
diff --git a/pycco/main.py b/pycco/main.py
index 77c7650..6d45673 100644
--- a/pycco/main.py
+++ b/pycco/main.py
@@ -1,4 +1,5 @@
#!/usr/bin/env python
+from __future__ import print_function
"""
"**Pycco**" is a Python port of [Docco](http://jashkenas.github.com/docco/):
@@ -33,8 +34,9 @@ Or, to install the latest source
# === Main Documentation Generation Functions ===
+
def generate_documentation(source, outdir=None, preserve_paths=True,
- language=None):
+ language=None, encoding="utf8"):
"""
Generate the documentation for a source file by reading it in, splitting it
up into comment/code sections, highlighting them for the appropriate
@@ -43,18 +45,25 @@ def generate_documentation(source, outdir=None, preserve_paths=True,
if not outdir:
raise TypeError("Missing the required 'outdir' keyword argument.")
- code = open(source, "r").read()
- language = get_language(source, code, language=language)
- sections = parse(source, code, language)
- highlight(source, sections, language, preserve_paths=preserve_paths, outdir=outdir)
- return generate_html(source, sections, preserve_paths=preserve_paths, outdir=outdir)
+ code = open(source, "rb").read().decode(encoding)
+ return _generate_documentation(source, code, outdir, preserve_paths, language)
+
+
+def _generate_documentation(file_path, code, outdir, preserve_paths, language):
+ """
+ Helper function to allow documentation generation without file handling.
+ """
+ language = get_language(file_path, code, language=language)
+ sections = parse(code, language)
+ highlight(sections, language, preserve_paths=preserve_paths, outdir=outdir)
+ return generate_html(file_path, sections, preserve_paths=preserve_paths, outdir=outdir)
-def parse(source, code, language):
+
+def parse(code, language):
"""
Given a string of source code, parse out each comment and the code that
follows it, and create an individual **section** for it.
Sections take the form:
-
{ "docs_text": ...,
"docs_html": ...,
"code_text": ...,
@@ -76,7 +85,6 @@ def parse(source, code, language):
lines.pop(linenum)
break
-
def save(docs, code):
if docs or code:
sections.append({
@@ -86,27 +94,25 @@ def parse(source, code, language):
# Setup the variables to get ready to check for multiline comments
multi_line = False
- multi_line_delimiters = [language.get("multistart"), language.get("multiend")]
+ multistart, multiend = language.get("multistart"), language.get("multiend")
+ comment_matcher = language['comment_matcher']
for line in lines:
-
# Only go into multiline comments section when one of the delimiters is
# found to be at the start of a line
- if all(multi_line_delimiters) and any([line.lstrip().startswith(delim) or line.rstrip().endswith(delim) for delim in multi_line_delimiters]):
- if not multi_line:
- multi_line = True
-
- else:
- multi_line = False
+ if multistart and multiend and \
+ any(line.lstrip().startswith(delim) or line.rstrip().endswith(delim)
+ for delim in (multistart, multiend)):
+ multi_line = not multi_line
if (multi_line
- and line.strip().endswith(language.get("multiend"))
- and len(line.strip()) > len(language.get("multiend"))):
+ and line.strip().endswith(multiend)
+ and len(line.strip()) > len(multiend)):
multi_line = False
# Get rid of the delimiters so that they aren't in the final docs
- line = line.replace(language["multistart"], '')
- line = line.replace(language["multiend"], '')
+ line = line.replace(multistart, '')
+ line = line.replace(multiend, '')
docs_text += line.strip() + '\n'
indent_level = re.match("\s*", line).group(0)
@@ -117,19 +123,20 @@ def parse(source, code, language):
elif multi_line:
# Remove leading spaces
- if re.match(r' {%d}' % len(indent_level), line):
+ if re.match(r' {{{:d}}}'.format(len(indent_level)), line):
docs_text += line[len(indent_level):] + '\n'
else:
docs_text += line + '\n'
- elif re.match(language["comment_matcher"], line):
+ elif re.match(comment_matcher, line):
if has_code:
save(docs_text, code_text)
has_code = docs_text = code_text = ''
- docs_text += re.sub(language["comment_matcher"], "", line) + "\n"
+ docs_text += re.sub(comment_matcher, "", line) + "\n"
else:
- if code_text and any([line.lstrip().startswith(x) for x in ['class ', 'def ', '@']]):
+ if code_text and any(line.lstrip().startswith(x)
+ for x in ['class ', 'def ', '@']):
if not code_text.lstrip().startswith("@"):
save(docs_text, code_text)
code_text = has_code = docs_text = ''
@@ -137,14 +144,14 @@ def parse(source, code, language):
has_code = True
code_text += line + '\n'
-
save(docs_text, code_text)
return sections
# === Preprocessing the comments ===
-def preprocess(comment, section_nr, preserve_paths=True, outdir=None):
+
+def preprocess(comment, preserve_paths=True, outdir=None):
"""
Add cross-references before having the text processed by markdown. It's
possible to reference another file, like this : `[[main.py]]` which renders
@@ -157,6 +164,7 @@ def preprocess(comment, section_nr, preserve_paths=True, outdir=None):
if not outdir:
raise TypeError("Missing the required 'outdir' keyword argument.")
+
def sanitize_section_name(name):
return "-".join(name.lower().strip().split(" "))
@@ -164,24 +172,28 @@ def preprocess(comment, section_nr, preserve_paths=True, outdir=None):
# Check if the match contains an anchor
if '#' in match.group(1):
name, anchor = match.group(1).split('#')
- return "[%s](%s#%s)" % (name,
- path.basename(destination(name,
- preserve_paths=preserve_paths,
- outdir=outdir)),
- anchor)
+
+ return " [{}]({}#{})".format(name,
+ path.basename(destination(name,
+ preserve_paths=preserve_paths,
+ outdir=outdir)),
+ anchor)
else:
- return "[%s](%s)" % (match.group(1),
- path.basename(destination(match.group(1),
- preserve_paths=preserve_paths,
- outdir=outdir)))
+ return " [{}]({})".format(match.group(1),
+ path.basename(destination(match.group(1),
+ preserve_paths=preserve_paths,
+ outdir=outdir)))
def replace_section_name(match):
- return '%(lvl)s <span id="%(id)s" href="%(id)s">%(name)s</span>' % {
- "lvl" : re.sub('=', '#', match.group(1)),
- "id" : sanitize_section_name(match.group(2)),
- "name" : match.group(2)
- }
+ """
+ Replace equals-sign-formatted section names with anchor links.
+ """
+ return '{lvl} <span id="{id}" href="{id}">{name}</span>'.format(
+ lvl=re.sub('=', '#', match.group(1)),
+ id=sanitize_section_name(match.group(2)),
+ name=match.group(2)
+ )
comment = re.sub('^([=]+)([^=]+)[=]*\s*$', replace_section_name, comment)
comment = re.sub('(?<!`)\[\[(.+?)\]\]', replace_crossref, comment)
@@ -190,7 +202,8 @@ def preprocess(comment, section_nr, preserve_paths=True, outdir=None):
# === Highlighting the source code ===
-def highlight(source, sections, language, preserve_paths=True, outdir=None):
+
+def highlight(sections, language, preserve_paths=True, outdir=None):
"""
Highlights a single chunk of code using the **Pygments** module, and runs
the text of its corresponding comment through **Markdown**.
@@ -215,14 +228,16 @@ def highlight(source, sections, language, preserve_paths=True, outdir=None):
docs_text = unicode(section["docs_text"])
except UnicodeError:
docs_text = unicode(section["docs_text"].decode('utf-8'))
+ except NameError:
+ docs_text = section['docs_text']
section["docs_html"] = markdown(preprocess(docs_text,
- i,
preserve_paths=preserve_paths,
outdir=outdir))
section["num"] = i
# === HTML Code generation ===
+
def generate_html(source, sections, preserve_paths=True, outdir=None):
"""
Once all of the code is finished highlighting, we can generate the HTML file
@@ -245,12 +260,12 @@ def generate_html(source, sections, preserve_paths=True, outdir=None):
sect["code_html"] = re.sub(r"\{\{", r"__DOUBLE_OPEN_STACHE__", sect["code_html"])
rendered = pycco_template({
- "title" : title,
- "stylesheet" : csspath,
- "sections" : sections,
- "source" : source,
- "path" : path,
- "destination" : destination
+ "title": title,
+ "stylesheet": csspath,
+ "sections": sections,
+ "source": source,
+ "path": path,
+ "destination": destination
})
return re.sub(r"__DOUBLE_OPEN_STACHE__", "{{", rendered).encode("utf-8")
@@ -276,39 +291,39 @@ from pygments import lexers, formatters
# the name of the Pygments lexer and the symbol that indicates a comment. To
# add another language to Pycco's repertoire, add it here.
languages = {
- ".coffee": { "name": "coffee-script", "symbol": "#",
- "multistart": '###', "multiend": '###' },
+ ".coffee": {"name": "coffee-script", "symbol": "#",
+ "multistart": '###', "multiend": '###'},
- ".pl": { "name": "perl", "symbol": "#" },
+ ".pl": {"name": "perl", "symbol": "#"},
- ".sql": { "name": "sql", "symbol": "--" },
+ ".sql": {"name": "sql", "symbol": "--"},
- ".c": { "name": "c", "symbol": "//",
- "multistart": "/*", "multiend": "*/"},
+ ".c": {"name": "c", "symbol": "//",
+ "multistart": "/*", "multiend": "*/"},
- ".cpp": { "name": "cpp", "symbol": "//"},
+ ".cpp": {"name": "cpp", "symbol": "//"},
- ".js": { "name": "javascript", "symbol": "//",
- "multistart": "/*", "multiend": "*/"},
+ ".js": {"name": "javascript", "symbol": "//",
+ "multistart": "/*", "multiend": "*/"},
- ".rb": { "name": "ruby", "symbol": "#",
- "multistart": "=begin", "multiend": "=end"},
+ ".rb": {"name": "ruby", "symbol": "#",
+ "multistart": "=begin", "multiend": "=end"},
- ".py": { "name": "python", "symbol": "#",
- "multistart": '"""', "multiend": '"""' },
+ ".py": {"name": "python", "symbol": "#",
+ "multistart": '"""', "multiend": '"""'},
- ".scm": { "name": "scheme", "symbol": ";;",
- "multistart": "#|", "multiend": "|#"},
+ ".scm": {"name": "scheme", "symbol": ";;",
+ "multistart": "#|", "multiend": "|#"},
- ".lua": { "name": "lua", "symbol": "--",
- "multistart": "--[[", "multiend": "--]]"},
+ ".lua": {"name": "lua", "symbol": "--",
+ "multistart": "--[[", "multiend": "--]]"},
- ".erl": { "name": "erlang", "symbol": "%%" },
+ ".erl": {"name": "erlang", "symbol": "%%"},
- ".tcl": { "name": "tcl", "symbol": "#" },
+ ".tcl": {"name": "tcl", "symbol": "#"},
- ".hs": { "name": "haskell", "symbol": "--",
- "multistart": "{-", "multiend": "-}"},
+ ".hs": {"name": "haskell", "symbol": "--",
+ "multistart": "{-", "multiend": "-}"},
}
# Build out the appropriate matchers and delimiters for each language.
@@ -327,6 +342,7 @@ for ext, l in languages.items():
# Get the Pygments Lexer for this language.
l["lexer"] = lexers.get_lexer_by_name(l["name"])
+
def get_language(source, code, language=None):
"""Get the current language we're documenting, based on the extension."""
@@ -337,17 +353,24 @@ def get_language(source, code, language=None):
else:
raise ValueError("Unknown forced language: " + language)
- m = re.match(r'.*(\..+)', os.path.basename(source))
+ m = re.match(r'.*(\..+)', os.path.basename(source)) if source else None
if m and m.group(1) in languages:
return languages[m.group(1)]
else:
- lang = lexers.guess_lexer(code).name.lower()
- for l in languages.values():
- if l["name"] == lang:
- return l
- else:
+ try:
+ lang = lexers.guess_lexer(code).name.lower()
+ for l in languages.values():
+ if l["name"] == lang:
+ return l
+ else:
+ raise ValueError()
+ except ValueError:
+ # If pygments can't find any lexers, it will raise its own
+ # subclass of ValueError. We will catch it and raise ours
+ # for consistency.
raise ValueError("Can't figure out the language!")
+
def destination(filepath, preserve_paths=True, outdir=None):
"""
Compute the destination HTML path for an input source file path. If the
@@ -363,7 +386,14 @@ def destination(filepath, preserve_paths=True, outdir=None):
name = filename
if preserve_paths:
name = path.join(dirname, name)
- return path.join(outdir, "%s.html" % name)
+ dest = path.join(outdir, u"{}.html".format(name))
+ # If `join` is passed an absolute path, it will ignore any earlier path
+ # elements. We will force outdir to the beginning of the path to avoid
+ # writing outside our destination.
+ if not dest.startswith(outdir):
+ dest = outdir + os.sep + dest
+ return dest
+
def shift(list, default):
"""
@@ -376,12 +406,27 @@ def shift(list, default):
except IndexError:
return default
-def ensure_directory(directory):
- """Ensure that the destination directory exists."""
+def remove_control_chars(s):
+ # Sanitization regexp copied from
+ # http://stackoverflow.com/questions/92438/stripping-non-printable-characters-from-a-string-in-python
+ from pycco.compat import pycco_unichr
+ control_chars = ''.join(map(pycco_unichr, list(range(0, 32)) + list(range(127, 160))))
+ control_char_re = re.compile(u'[{}]'.format(re.escape(control_chars)))
+ return control_char_re.sub('', s)
+
+
+def ensure_directory(directory):
+ """
+ Sanitize directory string and ensure that the destination directory exists.
+ """
+ directory = remove_control_chars(directory)
if not os.path.isdir(directory):
os.makedirs(directory)
+ return directory
+
+
def template(source):
return lambda context: pystache.render(source, context)
@@ -397,7 +442,8 @@ highlight_start = "<div class=\"highlight\"><pre>"
# The end of each Pygments highlight block.
highlight_end = "</pre></div>"
-def process(sources, preserve_paths=True, outdir=None, language=None):
+
+def process(sources, preserve_paths=True, outdir=None, language=None, encoding="utf8"):
"""For each source file passed as argument, generate the documentation."""
if not outdir:
@@ -409,9 +455,9 @@ def process(sources, preserve_paths=True, outdir=None, language=None):
# Proceed to generating the documentation.
if sources:
- ensure_directory(outdir)
- css = open(path.join(outdir, "pycco.css"), "w")
- css.write(pycco_styles)
+ outdir = ensure_directory(outdir)
+ css = open(path.join(outdir, "pycco.css"), "wb")
+ css.write(pycco_styles.encode(encoding))
css.close()
def next_file():
@@ -423,11 +469,13 @@ def process(sources, preserve_paths=True, outdir=None, language=None):
except OSError:
pass
- with open(dest, "w") as f:
- f.write(generate_documentation(s, preserve_paths=preserve_paths, outdir=outdir,
- language=language))
+ with open(dest, "wb") as f:
+ f.write(generate_documentation(s, preserve_paths=preserve_paths,
+ outdir=outdir,
+ language=language,
+ encoding=encoding))
- print "pycco = %s -> %s" % (s, dest)
+ print("pycco = {} -> {}".format(s, dest))
if sources:
next_file()
@@ -451,6 +499,7 @@ def monitor(sources, opts):
class RegenerateHandler(watchdog.events.FileSystemEventHandler):
"""A handler for recompiling files which triggered watchdog events"""
+
def on_modified(self, event):
"""Regenerate documentation for a file which triggered an event"""
# Re-generate documentation from a source file if it was listed on
diff --git a/requirements.test.txt b/requirements.test.txt
new file mode 100644
index 0000000..8439fc2
--- /dev/null
+++ b/requirements.test.txt
@@ -0,0 +1,3 @@
+hypothesis==1.14.0
+pytest-cov==2.2.0
+coveralls==1.1
diff --git a/requirements.txt b/requirements.txt
new file mode 100644
index 0000000..38964da
--- /dev/null
+++ b/requirements.txt
@@ -0,0 +1,3 @@
+pystache==0.5.4
+Pygments==2.0.2
+markdown==2.6.3
diff --git a/tests/__init__.py b/tests/__init__.py
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/tests/__init__.py
diff --git a/tests/test_pycco.py b/tests/test_pycco.py
new file mode 100644
index 0000000..5a38dd5
--- /dev/null
+++ b/tests/test_pycco.py
@@ -0,0 +1,115 @@
+import copy
+import os
+import tempfile
+import time
+
+import pytest
+from hypothesis import given, example, assume
+from hypothesis.strategies import lists, text, booleans, choices, none
+
+import pycco.main as p
+
+
+PYTHON = p.languages['.py']
+PYCCO_SOURCE = 'pycco/main.py'
+FOO_FUNCTION = """def foo():\n return True"""
+
+
+def get_language(choice):
+ return choice(list(p.languages.values()))
+
+
+@given(lists(text()), text())
+def test_shift(fragments, default):
+ if fragments == []:
+ assert p.shift(fragments, default) == default
+ else:
+ fragments2 = copy.copy(fragments)
+ head = p.shift(fragments, default)
+ assert [head] + fragments == fragments2
+
+
+@given(text(), booleans(), text(min_size=1))
+@example("/foo", True, "0")
+def test_destination(filepath, preserve_paths, outdir):
+ dest = p.destination(filepath, preserve_paths=preserve_paths, outdir=outdir)
+ assert dest.startswith(outdir)
+ assert dest.endswith(".html")
+
+
+@given(choices(), text())
+def test_parse(choice, source):
+ l = get_language(choice)
+ parsed = p.parse(source, l)
+ for s in parsed:
+ assert {"code_text", "docs_text"} == set(s.keys())
+
+
+def test_skip_coding_directive():
+ source = "# -*- coding: utf-8 -*-\n" + FOO_FUNCTION
+ parsed = p.parse(source, PYTHON)
+ for section in parsed:
+ assert "coding" not in section['code_text']
+
+
+def test_multi_line_leading_spaces():
+ source = "# This is a\n# comment that\n# is indented\n"
+ source += FOO_FUNCTION
+ parsed = p.parse(source, PYTHON)
+ # The resulting comment has leading spaces stripped out.
+ assert parsed[0]["docs_text"] == "This is a\ncomment that\nis indented\n"
+
+
+@given(text(), text())
+def test_get_language_specify_language(source, code):
+ assert p.get_language(source, code, language="python") == p.languages['.py']
+
+ with pytest.raises(ValueError):
+ p.get_language(source, code, language="non-existent")
+
+
+@given(text() | none())
+def test_get_language_bad_source(source):
+ code = "#!/usr/bin/python\n"
+ code += FOO_FUNCTION
+ assert p.get_language(source, code) == PYTHON
+ with pytest.raises(ValueError) as e:
+ assert p.get_language(source, "badlang")
+
+ msg = "Can't figure out the language!"
+ try:
+ assert e.value.message == msg
+ except AttributeError:
+ assert e.value.args[0] == msg
+
+
+@given(text() | none())
+def test_get_language_bad_code(code):
+ source = "test.py"
+ assert p.get_language(source, code) == PYTHON
+
+
+@given(text(max_size=64))
+def test_ensure_directory(dir_name):
+ tempdir = os.path.join(tempfile.gettempdir(), str(int(time.time())), dir_name)
+
+ # Use sanitization from function, but only for housekeeping. We
+ # pass in the unsanitized string to the function.
+ safe_name = p.remove_control_chars(dir_name)
+
+ if not os.path.isdir(safe_name) and os.access(safe_name, os.W_OK):
+ p.ensure_directory(tempdir)
+ assert os.path.isdir(safe_name)
+
+# The following functions get good test coverage, but effort should be put into
+# decomposing the functions they test and actually testing their output.
+
+
+def test_generate_documentation():
+ p.generate_documentation(PYCCO_SOURCE, outdir=tempfile.gettempdir())
+
+
+@given(booleans(), choices())
+def test_process(preserve_paths, choice):
+ lang_name = choice([l["name"] for l in p.languages.values()])
+ p.process([PYCCO_SOURCE], preserve_paths=preserve_paths, outdir=tempfile.gettempdir(), language=lang_name)