Skip to content

Commit

Permalink
Merge pull request #14 from joncotton/v2.0-prep
Browse files Browse the repository at this point in the history
v2.0 prep
  • Loading branch information
joncotton committed Mar 31, 2014
2 parents c86da50 + e94c594 commit 0ec87b1
Show file tree
Hide file tree
Showing 23 changed files with 908 additions and 532 deletions.
7 changes: 4 additions & 3 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -1,9 +1,10 @@
*.pyc
*~
*.swp
coverage/*
build/*
*.egg-info
coverage*/
build/
mydatabase
docs/_build/*
docs/_build/
MANIFEST
dist/
66 changes: 66 additions & 0 deletions CHANGES.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
CHANGES
=======

2.0 (unreleased)
------------------

- **Backward incompatible:** ``env_settings.py`` is required and contains
the settings that ``fabfile.py`` used to have. It is written in the normal
Django settings.py style of variable assignments.

- **Backward incompatible:** ``tasks.py`` is required and replaces the
previously required ``fabfile.py``. It no longer defines Django settings
(now handled in ``env_settings.py``).

- **Backward incompatible:** switched from Fabric to Invoke, so the ``fab``
command is replaced with ``invoke``.

- **Backward incompatible:** Lettuce testing support is gone.

- **Backward incompatible:** the Sphinx documentation command ``docs`` is gone.

- **Backward incompatible:** remove ``assertInContext``, ``assertNone``,
``assertIsA`` and ``assertDoesNotHave``. Some of these duplicated native
Python methods and the potential confusion was greater than the benefit.

- **Backward incompatible:** individual components need to specify the
``fudge`` requirement if they need it.

- **Backward incompatible:** Fix ``generate_random_users()`` and turn it
into a generator.

- **Backward incompatible:** remove ``concrete`` decorators. The Armstrong
standard practice is to use "support" models when necessary in testing,
which are much easier to use and understand.

- Setuptools is explictly used. This is not a backwards incompatible change
because anything installed with Pip was automatically and transparently
using Setuptools/Distribute *anyway*. We rely on setup kwargs that Distutils
doesn't support and that only worked because of Pip's behind the scenes swap.
This allowed us to remove boilerplate and better prepares us for Python 3
and perhaps even more simplifying refactors. Functionally though, this
doesn't change anything.

- Drop the atypical VirtualDjango in favor of the ``settings.configure()``
Django bootstrapping method.

- Bare minimum package requirements for as-fast-as-possible virtualenv
creation. Even Invoke is optional when running tests. Individual tasks
can specify package requirements and will nicely message their needs if
the package is not installed.

- Run any Django ``manage.py`` command from import or the CLI with
component-specific settings bootstrapped in.

- Run tests with arguments. Use any args that ``manage.py test`` accepts
to run only specific test cases, change output verbosity, etc.

- Coverage testing is ready for multiple environments at once (like with Tox).

- Use (and backport) the Django 1.6 test runner. This standardizes testing
in favor of the newest method so we don't need to be cognizant of the current
Django version as we test across multiple versions. Bonus: because the new
runner is explicit about test discovery, drop the ``TESTED_APP`` code.

- New ``remove_armstrong`` task command to uninstall every Armstrong component
(except for ArmDev).
3 changes: 2 additions & 1 deletion MANIFEST.in
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
include LICENSE
include README.rst
include CHANGES.rst
include package.json
include armstrong/cli/templates/standard/requirements/*.txt
prune build/*
198 changes: 152 additions & 46 deletions README.rst
Original file line number Diff line number Diff line change
Expand Up @@ -3,78 +3,184 @@ armstrong.dev
Tools and such for handling development of Armstrong applications

This package contains some of the various helpers needed to do development work
on the Armstrong packages. If you're not actively developing, or working with
on the Armstrong packages. If you're not actively developing, or working with
development versions of Armstrong, you probably don't need this package.


Installation & Configuration
----------------------------
If you are just running tests for a component, Tox will grab everything it
needs including ArmDev.

- ``pip install tox`` and run ``tox``

Otherwise:

- ``pip install armstrong.dev invoke``

`Invoke`_ is not strictly required. ArmDev is as lean as possible to support
fast virtualenv creation so multi-environment testing tools like TravisCI
and Tox will complete ASAP.

Many of the Invoke tasks have their own package requirements and they will
nicely notify you if something they require needs to be installed.

**Optional Settings:** (Used in ``env_settings.py``)

``COVERAGE_EXCLUDE_FILES = ['*/migrations/*']``
A list of filename patterns for files to exclude during coverage testing.
Individual components are free to extend or replace this setting.

.. _Invoke: http://docs.pyinvoke.org/en/latest/index.html


Usage
-----
Most Armstrong components already have the necessary configuration to use these
Dev tools. Specifically, components need ``tasks.py`` and ``env_settings.py``
files. Assuming these are present:

Create a `fabfile` (either `fabfile/__init__.py` or simply `fabfile.py`) in
your project and add the following::
``invoke --list``
to see a list of all available commands

from armstrong.dev.tasks import *
``invoke --help <cmd>``
for help on a specific command

Several of the tasks take an optional ``--extra`` argument that is used as a
catch-all way of passing arbitrary arguments to the underlying command. Invoke
cannot handle arbitrary args (like Fabric 1.x could) so this is our workaround.
Two general rules: 1) enclose multiple args in quotes 2) kwargs need to use
"=" with no spaces (our limitation, not Invoke's). Example:
``invoke test --extra "--verbosity=2 <path.to.app.test1> <path.to.app.test2>"``

settings = {
"DEBUG": True,
# And so on with the keys being the name of the setting and the values
# the appropriate value.
}
``invoke install [--editable]``
to "pip install" the component, by default as an `editable`_ install. For
a regular install, use ``--no-editable`` or ``--editable=False``.

main_app = "name.of.your.app"
tested_apps ("another_app", main_app, )
``invoke test [--extra ...]``
to run tests where --extra handles anything the normal Django
"manage.py test" command accepts.

``invoke coverage [--reportdir=<directory>] [--extra ...]``
for running test coverage. --extra works the same as in "invoke test" passing
arbitrary args to the underlying test command. --reportdir is where the HTML
report will be created; by default this directory is named "coverage".

Now your fabfile will expose the various commands for setting up and running
your reusable app inside a virtualenv for testing, interacting with via the
shell, and even running a simple server.
``invoke managepy <cmd> [--extra ...]``
to run any Django "manage.py" command where --extra handles any arbitrary
args. Example: ``invoke managepy shell`` or
``invoke managepy runserver --extra 9001``

Type ``fab -l`` to see a list of all of the commands.
``invoke create_migration [--initial]``
to create a South migration for the component. An "auto" migration is
default if the --initial flag is not used.

There are other commands as well, but these are the most useful. Remember
that individual components may provide additional Invoke tasks as well. So
run ``invoke --list`` to discover them all.

Installation
------------

::
.. _editable: http://pip.readthedocs.org/en/latest/reference/pip_install.html#editable-installs

name="armstrong.dev"
pip install -e git://github.com/armstrong/$name#egg=$name

**Note**: This currently relies on a development version of Fabric. This
requirement is set to be dropped once Fabric 1.1 is released. To ensure this
runs as expected, install the ``tswicegood/fabric`` fork of Fabric:
Component Setup
---------------
If you are creating a new Armstrong component or updating one that uses the
pre-2.0 ArmDev, you'll need to create (or port to) these two files:

::
1. Create a ``tasks.py`` and add the following::

pip install -e git://github.com/tswicegood/fabric.git#egg=fabric
from armstrong.dev.tasks import *

# any additional Invoke commands
# ...

Contributing
------------
2. Create an ``env_settings.py`` and add the following::

* Create something awesome -- make the code better, add some functionality,
whatever (this is the hardest part).
* `Fork it`_
* Create a topic branch to house your changes
* Get all of your commits in the new topic branch
* Submit a `pull request`_
from armstrong.dev.default_settings import *

# any additional settings
# it's likely you'll need to extend the list of INSTALLED_APPS
# ...

License
-------
Copyright 2011 Bay Citizen and Texas Tribune
Not required but as long as you are reviewing the general state of things,
take care of these too!

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
- Review the ``requirements`` files
- Review the TravisCI configuration
- Drop Lettuce tests and requirements
- Add a ``tox.ini`` file
- Review the README text and setup.py metadata
- Use Setuptools and fix any improper namespacing
- Stop shipping tests by moving tests/ to the root directory
- If the component uses logging, consider namespacing it with
``logger = logging.getLogger(__name__)``.
- Add a ``CHANGES.rst`` file and include it in the MANIFEST
- Review ``.gitignore``. You might want to ignore these::

http://www.apache.org/licenses/LICENSE-2.0
.tox/
coverage*/
*.egg-info

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.

