Extracting Generic Type References in Python
Project description
generic-preserver
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 applyGenericMetawithout 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
typingmodule. - Compatibility: May not be compatible with other metaclass-based libraries or complex metaclass hierarchies.
- TypeVar Constraints: Does not enforce
TypeVarconstraints 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:
- Fork the repository.
- Create a new branch (
git checkout -b feature/my-feature). - Commit your changes (
git commit -am 'Add my feature'). - Push to your branch (
git push origin feature/my-feature). - 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
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file generic_preserver-0.1.3.tar.gz.
File metadata
- Download URL: generic_preserver-0.1.3.tar.gz
- Upload date:
- Size: 6.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.8.5 CPython/3.10.15 Linux/6.5.0-1025-azure
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5b80c1ebaadc45446a1018ec5ef87b4672abc39ca9148ec85bd3fbdcd72b9d87
|
|
| MD5 |
577ecf04618e807034e7e115f2b5f375
|
|
| BLAKE2b-256 |
6d97dd10ebe7cf84ed3eda9fdb3b6038f2d99339bc3fc0177ff8d43dd84156dc
|
File details
Details for the file generic_preserver-0.1.3-py3-none-any.whl.
File metadata
- Download URL: generic_preserver-0.1.3-py3-none-any.whl
- Upload date:
- Size: 8.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.8.5 CPython/3.10.15 Linux/6.5.0-1025-azure
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6d9249ef4839e7387e8336d26d5c47b8554bd3deba7ab9b24acfa1102a56f6e9
|
|
| MD5 |
65db20a376a3f69fe6c7041f7fab8d04
|
|
| BLAKE2b-256 |
41c8895ac18252b4e77d1f67da8a0894f5a0b1f2a4c9d33c86c6a656ec707308
|