Skip to main content

Extracting Generic Type References in Python

Project description

generic-preserver

logo

Extracting Generic Type References in Python

Introduction

In Python, generic types are a powerful feature for writing reusable and type-safe code. However, one limitation is that generic type arguments are typically not preserved at runtime, making it challenging to access or utilize these types dynamically. generic-preserver is a Python package that overcomes this limitation by capturing and preserving generic type arguments, allowing you to access them at runtime.

This package is particularly useful when you need to perform operations based on the specific types used in your generic classes, such as serialization, deserialization, or dynamic type checking.

Features

  • Preserve Generic Types at Runtime: Capture and retain generic type arguments for classes and instances.
  • Runtime Access to Type Parameters: Easily access the type parameters passed to generic classes from their instances.
  • Supports Inheritance and Nested Generics: Works seamlessly with class hierarchies and nested generic types.
  • Simple and Intuitive API: Use either a metaclass or a decorator to enable functionality with minimal code changes.
  • Python 3.9+ Support: Leverages modern Python features for type hinting and annotations.

Installation

Install generic-preserver via pip:

pip install generic-preserver

Or install using Poetry:

poetry add generic-preserver

Requirements

  • Python 3.9 or higher

Usage

Using the GenericMeta Metaclass

To enable capturing generic type arguments, use the GenericMeta metaclass in your base class definition.

from typing import TypeVar, Generic
from generic_preserver.metaclass import GenericMeta

# Define type variables
A = TypeVar("A")
B = TypeVar("B")
C = TypeVar("C")

# Example classes to use as type arguments
class ExampleA:
    pass

class ExampleB:
    pass

class ExampleC:
    pass

# Base class with GenericMeta metaclass
class Parent(Generic[A, B], metaclass=GenericMeta):
    pass

# Child classes specifying some generic type arguments
class Child(Parent[ExampleA, B], Generic[B, C]):
    pass

class GrandChild(Child[ExampleB, C], Generic[C]):
    pass

# Create an instance of the generic class with type arguments
instance = GrandChild[ExampleC]()

# Access the preserved generic type arguments
print(instance[A])  # Output: <class '__main__.ExampleA'>
print(instance[B])  # Output: <class '__main__.ExampleB'>
print(instance[C])  # Output: <class '__main__.ExampleC'>

# View the internal generic map
print(instance.__generic_map__)
# Output:
# {
#     ~A: <class '__main__.ExampleA'>,
#     ~B: <class '__main__.ExampleB'>,
#     ~C: <class '__main__.ExampleC'>,
# }

Using the @generic_preserver Decorator

Alternatively, use the @generic_preserver decorator to enable capturing generic arguments without explicitly specifying the metaclass.

from typing import TypeVar, Generic
from generic_preserver.wrapper import generic_preserver

# Define type variables
A = TypeVar("A")
B = TypeVar("B")
C = TypeVar("C")

# Example classes to use as type arguments
class ExampleA:
    pass

class ExampleB:
    pass

class ExampleC:
    pass

# Use the decorator to enable generic preservation
@generic_preserver
class Parent(Generic[A, B]):
    pass

# Child classes specifying some generic type arguments
class Child(Parent[ExampleA, B], Generic[B, C]):
    pass

class GrandChild(Child[ExampleB, C], Generic[C]):
    pass

# Create an instance of the generic class with type arguments
instance = GrandChild[ExampleC]()

# Access the preserved generic type arguments
print(instance[A])  # Output: <class '__main__.ExampleA'>
print(instance[B])  # Output: <class '__main__.ExampleB'>
print(instance[C])  # Output: <class '__main__.ExampleC'>

# View the internal generic map
print(instance.__generic_map__)
# Output:
# {
#     ~A: <class '__main__.ExampleA'>,
#     ~B: <class '__main__.ExampleB'>,
#     ~C: <class '__main__.ExampleC'>,
# }

Accessing Type Variables

You can access the type arguments by indexing the instance with the corresponding TypeVar.

print(instance[A])  # Output: <class '__main__.ExampleA'>

If you attempt to access a type variable that was not defined or is not in the generic map, a KeyError will be raised.

D = TypeVar("D")
try:
    print(instance[D])
except KeyError as e:
    print(e)  # Output: No generic type found for generic arg ~D

Accessing Multiple Type Variables

You can retrieve multiple type variables at once by passing an iterable of TypeVar instances.

types = instance[A, B, C]
print(types)
# Output: (<class '__main__.ExampleA'>, <class '__main__.ExampleB'>, <class '__main__.ExampleC'>)