Notable changes in 2.0
----------------------
Setuptools is now explicitly used/required instead of Distutils.

Invoke replaces Fabric for a leaner install without the SSH and crypto
stuff. Invoke is still pre-1.0 release so we might have some adjustment
to do later.

This version offers an easier and more standard way to run a Django
environment with a component's specific settings, either from the
commandline or via import.

It provides an "a la carte" requirements approach. Meaning that if you run an
Invoke command that needs a package that isn't installed, it will prompt you
to install it instead of requiring everything up-front. This allows for much
faster virtualenv creation (which saves considerable time in testing) and
doesn't pollute your virtualenv with packages for features you don't use.

``test`` and ``coverage`` will work better with automated test tools like
TravisCI and Tox. These commands also now work like Django's native test
command so that you can pass arguments for running selective tests or
changing the output verbosity.

Settings are now defined in the normal Django style in an ``env_settings.py``
file instead of as a dict within the tasks file. It's not called "settings.py"
to make it clearer that these are settings for the development and testing
of this component, not necessarily values to copy/paste for incorporating
the component into other projects.

The full list of changes and backward incompatibilties is available
in **CHANGES.rst**.


Contributing
------------
Development occurs on Github. Participation is welcome!

* Found a bug? File it on `Github Issues`_. Include as much detail as you
can and make sure to list the specific component since we use a centralized,
project-wide issue tracker.
* Have code to submit? Fork the repo, consolidate your changes on a topic
branch and create a `pull request`_.
* Questions, need help, discussion? Use our `Google Group`_ mailing list.

