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

Improve converters and type annotations #1014

Merged
merged 4 commits into from
Jan 12, 2025
Merged

Conversation

injust
Copy link
Contributor

@injust injust commented Jan 9, 2025

Changes

  • Add proper type annotations to converters, instead of Any
  • as_timezone: Remove unreachable None handling (fixes Unreachable None handling in as_timezone() converter #1013)
    • It doesn't make sense for None to do the same thing as the "local" default while being less explicit, so I omitted it from the type annotation and dropped the unreachable code
    • The current runtime behaviour is maintained, but a type checker will now warn you about it
  • CronTrigger.from_crontab: Swap timezone parameter type annotation ordering to tzinfo | str for consistency
    • This ordering seems more logical than str | tzinfo
  • as_enum: Use Enum.__getitem__ instead of Enum.__members__.__getitem__

Checklist

If this is a user-facing code change, like a bugfix or a new feature, please ensure that
you've fulfilled the following conditions (where applicable):

  • You've added tests (in tests/) added which would fail without your patch
  • You've updated the documentation (in docs/, in case of behavior changes or new
    features)
  • You've added a new changelog entry (in docs/versionhistory.rst).

If this is a trivial change, like a typo fix or a code reformatting, then you can ignore
these instructions.

Updating the changelog

If there are no entries after the last release, use **UNRELEASED** as the version.
If, say, your patch fixes issue #999, the entry should look like this:

* Fix big bad boo-boo in the async scheduler (#999 <https://github.com/agronholm/apscheduler/issues/999>_; PR by @yourgithubaccount)

If there's no issue linked, just link to your pull request instead by updating the
changelog after you've created the PR.

@coveralls
Copy link

coveralls commented Jan 9, 2025

Coverage Status

coverage: 92.002% (-0.03%) from 92.033%
when pulling 3334665 on injust:patch-1
into 74a5ba4 on agronholm:master.

@injust
Copy link
Contributor Author

injust commented Jan 9, 2025

I'm not sure how to best type list_converter -- going to leave it as is.

@injust
Copy link
Contributor Author

injust commented Jan 9, 2025

I diffed the current docs against the RTD build for this PR and the changes look good. For example, DateTrigger()'s run_time argument is now typed as datetime | str instead of Any.

@injust injust marked this pull request as ready for review January 9, 2025 04:08
@injust
Copy link
Contributor Author

injust commented Jan 9, 2025

But somehow I made sphinx-autodoc-typehints unhappy -- https://apscheduler--1014.org.readthedocs.build/en/1014/api.html#apscheduler.triggers.interval.IntervalTrigger looks like this:

image

The types aren't in code blocks and don't link to the Python docs anymore. No idea what happened.

@injust injust changed the title Add proper converter type annotations and remove unreachable as_timezone() None handling Improve converter types Jan 9, 2025
@injust injust changed the title Improve converter types Improve converters and type annotations Jan 9, 2025
@injust
Copy link
Contributor Author

injust commented Jan 9, 2025

as_enum() should be typed like this:

def as_enum(enum_class: type[Enum]) -> Callable[[Enum | str], Enum]:
    def converter(value: Enum | str) -> Enum: ...

but it breaks the docs build (both locally and on RTD):

WARNING: Cannot resolve forward reference in type annotations of "apscheduler.Schedule": name 'Enum' is not defined
WARNING: Cannot resolve forward reference in type annotations of "apscheduler.JobResult": name 'Enum' is not defined
WARNING: Cannot resolve forward reference in type annotations of "apscheduler.AsyncScheduler": name 'Enum' is not defined

So I'll undo the type annotations but keep the change from enum_class.__members__[value] to enum_class[value], since that's how enum members should be accessed by name (https://docs.python.org/3/howto/enum.html#programmatic-access-to-enumeration-members-and-their-attributes).

@agronholm
Copy link
Owner

The type annotations for converters are not appropriate, as converters are run before validators, so the input value could be literally anything.

@injust
Copy link
Contributor Author

injust commented Jan 9, 2025

The type annotations for converters are not appropriate, as converters are run before validators, so the input value could be literally anything.

I'm not sure I follow. Type annotations should describe what's intended, so your type checker can catch errors before you get a TypeError from the validator at runtime.

@agronholm
Copy link
Owner

The type annotations for converters are not appropriate, as converters are run before validators, so the input value could be literally anything.

I'm not sure I follow. Type annotations should describe what's intended, so your type checker can catch errors before you get a TypeError from the validator at runtime.

Sure, but does the type checker know that it should look at the attrs converter to determine what types are allowed as input?

@injust
Copy link
Contributor Author

injust commented Jan 9, 2025

Sure, but does the type checker know that it should look at the attrs converter to determine what types are allowed as input?

Absolutely. I guess the attrs docs aren't the clearest about this, but a field's type annotation is overridden by that of its converter. https://www.attrs.org/en/stable/init.html#converters says:

If a converter’s first argument has a type annotation, that type will appear in the signature for __init__. A converter will override an explicit type annotation or type argument.

Which means that currently, converted fields are typed as Any (to a type checker as well as in the docs).

After this PR:

image image

@agronholm
Copy link
Owner

Since you've verified that these changes improve the type checking and documentation, I'm merging this. Thanks!

@agronholm agronholm merged commit 8548e7f into agronholm:master Jan 12, 2025
13 checks passed
@injust injust deleted the patch-1 branch January 12, 2025 12:21
@injust
Copy link
Contributor Author

injust commented Jan 16, 2025

This PR breaks pyright, but it seems to only happen when APScheduler is used in an attrs field like this:

from apscheduler import AsyncScheduler
from attrs import Factory, define


@define
class Foo:
    scheduler: AsyncScheduler = Factory(AsyncScheduler)
./foo.py:7:41 - error: Argument of type "type[AsyncScheduler]" cannot be assigned to parameter "factory" of type "() -> _T@Factory" in function "Factory"
  Type "type[AsyncScheduler]" is not assignable to type "() -> AsyncScheduler"
    Type "timedelta | None" is not assignable to type "timedelta | int"
      Type "None" is not assignable to type "timedelta | int"
        "None" is not assignable to "timedelta"
        "None" is not assignable to "int" (reportArgumentType)

pyright's behaviour seems to be inconsistent, and I'll open a discussion with them to see why this happens and if it's a bug.

I'll also open a follow-up PR to add attrs.converters.optional() to fields that should take None.

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

Successfully merging this pull request may close these issues.

Unreachable None handling in as_timezone() converter
3 participants