How It Works

The generic-preserver package uses a custom metaclass GenericMeta to intercept class creation and capture generic type arguments when a generic class is subscripted (e.g., MyClass[int, str]). Here's a brief overview:

  • Metaclass (GenericMeta): Overrides the __getitem__ method to capture the type arguments and store them in a __generic_map__.
  • Class Wrapper: Creates a wrapper class that inherits from the original class and includes the __generic_map__.
  • Instance Access: Allows instances to access the type arguments via the __getitem__ method.
  • Decorator (@generic_preserver): Provides a convenient way to apply GenericMeta without altering the class definition directly.

By preserving the generic type arguments in __generic_map__, you can access them at runtime, enabling more dynamic and type-aware programming patterns.

Testing

The package includes a test suite to verify its functionality. To run the tests, first install the development dependencies:

poetry install --with dev

Then, run the tests using pytest:

pytest

An example test case is provided in tests/test_wrapper.py:

def test_template():
    A = TypeVar("A")
    B = TypeVar("B")
    C = TypeVar("C")

    class ExampleA: pass
    class ExampleB: pass
    class ExampleC: pass

    @generic_preserver
    class Parent(Generic[A, B]): pass

    class Child(Parent[ExampleA, B], Generic[B, C]): pass

    class GrandChild(Child[ExampleB, C], Generic[C]): pass

    instance = GrandChild[ExampleC]()

    assert instance[A] is ExampleA
    assert instance[B] is ExampleB
    assert instance[C] is ExampleC

    D = TypeVar("D")
    with pytest.raises(KeyError):
        instance[D]

Limitations

  • Python Version: Requires Python 3.9 or higher due to the use of internal structures from the typing module.
  • Compatibility: May not be compatible with other metaclass-based libraries or complex metaclass hierarchies.
  • TypeVar Constraints: Does not enforce TypeVar constraints or bounds at runtime; it only captures the types provided.

Contributing

Contributions are welcome! If you find a bug or have an idea for a new feature, please open an issue or submit a pull request.

To contribute:

  1. Fork the repository.
  2. Create a new branch (git checkout -b feature/my-feature).
  3. Commit your changes (git commit -am 'Add my feature').
  4. Push to your branch (git push origin feature/my-feature).
  5. Open a Pull Request.

Please ensure that your code passes all tests and follows the existing coding style.

License

This project is licensed under the MIT License. See the LICENSE file for details.

Acknowledgements

  • Inspired by the need to access and utilize generic type parameters at runtime in Python applications.
  • Special thanks to the Python community for their contributions and support.

To learn more about how I came up with this solution, please read my blog post: Extracting Generic Type References in Python

Contact

For questions, suggestions, or feedback, please contact:

Matthew Coulter
Email: mattcoul7@gmail.com


Thank you for using generic-preserver! If you find this package helpful, consider giving it a star on GitHub.

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

generic_preserver-0.1.6.tar.gz (8.3 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

generic_preserver-0.1.6-py3-none-any.whl (10.9 kB view details)

Uploaded Python 3

File details

Details for the file generic_preserver-0.1.6.tar.gz.

File metadata

  • Download URL: generic_preserver-0.1.6.tar.gz
  • Upload date:
  • Size: 8.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/2.2.1 CPython/3.10.19 Linux/6.11.0-1018-azure

File hashes

Hashes for generic_preserver-0.1.6.tar.gz
Algorithm Hash digest
SHA256 f95cae0334f31ea5b457adf9d3f21ef837269c68980944ad04108d0f9eadf82b
MD5 886acf527bea88152a68b4d60ba18e1a
BLAKE2b-256 ec305483c4d54a076e2a49f3573ea232aaae2afdb84a9ee238eff03ec21d7036

See more details on using hashes here.

File details

Details for the file generic_preserver-0.1.6-py3-none-any.whl.

File metadata

  • Download URL: generic_preserver-0.1.6-py3-none-any.whl
  • Upload date:
  • Size: 10.9 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/2.2.1 CPython/3.10.19 Linux/6.11.0-1018-azure

File hashes

Hashes for generic_preserver-0.1.6-py3-none-any.whl
Algorithm Hash digest
SHA256 d71496f0c2c64a3ce5a13f2c0363867a3ab48f6975e19cbf3c2a1b7a8f5d747b
MD5 1aba1a35f961b97c444e2dfdf73b4aba
BLAKE2b-256 b3f7295be2cb4c3cc0434f477507298cd619a0eae4af37e8f27e549f2849792c

See more details on using hashes here.

Supported by

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