KEMBAR78
Support Negative Int Literal by TH3CHARLie · Pull Request #7878 · python/mypy · GitHub
Skip to content

Conversation

@TH3CHARLie
Copy link
Collaborator

@TH3CHARLie TH3CHARLie commented Nov 5, 2019

This PR aims to support negative int Literal, resolves #7844

Currently, the newly added test will broke and show inconsistent output with directly invoking mypy from command line

Commit 278ec69 shows the inconsistent error output between running mypy from command line and running from tests

Commit 7b3d1e8 now behaves ideally for testcase like #7844. But it breaks current codebase and tests.

Copy link
Member

@ilevkivskyi ilevkivskyi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks, looks like a god start, I have some action items for you.

This is mainly used to infer the return type as LiteralType
if the original underlying object is a LiteralType object
"""
if isinstance(ctx.type, Instance) and isinstance(ctx.type.last_known_value, LiteralType):
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
if isinstance(ctx.type, Instance) and isinstance(ctx.type.last_known_value, LiteralType):
if isinstance(ctx.type, Instance) and ctx.type.last_known_value is not None:

value = ctx.type.last_known_value.value
fallback = ctx.type.last_known_value.fallback
if isinstance(value, int):
return LiteralType(-value, fallback)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You should return an Instance here with the last known value set to this literal, instead of the literal directly. This is what screws type inference and makes many tests fail.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That sounds very reasonable. I patched this up in 473c93a but it fails at newly added test, showing that rhs of the assignment is still int type.

I also tried modifying ctx.type in place by changing ctx.type.last_known_value.value if Instance and ctx.type.value if LiteralType and then returning ctx.type directly, it fails the same.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hm, it looks like in both branches you need to return either instance or literal depending on context, essentially you need to copy logic from ExpressionChecker.infer_literal_expr_type(). For this however you must know type_context. It is not however a part of CheckerPluginInterface. It actually may be a useful addition, so I propose to add type_context there, and consequently make it a property on TypeChecker.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I summary the message as below:

  • add type_context to CheckerPluginInterface
  • fill the missing logic so that TypeChecker correctly has a type_context
  • copy logic from ExpressionChecker.infer_literal_expr_type() to complete this task
    If my understanding is correct, I will start working on it ASAP

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes.

(about the last bullet, copy the logic to your plugin)

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I took a quick look at checker.py and checkerexpr.py and got a little bit confused. The callback function returned from the plugin and thus has no access to the TypeChecker and the plugin passed in when constructing a TypeChecker directly goes into the ExpressionChecker and therefore I see no ways that will be able to check type_context inside the callback function without modifying its API. Please enlighten me

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Out of the three steps above which one you can't make?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's my misunderstanding. Now I think it's going to work, please check 32a99a2

Copy link
Member

@ilevkivskyi ilevkivskyi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for updates, this looks almost ready! I have few minor comments.

mypy/checker.py Outdated
self.msg = MessageBuilder(errors, modules)
self.plugin = plugin
self.expr_checker = mypy.checkexpr.ExpressionChecker(self, self.msg, self.plugin)
self.type_context = self.expr_checker.type_context
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could you please make this a property, to make it clear it is not writeable.

mypy/plugin.py Outdated
options = None # type: Options
path = None # type: str
# Type context for type inference
type_context = None # type: List[Optional[Type]]
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Here it should be an abstract property.

value = ctx.type.value
fallback = ctx.type.fallback
if isinstance(value, int):
return LiteralType(value=-value, fallback=fallback)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could you please also add this test:

ONE: Final = 1
TWO: Final = 2

err_code = -ONE
if bool():
    err_code = -TWO

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should I add it as a new test or append to testNegativeIntLiteralWithFinal in check-literal.test @ilevkivskyi

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think appending to the existing test is better.

Copy link
Member

@ilevkivskyi ilevkivskyi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Great, thanks!

@ilevkivskyi ilevkivskyi merged commit 8010414 into python:master Nov 6, 2019
@TH3CHARLie
Copy link
Collaborator Author

Thanks for the guidance! It’s a great education

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.

Literal[negative int]

2 participants