Drop-in read-only dictionary with 100% typing and runtime compatibility
Project description
ReadonlyDict
Drop-in read-only dictionary with 100% typing and runtime compatibility
Overview: Why ReadonlyDict?
This package is built strictly on the following formula:
ReadonlyDict = (built-in dictionary features) - (in-place features) + (read-only features).
- 100% compatibility and zero custom API: Our goal is to achieve flawless compatibility with Python's built-in dictionary in both static type checking (e.g., mypy, Pyright) and runtime behavior. We simply removed in-place methods (e.g.,
pop(),update()). We do not introduce any custom methods. - True immutable semantics: The only additions are those strictly required for a read-only data structure: it is fully hashable (only if all values are hashable), and shallow copies (i.e.,
self.copy(),copy.copy(self)) simply return itself to save memory. - When to use this package: If you want extended read-only features, existing packages like frozendict, immutabledict, or immutables are better choices. However, if your priority is pure compatibility and perfect static type inference, ReadonlyDict should be the optimal choice.
Installation
pip install readonlydict
Basic Usage
It works exactly like a built-in dictionary, but raises an error if you try to modify it.
from readonlydict import ReadonlyDict
# Initialization works just like the built-in dictionary:
>>> ro = ReadonlyDict(a=0, b=1)
>>> ro
ReadonlyDict({'a': 0, 'b': 1})
# It is fully hashable (can be used as a dictionary key or in a set):
>>> hash(ro)
-5925576189957013898
>>> {ro, ro}
{ReadonlyDict({'a': 0, 'b': 1})}
# Mutation is strictly prohibited (static type checkers will also warn you):
>>> ro["c"] = 2
TypeError: 'ReadonlyDict' object does not support item assignment
>>> ro.update(c=2)
AttributeError: 'ReadonlyDict' object has no attribute 'update'
Advanced Usage: Subclassing with Type Hints
If you want to create your own custom read-only dictionary by subclassing ReadonlyDict, you can maintain static type inference by utilizing TYPE_CHECKING and @overload.
Here is the best-practice template for subclassing:
# standard library
from collections.abc import Iterable, Mapping
from typing import TYPE_CHECKING, Any, TypeVar, overload
# dependencies
from readonlydict import ReadonlyDict, Tuples
# type variables
K = TypeVar("K")
V = TypeVar("V")
K2 = TypeVar("K2")
V2 = TypeVar("V2")
class CustomDict(ReadonlyDict[K, V]):
# Modify the return types to guarantee type inference:
if TYPE_CHECKING:
@overload
def __new__(cls, **kwargs: V) -> "CustomDict[str, V]": ...
@overload
def __new__(cls, mapping: Mapping[K, V], /, **kwargs: V2) -> "CustomDict[K | str, V | V2]": ...
@overload
def __new__(cls, iterable: Tuples[K, V], /, **kwargs: V2) -> "CustomDict[K | str, V | V2]": ...
def __new__(cls, *args: Any, **kwargs: Any) -> Any: ... # type: ignore[misc]
@overload
@classmethod
def fromkeys(cls, iterable: Iterable[K2], /) -> "CustomDict[K2, None]": ...
@overload
@classmethod
def fromkeys(cls, iterable: Iterable[K2], value: V2, /) -> "CustomDict[K2, V2]": ...
@classmethod
def fromkeys(cls, *args: Any, **kwargs: Any) -> Any: ...
def __or__(self, other: Mapping[K2, V2], /) -> "CustomDict[K | K2, V | V2]": ...
# Then add your custom properties or methods:
@property
def first(self) -> tuple[K, V]:
return next(iter(self.items()))
Project details
Release history Release notifications | RSS feed
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 readonlydict-1.1.0.tar.gz.
File metadata
- Download URL: readonlydict-1.1.0.tar.gz
- Upload date:
- Size: 73.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.9.30 {"installer":{"name":"uv","version":"0.9.30","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Debian GNU/Linux","version":"12","id":"bookworm","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
eab1f19e733025727e526dc17dd31d3c6dd9310142dd28cd5f571fbd2000df18
|
|
| MD5 |
424f5fb6bf828afb5ce901420eccbca8
|
|
| BLAKE2b-256 |
dd5d5c1f61dbbf9b6a92db09c7b5c1c04e1ebebb127c7ac74c8ddb22d7c2d460
|
File details
Details for the file readonlydict-1.1.0-py3-none-any.whl.
File metadata
- Download URL: readonlydict-1.1.0-py3-none-any.whl
- Upload date:
- Size: 6.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.9.30 {"installer":{"name":"uv","version":"0.9.30","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Debian GNU/Linux","version":"12","id":"bookworm","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
71f501893de8aede904755682f2ae3e9b960d1bdb42f83dd8556b3d8337c493e
|
|
| MD5 |
41bcd7472df4d089c0d40bef9af99ef7
|
|
| BLAKE2b-256 |
f7bba0060621167036e3b3026b0cda0d8c1bcc1c462c54dc18982b6088463dba
|