Metadata-Version: 2.4
Name: handy
Version: 0.7
Summary: A collection of tools to make your django life easier.
Home-page: http://github.com/Suor/handy
Author: Alexander Schepanovski
Author-email: suor.web@gmail.com
License: BSD
Classifier: Development Status :: 4 - Beta
Classifier: License :: OSI Approved :: BSD License
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python
Classifier: Programming Language :: Python :: 2
Classifier: Programming Language :: Python :: 2.7
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.4
Classifier: Programming Language :: Python :: 3.5
Classifier: Programming Language :: Python :: 3.6
Classifier: Programming Language :: Python :: Implementation :: CPython
Classifier: Programming Language :: Python :: Implementation :: PyPy
Classifier: Framework :: Django
Classifier: Framework :: Django :: 1.7
Classifier: Framework :: Django :: 1.8
Classifier: Framework :: Django :: 1.9
Classifier: Framework :: Django :: 1.10
Classifier: Framework :: Django :: 1.11
Classifier: Framework :: Django :: 2.0
Classifier: Environment :: Web Environment
Classifier: Intended Audience :: Developers
Classifier: Topic :: Internet :: WWW/HTTP
Classifier: Topic :: Software Development :: Libraries :: Python Modules
License-File: LICENSE
Requires-Dist: Django>=1.2
Dynamic: author
Dynamic: author-email
Dynamic: classifier
Dynamic: description
Dynamic: home-page
Dynamic: license
Dynamic: license-file
Dynamic: requires-dist
Dynamic: summary

Handy
=====

A collection of different tools, shortcuts, decorators, form and model fields
to make your django life easier.


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

::

    pip install handy


Overview
--------

Here are quick overview of what you can do with handy.
You can also read `full docs <http://handy.readthedocs.org/>`_.

Avoid ``HttpResponse`` and ``render_to_response()`` biolerplate with ``@render_to()`` decorator.
This one will render result dict to ``'app_name/foo.html'``:

.. code:: python

    @render_to()
    def foo(request):
        return {
            'bar': Bar.objects.all()
            # You can easily override default template, content type,
            # status code and add cookies to response:
            'STATUS': 410,
            'CONTENT_TYPE': 'text/plain'
        }


Easy JSON responders with ``@render_to_json()`` decorator:

.. code:: python

    @render_to_json()
    def posts_by_tag(request, tag=None):
        posts = Post.object.values().filter(tag=tag)
        return list(posts)


And higher order ``@ajax`` decorator to handle more complex asynchronous actions:

.. code:: python

    @ajax
    @ajax.login_required
    @ajax.catch(Post.DoesNotExist)
    def enable_post(request):
        post = Post.objects.get(pk=request.GET['id'])

        if post.author != request.user:
            raise ajax.error('permission_denied')

        post.enabled = True
        post.save()


Send emails rendered from templates:

.. code:: python

    render_to_email(article.author.email, 'approved.html', {'article': article})


A collection of model fields with accompanying form fields and widgets. Most notably different array fields to store array of values or choices:

.. code:: python

    DAYS = zip(range(7), 'Sun Mon Tue Wed Thu Fri Sat'.split())

    class Company(models.Model):
        phones = StringArrayField('Phone numbers', blank=True, default=lambda: [])
        workdays = IntegerArrayField('Work days', choices=DAYS)

    company = Company(phones=['234-5016', '516-2314'], workdays=[1,2,3,4])
    company.save()

In model form ``phones`` field would be represented as ``CommaSeparatedInput`` and
``workdays`` as multiple checkboxes:

.. code:: python

    class CompanyForm(forms.ModelForm):
        class Meta:
            model = Company


A handy ``@cached_property`` utility:

.. code:: python

    class UserProfile(models.Model):
        phones = models.ManyToManyField(Phone)

        ...

        @cached_property
        def public_phones(self):
            # this queryset will be constructed only once, thanks to @cached_property
            return self.phones.filter(public=True)


A bunch of database querying utilities to not tinker with ``cursor()`` anymore:

.. code:: python

    best_offer = fetch_val('select max(value) from offers where ...')

    for row in fetch_named('select * from blog_post'):
        print row.id, row.title


And a middleware to make your html output slimmer by stripping out unnecessary spaces:

.. code:: python

    MIDDLEWARE_CLASSES = (
        ...
        'handy.middleware.StripWhitespace',
    )


And more:

- simple logger wrap up
- ``CommaSeparatedInput`` and ``MultilineInput`` widgets for array fields
- ``JSONField``, ``AdditionalAutoField`` and ``BigAutoField``
- and a bunch of database, email, text and debugging utilities

NOTE: functional tools, namely ``@memoize`` and ``@cache``, moved to
`funcy <https://github.com/Suor/funcy>`_


How you can help
----------------

- give me any feedback. What bits are most useful? What can be added or changed?
- bring your ideas and/or code that can make all our django experience more fun


TODO
----

- docs for text and just utils
- add support for querysets and models in JSON decorators
