Metadata-Version: 2.1
Name: simple-specification
Version: 0.3
Summary: A simple pythonic implementation of the Specification Pattern for Python 3.7 or higher.
Home-page: https://github.com/victor-o-silva/simple-specification
Author: Victor Oliveira da Silva
Author-email: victor_o_silva@hotmail.com
License: UNKNOWN
Download-URL: https://github.com/victor-o-silva/simple-specification/tarball/0.3
Platform: UNKNOWN
Classifier: License :: OSI Approved :: MIT License
Classifier: Programming Language :: Python :: 3.7
Description-Content-Type: text/x-rst

========================
simple_specification
========================

.. image:: https://travis-ci.org/victor-o-silva/simple-specification.svg?branch=master
   :target: https://travis-ci.org/victor-o-silva/simple-specification
   :alt: Build Status

.. image:: https://coveralls.io/repos/victor-o-silva/simple-specification/badge.svg?branch=master
   :target: https://coveralls.io/r/victor-o-silva/simple-specification?branch=master
   :alt: Code Coverage

.. image:: https://badge.fury.io/py/simple-specification.svg
   :target: https://badge.fury.io/py/simple-specification


A simple pythonic implementation of the `Specification Pattern <https://www.martinfowler.com/apsupp/spec.pdf>`_ for Python 3.7 or higher.

Installation
############

.. code-block:: sh

    pip install simple-specification

Examples
##########

Creating specification classes to check integer numbers
*******************************************************

.. code-block:: python

    from simple_specification import Specification


    # hardcoded specification
    class Odd(Specification):
        def is_satisfied_by(self, candidate: int) -> bool:
            return candidate % 2 != 0


    # parametrized specification
    class DivisibleBy(Specification):
        def __init__(self, number: int):
            self.number = number

        def is_satisfied_by(self, candidate: int) -> bool:
            return candidate % self.number == 0

Instantiating the specifications
********************************

.. code-block:: python

    >>> odd = Odd()
    >>> even = ~odd
    >>> div3 = DivisibleBy(3)
    >>> div5 = DivisibleBy(5)

Using specifications - method ``is_satisfied_by(candidate)``
************************************************************

.. code-block:: python

    >>> odd.is_satisfied_by(1)
    True
    >>> odd.is_satisfied_by(2)
    False
    >>> even.is_satisfied_by(3)
    False
    >>> even.is_satisfied_by(4)
    True
    >>> div3.is_satisfied_by(9)
    True
    >>> div5.is_satisfied_by(12)
    False

Composing specifications - operators ``&`` and ``|``
****************************************************

.. code-block:: python

    >>> # AND
    >>> (even & div3).is_satisfied_by(6)
    True
    >>> (even & div3).is_satisfied_by(9)
    False
    >>> (even & div3 & div5).is_satisfied_by(30)
    True
    >>> (even & div3 & div5).is_satisfied_by(36)
    False
    >>>
    >>> # OR
    >>> (even | div3).is_satisfied_by(10)
    True
    >>> (even | div3).is_satisfied_by(15)
    True
    >>> (even | div3).is_satisfied_by(5)
    False

Negating specifications - operator ``~``
****************************************

.. warning:: Always surround the negation with parenthesis

.. code-block:: python

    >>> (~odd).is_satisfied_by(2)
    True
    >>> (~div5).is_satisfied_by(25)
    False
    >>> (~div5).is_satisfied_by(11)
    True
    >>> (~(odd | div5)).is_satisfied_by(6)
    True
    >>> (~(odd | div5)).is_satisfied_by(7)
    False

Composing a list of specifications with ``&`` - class method ``Specification.all(specs)``
********************************************************************************************

.. code-block:: python

    >>> all_spec = Specification.all([~even, div3, div5])
    >>> all_spec.is_satisfied_by(15)
    True
    >>> all_spec.is_satisfied_by(30)
    False
    >>> all_spec.is_satisfied_by(25)
    False
    >>> all_spec.is_satisfied_by(6)
    False

Composing a list of specifications with ``|`` - class method ``Specification.any(specs)``
**********************************************************************************************

.. code-block:: python

    >>> div15 = DivisibleBy(3) & DivisibleBy(5)
    >>> any_spec = Specification.any([div15, even])
    >>> any_spec.is_satisfied_by(2)
    True
    >>> any_spec.is_satisfied_by(15)
    True
    >>> any_spec.is_satisfied_by(30)
    True
    >>> any_spec.is_satisfied_by(3)
    False
    >>> any_spec.is_satisfied_by(5)
    False