.. _Github Issues: https://github.com/armstrong/armstrong/issues
.. _pull request: http://help.github.com/pull-requests/
.. _Fork it: http://help.github.com/forking/
.. _Google Group: http://groups.google.com/group/armstrongcms


State of Project
----------------
`Armstrong`_ is an open-source news platform that is freely available to any
organization. It is the result of a collaboration between the `Texas Tribune`_
and `The Center for Investigative Reporting`_ and a grant from the
`John S. and James L. Knight Foundation`_. Armstrong is available as a
complete bundle and as individual, stand-alone components.

.. _Armstrong: http://www.armstrongcms.org/
.. _Texas Tribune: http://www.texastribune.org/
.. _The Center for Investigative Reporting: http://cironline.org/
.. _John S. and James L. Knight Foundation: http://www.knightfoundation.org/
3 changes: 1 addition & 2 deletions armstrong/__init__.py
Original file line number Diff line number Diff line change
@@ -1,2 +1 @@
from pkgutil import extend_path
__path__ = extend_path(__path__, __name__)
__import__('pkg_resources').declare_namespace(__name__)
2 changes: 0 additions & 2 deletions armstrong/dev/__init__.py
Original file line number Diff line number Diff line change
@@ -1,2 +0,0 @@
from pkgutil import extend_path
__path__ = extend_path(__path__, __name__)
53 changes: 53 additions & 0 deletions armstrong/dev/default_settings.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
"""
Default settings for Armstrong components running in a dev/test environment
A component may (and might have to) override or supply additional settings
by creating an `env_settings.py` file in its root directory that imports
from this file.
from armstrong.dev.default_settings import *
"""
# Since we are using configure() we need to manually load the defaults
from django.conf.global_settings import *

# Grab our package information
import json
package = json.load(open("./package.json"))
app_name = package['name'].rsplit('.', 1)[1]

#
# Armstrong default settings
#
DEBUG = True
INSTALLED_APPS = [package['name']]
DATABASES = {
"default": {
"ENGINE": "django.db.backends.sqlite3",
"NAME": 'mydatabase'
}
}
TEST_RUNNER = "armstrong.dev.tests.runner.ArmstrongDiscoverRunner"

COVERAGE_EXCLUDE_FILES = ['*/migrations/*']

# Add a DEBUG console "armstrong" logger
LOGGING = {
'version': 1,
'disable_existing_loggers': False,
'formatters': {
'basic': {'format': '%(levelname)s %(name)s--%(message)s'}
},
'handlers': {
'console': {
'class': 'logging.StreamHandler',
'formatter': 'basic'
}
},
'loggers': {
'armstrong': {
'level': 'DEBUG',
'handlers': ['console']
}
}
}
Loading

0 comments on commit 0ec87b1

Please sign in to comment.