Skip to main content

Pythonic interfaces

Project description

Build Status Documentation Status PyPI Version

Pythonic interface using decorators

Install

Implements is available on PyPI can be installed with pip.:

pip install implements

Advantages

  1. Favor composition over inheritance.

  2. Inheriting from multiple classes can be problematic, especially when the superclasses have the same method name but different signatures. Implements will throw a descriptive error if that happens to ensure integrity of contracts.

  3. The decorators are evaluated at import time. Any errors will be raised then and not when an object is instantiated or a method is called.

  4. It’s cleaner. Using decorators makes it clear we want share behavior. Also, arguments are not allowed to be renamed.

  5. Codebase is tiny: you can just copy the file over. This repo exists more for test coverage.

Usage

Consider the implementation using inheritance:

class Duck:
    def __init__(self, age):
        self.age = age

class Flyable:
    @staticmethod
    def migrate(direction):
        raise NotImplementedError("Flyable is an abstract class")

    def fly(self) -> str:
        raise NotImplementedError("Flyable is an abstract class")


class Quackable:
    def fly(self) -> bool:
        raise NotImplementedError("Quackable is an abstract class")

    def quack(self):
        raise NotImplementedError("Quackable is an abstract class")


class MallardDuck(Duck, Quackable, Flyable):

    def __init__(self, age):
        super(MallardDuck, self).__init__(age)
        pass

    def migrate(self, dir):
        return True

    def fly(self):
        pass

A couple drawbacks implementing it this way:

  1. It’s unclear without checking each parent class where super is being called.

  2. Similarly the return types of fly in Flyable and Quackable are different. Someone unfamiliar with Python would have to read up on Method Resolution Order.

  3. We would only get a NotImplementedError when calling quack which can happen much later during runtime. Also, raising NotImplementedError everywhere looks clunky.

  4. The writer of MallardDuck made method migrate an instance method and renamed the argument to dir which is confusing.

  5. We really want to be differentiating between behavior and inheritance.

The advantage of using implements is it looks cleaner and you would get errors at import time instead of when the method is actually called.

In the above example we would rewrite everything as:

from implements import Interface, implements


class Duck:
    def __init__(self, age):
        self.age = age


class Flyable(Interface):
    def fly(self) -> str:
        pass

    @staticmethod
    def migrate(direction):
        pass


class Quackable(Interface):
    def fly(self) -> bool:
        pass

    def quack(self):
        pass


@implements(Flyable)
@implements(Quackable)
class MallardDuck(Duck):
    def __init__(self, age):
        super(MallardDuck, self).__init__(age)

    def migrate(self, dir):
        return True

    def fly(self):
        pass

The above would now throw the following errors:

NotImplementedError: 'MallardDuck' must implement method 'fly((self) -> bool)' defined in interface 'Quackable'
NotImplementedError: 'MallardDuck' must implement method 'quack((self))' defined in interface 'Quackable'
NotImplementedError: 'MallardDuck' must implement method 'migrate((direction))' defined in interface 'Flyable'

We can solve the errors by rewriting for example as:

class Quackable(Interface):
    def fly(self) -> str:
        pass

    def quack(self):
        pass

@implements(Flyable)
@implements(Quackable)
class MallardDuck(Duck):
    def __init__(self, age):
        super(MallardDuck, self).__init__(age)

    @staticmethod
    def migrate(direction):
        pass

    def fly(self) -> str:
        pass

    def quack(self):
        pass

Credit

Implementation was inspired by a PR of @elifiner.

Test

Running unit tests:

make test

Running linter:

make lint

Running tox:

make test-all

License

MIT

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

implements-0.1.1.tar.gz (14.5 kB view hashes)

Uploaded Source

Built Distribution

implements-0.1.1-py2.py3-none-any.whl (5.4 kB view hashes)

Uploaded Python 2 Python 3

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page