Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Anonymous and inline declaration of TypedDict types #9884

Closed
jetzhou opened this issue Jan 7, 2021 · 20 comments · Fixed by #17457
Closed

Anonymous and inline declaration of TypedDict types #9884

jetzhou opened this issue Jan 7, 2021 · 20 comments · Fixed by #17457

Comments

@jetzhou
Copy link

jetzhou commented Jan 7, 2021

Feature
Currently, TypedDict must be declared and used as so:

MyDict = TypedDict('MyDict', {'foo': int, 'bar': bool})
def fn() -> MyDict:
    pass

In many situations, the type doesn't actually get re-used outside the context of this one local function, so a much more concise declaration would be:

def fn() -> TypedDict({'foo': int, 'bar': bool}):
    pass

or alternatively:

def fn() -> TypedDict(foo=int, bar=bool):
    pass

This syntax makes the TypedDict definition both anonymous and inline, which are two separate features I guess, even though they are pretty much hand in hand. Both are desirable in situations where the return type isn't re-used in multiple places so there is no need for the separate MyDict definition.

Pitch

I've read through #985 and python/typing#28. Both issues had a little bit of discussion on anonymous TypedDict and dismissed the feature as not being useful enough.

In my case, we run a GraphQL server and many of our mutation types return dictionaries/JSON objects with the shape {'ok': bool, 'error': str, 'item': <ItemType>}. These objects have no relevance outside of where their mutation function is defined, so it is extremely verbose and pointless to type:

ReturnType = TypedDict('ReturnedType', {'ok': bool, 'error': str, 'item': <ItemType>})
def mutation() -> ReturnType:
    return {'ok', True, 'error', '', 'item': <item>}

When everything can be succinctly expressed as:

def mutation() -> TypedDict({'ok': bool, 'error': str, 'item': <ItemType>}):
    return {'ok', True, 'error', '', 'item': <item>}

The simplified syntax has better readability and loses none of the type safety that's achieved with the type hints.

Outside of my use cases, I can also imagine that inline anonymous TypedDict can be pretty useful when they are used in a nested fashion, something like:

MyType = TypedDict({
    'foo': int,
    'bar': TypedDict({
        'baz': int,
    })
})

Happy to provide more context on the use case that I have or to come up with more use cases. If this seems like a worthy direction to pursue, I would definitely love to dig into implementation and figure out what needs to be done to make this happen.

@jetzhou jetzhou added the feature label Jan 7, 2021
@bennettdams
Copy link

bennettdams commented Mar 16, 2021

Just to give a perspective from another language: One of TypeScript's main features is exactly this "object typing". It's even a main bullet point in their documentation and one of the first examples.
This way of "inline typing" is useful for many use cases, but especially for return values, as @jetzhou described.

https://www.typescriptlang.org/docs/handbook/2/objects.html

@zuckerruebe
Copy link

I'd prefer the following syntax:

def fn() -> {"foo": int, "bar": bool}:
    pass

That would be so much better than having to do

def fn() -> Tuple[it, bool]:
    pass

(as is the case now) and then having to remember whether [0] or [1] was foo.

@amh4r
Copy link

amh4r commented Apr 24, 2022

I'd love this feature! I'm writing a library that codegens JSON Schema into TypedDicts, but JSON schema has nested object schemas.

For example, I'd like to convert this JSON schema:

{
    "id": "#root",
    "properties": {
        "Foo": {
            "properties": {
                "bar": {
                    "type": "object",
                    "properties": {
                        "baz": {"type": "integer"}
                    }
                }
            }
        }
    }
}

Into something like this:

Foo = TypedDict(
    {
        "bar": {
            {"baz": int},
        },
    },
)

I could write logic that creates something like this:

class FooBar(TypedDict):
    baz: int

class Foo(TypedDict):
    bar: FooBar

But that adds complexity to my recursive functions:

  • They need to have extra context to generate the path-based name (i.e. Foo.bar -> FooBar).
  • They need to handle name conflicts. Like if the schema explicitly declares FooBar somewhere, then the generated name would need to be something like FooBar2.

TypeScript supports nested interfaces and it's made JSON Schema codegen much easier. Hopefully TypedDict can adopt some of that flexibility

@mahmoudajawad
Copy link

Does this require a PEP or it can be implemented on mypy as feature?

@JelleZijlstra
Copy link
Member

Mypy can implement its own extensions if it wants, but standardizing this feature would require a new PEP.

@JukkaL
Copy link
Collaborator

JukkaL commented May 19, 2022

TypedDict({'foo': int, 'bar': bool}) generates a TypeError at runtime. To properly support this we'd need changes to typing.TypedDict.

@erictraut
Copy link

If you want to pursue this, I recommend discussing it in the typing-sig email list or the python/typing discussion forum. It deserves broader discussion and input before it is implemented in type checkers.

@jetzhou
Copy link
Author

jetzhou commented May 20, 2022

@erictraut, @JelleZijlstra thanks for the pointers! I'm glad that this thread/feature is gaining some traction here. I have some time on my hands so I will look into starting the discussion on the mailing list and the PEP process. Will update here if it gets somewhere. Thanks all!

@andersk
Copy link
Contributor

andersk commented Jul 21, 2022

We could represent an anonymous TypedDict with an empty string for the name: TypedDict("", {'foo': int, 'bar': bool}). This already works at runtime.

@emmatyping
Copy link
Collaborator

I suggest we close this in favor of discussing this on typing-sig and having it standardized in a PEP, I'd love to see something like {'foo': int} be shorter syntax for an anonymous TypedDict, similar to how we now allow | for union, dict from builtins etc.

@michaeloliverx
Copy link

It would be great if type checkers were able to infer the return types of functions as typed dicts if returning literal dicts e.g.

def foo():
    return {
        "foo: "bar"
    }

TypeScript shines in this regard.

@DetachHead
Copy link
Contributor

pyright supports this using the following syntax:

foo: dict[{"a": str}] = {"a": "asdf"}

@MrLoh
Copy link

MrLoh commented Apr 18, 2024

@DetachHead is this pyright feature documented somewhere?

@DetachHead
Copy link
Contributor

it's an experimental feature that i don't think is documented anywhere except this thread python/typing#1391

@erictraut
Copy link

Yes, this feature is experimental in pyright. The idea was discussed in a typing forum thread, but it was never fully fleshed out in a specification. I will likely remove from pyright since no one has stepped up to write a PEP.

If you would like to see such a feature added to the type system, please consider starting a new thread in the typing forum, gather feedback and ideas, and write a draft PEP.

@ilevkivskyi
Copy link
Member

An experimental support for this has been merged to master, you can play with it using --enable-incomplete-feature=InlineTypedDict.

@layday
Copy link

layday commented Jul 7, 2024

Support for inline dicts was removed from Pyright in June, by the way, which seems rather unfortunate.

@erictraut
Copy link

Yes, I removed support for this experimental feature because no one stepped up to formally specify it, and there was little or no feedback (positive or negative) from pyright users.

If someone is interested in spearheading the formal specification for such a feature, the Python typing forum is a good place to start the discussion. This feature would require a PEP.

@JelleZijlstra
Copy link
Member

We had a typing meetup presentation from @sobolevn about this some time ago, and as I remember, there was significant disagreement over how inheritance should work, with the result that Nikita stopped pursuing the proposal.

@DetachHead
Copy link
Contributor

this feature is still supported in basedpyright

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging a pull request may close this issue.