Skip to main content

An implementation of the Optional object in Python

Project description

Optional.py

Build Status test lint typecheck format editorconfig License Python Versions Contributors Open Source Love Downloads

An Implementation of the Optional Object for Python

Why

There is a difference between None as Empty and None as the result for an Error. A common bad practice is to return None to indicate the absence of something. Doing this introduces ambiguity into you code.

For example:

thing = stuff.getSomeThing().getAnotherThing()

What will happen if the result from getSomeThing returns None? We will get an AttributeError: 'NoneType' object has no attribute 'getAnotherThing'.

What can you do to prevent these kinds of exceptions? You can write defensively:

something = stuff.getSomeThing()
if something is not None:
    thing = something.getAnotherThing()

However, if we add to our chain, you can imagine how the nesting of defensive checks adds up quickly. These defensive checks obfuscate our actual business logic, decreasing readability. Furthermore, defensive checking is an error prone process, because it is easy to forget to check a required condition.

So we present you with an Optional object as an alternative.

Install

Compatible with Python 3.10 and up!

pip install optional.py

Usage

  1. You can import it using:

    from optional import Nothing, Option, Optional, Something
    
  2. You can set it to empty:

    instead of: :scream_cat:

    return None
    

    you can do: :smile_cat:

    return Optional.empty()
    

    or

    return Optional.of()
    
  3. You can set it to have content:

    instead of: :scream_cat:

    return "thing"
    

    you can do: :smile_cat:

    return Optional.of("thing")
    
  4. You can check if its present:

    instead of: :scream_cat:

    if thing is not None:
    

    you can do: :smile_cat:

    thing = some_func_returning_an_optional()
    if thing:
    
  5. You can check if its empty:

    instead of: :scream_cat:

    if thing is None:
    

    you can do: :smile_cat:

    thing = some_func_returning_an_optional()
    if not thing:
    
  6. You can match against the result and destructure the value:

    instead of: :scream_cat:

    print(thing)
    

    you can do: :smirk_cat:

    match some_func_returning_an_optional():
        case Something(thing):
            print(thing)
    
  7. You can match against an empty optional, but can't destructure the value:

    instead of: :crying_cat_face:

    if thing is None:
        print(None) # very odd
    

    you can do: :smirk_cat:

    match some_func_returning_an_optional()
        case Nothing():
            print("We didn't get a thing!")
    
  8. You can compare two optionals: :smile_cat:

    Optional.empty() == Optional.empty() # True
    Optional.of("thing") == Optional.of("thing") # True
    Optional.of("thing") == Optional.empty() # False
    Optional.of("thing") == Optional.of("PANTS") # False
    

Tests

There is complete test coverage and they pass in all Python versions 3.10 and up.

Running Unit Tests

First, install poetry using the instructions located here.

Then, install the requirements using:

poetry install

You can run the tests (with coverage) using:

poetry run pytest

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

optional_py-2.0.0.tar.gz (4.0 kB view details)

Uploaded Source

Built Distribution

optional_py-2.0.0-py3-none-any.whl (4.9 kB view details)

Uploaded Python 3

File details

Details for the file optional_py-2.0.0.tar.gz.

File metadata

  • Download URL: optional_py-2.0.0.tar.gz
  • Upload date:
  • Size: 4.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/5.0.0 CPython/3.12.2

File hashes

Hashes for optional_py-2.0.0.tar.gz
Algorithm Hash digest
SHA256 567f4cb8b80e14dd36693afa39c2897c82912abdfba3574212aed9353d4a7725
MD5 e8dce283a2d458c9fa150808e4173974
BLAKE2b-256 dcccc14db2da7c2d643b4b0bb77ff6a957bcf86a63d7fbd15d268ecf089b76af

See more details on using hashes here.

File details

Details for the file optional_py-2.0.0-py3-none-any.whl.

File metadata

  • Download URL: optional_py-2.0.0-py3-none-any.whl
  • Upload date:
  • Size: 4.9 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/5.0.0 CPython/3.12.2

File hashes

Hashes for optional_py-2.0.0-py3-none-any.whl
Algorithm Hash digest
SHA256 b79434effb80ff77a5c10c2f577a8618f7ea21d96587eb6b15bab32150a4a9bd
MD5 f6bdff8e26d30a024cc749ec669f04af
BLAKE2b-256 9cb00f589f70c979bc12abf24a576caa5ed007d53b15a1737e23aad5aef482fb

See more details on using hashes here.

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