Metadata-Version: 2.1
Name: django-test-tools
Version: 2.2.1
Summary: Simple tests tools to make testing faster and easier.
Home-page: https://github.com/luiscberrocal/django-test-tools
Author: Luis Carlos Berrocal
Author-email: luis.berrocal.1942@gmail.com
License: MIT
Keywords: django-test-tools
Classifier: Development Status :: 5 - Production/Stable
Classifier: Environment :: Web Environment
Classifier: Framework :: Django
Classifier: Framework :: Django :: 3.2
Classifier: Framework :: Django :: 4.2
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: MIT License
Classifier: Natural Language :: English
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
License-File: LICENSE
License-File: AUTHORS.rst
Requires-Dist: Django
Requires-Dist: Jinja2 (>=3.0.1)
Requires-Dist: openpyxl (>=3.0.7)
Requires-Dist: requests (>=2.26.0)
Requires-Dist: pydantic (<2.0,>=1.10)

=============================
Django Test Tools
=============================

.. image:: https://badge.fury.io/py/django-test-tools.svg
    :target: https://badge.fury.io/py/django-test-tools

.. image:: https://codecov.io/gh/luiscberrocal/django-test-tools/branch/master/graph/badge.svg
    :target: https://codecov.io/gh/luiscberrocal/django-test-tools

.. image:: https://pyup.io/repos/github/luiscberrocal/django-test-tools/shield.svg
     :target: https://pyup.io/repos/github/luiscberrocal/django-test-tools/
     :alt: Updates

.. image:: https://readthedocs.org/projects/django-test-tools/badge/?version=latest
    :target: https://django-test-tools.readthedocs.io/en/latest/?badge=latest
    :alt: Documentation Status

Simple tests tools to make testing faster and easier. Most of the tools are to do a quick scaffolding for tests.

The tools presume a naming convention:

- **Tests:** Are named with the convention **TestCaseModelName**. For a model named *Poll* the test would be generated
  as the testing class would be *TestCasePoll*
- **Factories:** Are named with the convention **ModelName**. For a model named *Poll* the test would be generated
  as the testing class would be *PollFactory*
- **Serializers:** Are named with the convention **TestCaseSerializer**. For a model named *Poll* the test would be generated
  as the testing class would be *PollSerializer*


Compatibility matrix:

+----------------+---------------+--------------+--------------+
| Python version | Django 1.11.x | Django 2.2.x | Django 3.0.x |
+----------------+---------------+--------------+--------------+
|       3.7      |       x       |       x      |       x      |
+----------------+---------------+--------------+--------------+
|       3.6      |       x       |       x      |       x      |
+----------------+---------------+--------------+--------------+

Documentation
-------------

The full documentation is at https://django-test-tools.readthedocs.io.

Quickstart
----------

Install Django Test Tools:

.. code-block:: bash

    pip install django-test-tools


In your settings.py file add it to your `INSTALLED_APPS`

.. code-block:: python

    INSTALLED_APPS = (
        ...
        'django_test_tools.apps.DjangoTestToolsConfig',
        ...
    )


Create an output folder in the root folder of you project, name it what ever you want, and add the settings
variable **TEST_OUTPUT_PATH** pointing to it. Make sure to add this folder to your **.gitignore** file.

.. code-block:: python

    import environ

    ROOT_DIR = (
        environ.Path(__file__) - 3
    )  # (my_project/config/settings/base.py - 3 = alpha_clinic/)
    APPS_DIR = ROOT_DIR.path("my_project")
    TEST_OUTPUT_PATH = ROOT_DIR.path("output").root

Features
--------

Factory Generator
++++++++++++++++++

To create `Factory Boy`_ style factories.

For a django project named polling_app with an app name poll the following command will generate the scaffolding for
the tests for all the models in th app polls.

.. code-block:: bash

    $  python manage.py generate_factories polling_app.polls


For the following models


.. code-block:: python

    class OperatingSystem(models.Model):
        name = models.CharField(max_length=20)
        version = models.CharField(max_length=5)
        licenses_available = models.IntegerField()
        cost = models.DecimalField(decimal_places=2, max_digits=7)

        class Meta:
            unique_together = ('name', 'version')


    class Server(models.Model):
        PRODUCTION = 'PROD'
        DEVELOPMENT = 'DEV'
        USE_CHOICES = ((PRODUCTION, 'Prod'),
                       (DEVELOPMENT, 'Dev'))
        name = models.CharField(max_length=20, unique=True)
        notes = models.TextField()
        virtual = models.BooleanField()
        ip_address = models.GenericIPAddressField()
        created = models.DateTimeField()
        online_date = models.DateField()
        operating_system = models.ForeignKey(OperatingSystem, related_name='servers', on_delete=models.CASCADE)
        server_id = models.CharField(max_length=6)
        use = models.CharField(max_length=4, choices=USE_CHOICES, default=DEVELOPMENT)
        comments = models.TextField(null=True, blank=True)



running `python manage.py generate_factories example.servers > ./output/factories.py` will create the following factories

