.. _styleguide: Translate Styleguide ==================== The Translate styleguide is the styleguide for all Translate projects, including Translate Toolkit, Pootle, Virtaal and others. Patches are required to follow these guidelines. This Styleguide follows :pep:`8` with some clarifications. It is based almost verbatim on the `Flask Styleguide`_. .. _styleguide-general: General ------- Indentation: 4 real spaces, no tabs. Exceptions, modules that have copied into the source that don't follow this guideline. Maximum line length: 79 characters with a soft limit for 84 if absolutely necessary. Try to avoid too nested code by cleverly placing `break`, `continue` and `return` statements. Continuing long statements: To continue a statement you can use backslashes (preceeded by a space) in which case you should align the next line with the last dot or equal sign, or indent four spaces:: MyModel.query.filter(MyModel.scalar > 120) \ .order_by(MyModel.name.desc()) \ .limit(10) my_long_assignment = MyModel.query.filter(MyModel.scalar > 120) \ .order_by(MyModel.name.desc()) \ .limit(10) this_is_a_very_long(function_call, 'with many parameters') \ .that_returns_an_object_with_an_attribute If you break in a statement with parentheses or braces, align to the braces:: this_is_a_very_long(function_call, 'with many parameters', 23, 42, 'and even more') For lists or tuples with many items, break immediately after the opening brace:: items = [ 'this is the first', 'set of items', 'with more items', 'to come in this line', 'like this' ] Blank lines: Top level functions and classes are separated by two lines, everything else by one. Do not use too many blank lines to separate logical segments in code. Example:: def hello(name): print 'Hello %s!' % name def goodbye(name): print 'See you %s.' % name class MyClass(object): """This is a simple docstring""" def __init__(self, name): self.name = name def get_annoying_name(self): return self.name.upper() + '!!!!111' Expressions and Statements -------------------------- General whitespace rules: - No whitespace for unary operators that are not words (e.g.: ``-``, ``~`` etc.) as well on the inner side of parentheses. - Whitespace is placed between binary operators. Good:: exp = -1.05 value = (item_value / item_count) * offset / exp value = my_list[index] value = my_dict['key'] Bad:: exp = - 1.05 value = ( item_value / item_count ) * offset / exp value = (item_value/item_count)*offset/exp value=( item_value/item_count ) * offset/exp value = my_list[ index ] value = my_dict ['key'] Slice notation: While :pep:`8` calls for spaces around operators ``a = b + c`` this results in flags when you use ``a[b+1:c-1]`` but would allow the rather unreadable ``a[b + 1:c - 1]`` to pass. :pep:`8` is rather quiet on slice notation. - Don't use spaces with simple variables or numbers - Use brackets for expressions with spaces between binary operators Good:: a[1:2] a[start:end] a[(start - 1):(end + var + 2)] # Brackets help group things and don't hide the slice a[-1:(end + 1)] Bad:: a[start: end] # No spaces around : a[start-1:end+var+2] # Insanely hard to read, especially when your expressions are more complex a[start - 1:end + 2] # You lose sight of the fact that it is a slice a[- 1:end] # -1 is unary, no space .. note:: String slice formating is still under discussion. Comparisons: - against arbitrary types: ``==`` and ``!=`` - against singletons with ``is`` and ``is not`` (eg: ``foo is not None``) - never compare something with `True` or `False` (for example never do ``foo == False``, do ``not foo`` instead) Negated containment checks: use ``foo not in bar`` instead of ``not foo in bar`` Instance checks: ``isinstance(a, C)`` instead of ``type(A) is C``, but try to avoid instance checks in general. Check for features. If statements: - Use ``()`` brackets around complex if statements to allow easy wrapping, don't use backslash to wrap an if statements. - Wrap between ``and``, ``or``, etc. - Keep ``not`` with the expression - Use ``()`` alignment between expressions - Use extra ``()`` to eliminate abiguity, don't rely on an understanding of Python operator precedent rules. Good:: if length >= (upper + 2) if (length >= 25 and string != "Something" and not careful): do_something() Bad:: if length >= upper + 2: if (length... and string !=... Naming Conventions ------------------ .. note:: This has not been implemented or discussed. The Translate code is not at all consistent with these conventions. - Class names: ``CamelCase``, with acronyms kept uppercase (``HTTPWriter`` and not ``HttpWriter``) - Variable names: ``lowercase_with_underscores`` - Method and function names: ``lowercase_with_underscores`` - Constants: ``UPPERCASE_WITH_UNDERSCORES`` - precompiled regular expressions: ``name_re`` Protected members are prefixed with a single underscore. Double underscores are reserved for mixin classes. On classes with keywords, trailing underscores are appended. Clashes with builtins are allowed and **must not** be resolved by appending an underline to the variable name. If the function needs to access a shadowed builtin, rebind the builtin to a different name instead. Function and method arguments: - class methods: ``cls`` as first parameter - instance methods: ``self`` as first parameter - lambdas for properties might have the first parameter replaced with ``x`` like in ``display_name = property(lambda x: x.real_name or x.username)`` .. _styleguide-docs: Documentation ============= We use Sphinx_ to generate our API and user documentation. Read the `reStructuredText primer`_ and `Sphinx documentation`_ as needed. Special roles ------------- We introduce a number of special roles for documentation: * ``:bug:`` -- links to a bug in Translate's Bugzilla. * ``:bug:`123``` gives: :bug:`123` * ``:bug:`broken <123>``` gives: :bug:`broken <123>` * ``:opt:`` -- mark command options and command values. * ``:opt:`-P``` gives :opt:`-P` * ``:opt:`--progress=dots``` gives :opt:`--proress=dots` * ``:opt:`dots``` gives :opt:`dots` * ``:man:`` -- link to a Linux man page. * ``:man:`msgfmt``` gives :man:msgfmt Docstrings ---------- Docstring conventions: All docstrings are formatted with reStructuredText as understood by Sphinx. Depending on the number of lines in the docstring, they are laid out differently. If it's just one line, the closing triple quote is on the same line as the opening, otherwise the text is on the same line as the opening quote and the triple quote that closes the string on its own line:: def foo(): """This is a simple docstring.""" def bar(): """This is a longer docstring with so much information in there that it spans three lines. In this case the closing triple quote is on its own line. """ Please read :pep:`257` (Docstring Conventions) for a general overview, the important parts though are: - A docstring should have a brief one-line summary, ending with a period. - If there are more details there should be a blank line between the one-line summary and the rest of the text. Use pragraphs and formating as needed. - Use `reST field lists`_ to describe the input parameters and/or return types as the last part of the docstring. - Use proper capitalisation and punctuation. - Don't restate things that would appear in parameter descriptions. :: def foo(bar): """One line description. Further explanations that might be needed. :param bar: Parameter descriptions. """ :: def addunit(self, unit): """Appends the given unit to the object's list of units. This method should always be used rather than trying to modify the list manually. :type unit: TranslationUnit :param unit: Any object that inherits from :class:`TranslationUnit`. """ self.units.append(unit) Parameter documentation: Document parameters using `reST field lists`_ as follows:: def foo(bar): """Simple docstring :param bar: Something :type bar: Some type :return: Returns something :rtype: Return type """ Cross refencing code: When talking about other objects, methods, functions and variables it is good practice to cross-reference them with Sphinx's `Python cross-referencing`_. Other directives: Use `paragraph-level markup`_ when needed. .. note:: We still Need to gather the useful ones that we want you to use and how to use then. E.g. how to talk about a paramter in the docstring. How to reference classes in the module. How to reference other modules, etc. Module header: The module header consists of an utf-8 encoding declaration, copyright attribution, license block and a standard docstring:: # -*- coding: utf-8 -*- # ... LICENSE BLOCK... """A brief description""" .. """ package.module ~~~~~~~~~~~~~~ .. A brief description goes here. .. :copyright: (c) YEAR by AUTHOR. :license: LICENSE_NAME, see LICENSE_FILE for more details. """ Comments -------- General: - The ``#`` symbol (pound or hash) is used to start comments. - A space must follow the ``#`` between any written text. - Line length must be observed. - Inline comments are preceeded by two spaces. - Write sentences correctly: proper capitalisation and punctuation. Good:: # Good comment with space before and full sentence. statement # Good comment with two spaces Bad:: #Bad comment no space before statement # Bad comment, needs two spaces Docstring comments: Rules for comments are similar to docstrings. Both are formatted with reStructuredText. If a comment is used to document an attribute, put a colon after the opening pound sign (``#``):: class User(object): #: the name of the user as unicode string name = Column(String) #: the sha1 hash of the password + inline salt pw_hash = Column(String) .. _Flask Styleguide: http://flask.pocoo.org/docs/styleguide/ .. _reST field lists: http://sphinx.pocoo.org/domains.html#info-field-lists .. _Python cross-referencing: http://sphinx.pocoo.org/domains.html#cross-referencing-python-objects .. _Sphinx: http://sphinx.pocoo.org/ .. _reStructuredText primer: http://sphinx.pocoo.org/rest.html .. _Sphinx documentation: http://sphinx.pocoo.org/contents.html .. _paragraph-level markup: http://sphinx.pocoo.org/markup/para.html#paragraph-level-markup