.. code-block:: python

    import string

    from random import randint
    from pytz import timezone

    from django.conf import settings

    from factory import Iterator
    from factory import LazyAttribute
    from factory import SubFactory
    from factory import lazy_attribute
    from factory.django import DjangoModelFactory, FileField
    from factory.fuzzy import FuzzyText, FuzzyInteger
    from faker import Factory as FakerFactory

    from example.servers.models import OperatingSystem, Server

    faker = FakerFactory.create()


    class OperatingSystemFactory(DjangoModelFactory):
        class Meta:
            model = OperatingSystem

        name = LazyAttribute(lambda x: faker.text(max_nb_chars=20))
        version = LazyAttribute(lambda x: faker.text(max_nb_chars=5))
        licenses_available = LazyAttribute(lambda o: randint(1, 100))
        cost = LazyAttribute(lambda x: faker.pydecimal(left_digits=5, right_digits=2, positive=True))

    class ServerFactory(DjangoModelFactory):
        class Meta:
            model = Server

        name = LazyAttribute(lambda x: faker.text(max_nb_chars=20))
        notes = LazyAttribute(lambda x: faker.paragraph(nb_sentences=3, variable_nb_sentences=True))
        virtual = Iterator([True, False])
        ip_address = LazyAttribute(lambda o: faker.ipv4(network=False))
        created = LazyAttribute(lambda x: faker.date_time_between(start_date="-1y", end_date="now",
                                                               tzinfo=timezone(settings.TIME_ZONE)))
        online_date = LazyAttribute(lambda x: faker.date_time_between(start_date="-1y", end_date="now",
                                                               tzinfo=timezone(settings.TIME_ZONE)))
        operating_system = SubFactory(OperatingSystemFactory)
        server_id = LazyAttribute(lambda x: FuzzyText(length=6, chars=string.digits).fuzz())
        use = Iterator(Server.CHOICES, getter=lambda x: x[0])
        comments = LazyAttribute(lambda x: faker.paragraph(nb_sentences=3, variable_nb_sentences=True))

Important the use attribute is created incorrectly. **When you use choices you need to manually change it** to USE_CHOICES.

.. code-block:: python

        use = Iterator(Server.USE_CHOICES, getter=lambda x: x[0])


Model Test Case Generator
+++++++++++++++++++++++++

.. code-block:: bash

    $  python manage.py generate_model_test_cases project.app

Serializer Generator
++++++++++++++++++++

.. code-block:: bash

    $ python manage.py generate_serializers project.app -s ModelSerializer

Writing assertions
+++++++++++++++++++

One of the most boring steps of writing tests is checking the content of static dictionaries.
You now the content of the dictionary and you need to compare it with a result.

In the following code the assertions for the **data** dictionary will be generated in the
**fn** file. IMPORTANT the name of the dictionary in this case **data** must be the same as
the second argument of the method in order to generate the correct assertions.

.. code-block:: python

    fn = './output/_my_assertions.py'
    data = [
            {'name': 'kilo', 'password': 9999,
             'groups': ['admin', 'users'],
             'config': {'server': 'all', 'bulding': 116}},
            {'name': 'pasto', 'password': 'nogo',
             'groups': ['users'],
             'config': {'server': 'database', 'bulding': None},
             'created_date': date(2016, 1, 3),
             'modified': '2016-10-01'}
        ]
        filename = write_assertions(data, 'data', filename=fn, type_only=True,excluded_keys=['config']))

The result of this script:

.. code-block:: python

    self.assertEqual(len(data), 2)
    self.assertEqual(len(data[0]['groups']), 2)
    self.assertEqual(data[0]['groups'][0], 'admin')
    self.assertEqual(data[0]['groups'][1], 'users')
    self.assertEqual(data[0]['name'], 'kilo')
    self.assertEqual(data[0]['password'], 9999)
    self.assertEqual(len(data[1]['groups']), 1)
    self.assertEqual(data[1]['groups'][0], 'users')
    self.assertEqual(data[1]['name'], 'pasto')
    self.assertEqual(data[1]['password'], 'nogo')

Running Tests
-------------

Does the code actually work?

::

    source <YOURVIRTUALENV>/bin/activate
    (myenv) $ python runtests.py tests


Pushing code to Pypi
--------------------
1. Setup environment

  .. code-block:: bash

    source ./venv/bin/activate


2. Updated version. Instead of patch you could also use **major** o **minor** depending on the level of the release.

  .. code-block:: bash

    $ make patch


3. Check the .travis.yml to make sure the versions of Django are the latests. Check in https://www.djangoproject.com/download/
   for the latest versions.

4. Check setup.py for Django and Python versions.

5. Close the git-flow release manually.

6. Push to repo, Travis CI should deploy to pypi

  .. code-block:: bash

    make travis-push

Credits
-------

Tools used in rendering this package:

*  Cookiecutter_
*  `cookiecutter-djangopackage`_

.. _Cookiecutter: https://github.com/audreyr/cookiecutter
.. _`cookiecutter-djangopackage`: https://github.com/pydanny/cookiecutter-djangopackage
.. _`Factory Boy`: https://factoryboy.readthedocs.io/en/latest/




History
-------

0.1.0 (2017-04-26)
++++++++++++++++++

* First release on PyPI.
