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

Handle non-Job objects in Job equality check. #455

Merged
merged 8 commits into from
Jan 11, 2021

Conversation

bdice
Copy link
Member

@bdice bdice commented Dec 30, 2020

Description

I caught a couple minor errors in signac-flow tests (to be addressed separately) that were exposed by the changes in #442. This unrelated problem in signac-flow indicated an issue in the new __eq__ check, which doesn't correctly handle comparisons to non-Job objects. This PR fixes that bug so the Job class correctly adheres to the Python data model.

Specifically, the old __eq__ behavior did nothing to ensure that the objects being tested for equality were Job objects. If a job object and a tuple were compared, the previous Job equality test just checked if hash(job) == hash(tuple_of_data), which hid the test failures in signac-flow until #442 was merged and errors were raised by the comparison. However, the comparison should not raise errors or return False, and should instead return NotImplemented according to the Python data model and some StackOverflow posts I found.

Motivation and Context

Fixes minor bug in __eq__ implementation.

This issue from another library and this section of the Python documentation are helpful for explaining the motivation of the change.

I added tests that cover a range of cases.

Types of Changes

  • Documentation update
  • Bug fix
  • New feature
  • Breaking change1

1The change breaks (or has the potential to break) existing functionality.

Checklist:

If necessary:

  • I have updated the API documentation as part of the package doc-strings.
  • I have created a separate pull request to update the framework documentation on signac-docs and linked it here.
  • I have updated the changelog and added all related issue and pull request numbers for future reference (if applicable). See example below.

@bdice bdice requested review from a team as code owners December 30, 2020 05:45
@bdice bdice requested review from tcmoore3 and Charlottez112 and removed request for a team December 30, 2020 05:45
@bdice bdice self-assigned this Dec 30, 2020
@bdice bdice added the bug Something isn't working label Dec 30, 2020
@bdice bdice added this to the v1.6.0 milestone Dec 30, 2020
@bdice bdice changed the title Reject non-Job objects in Job equality check. Handle non-Job objects in Job equality check. Dec 30, 2020
@codecov
Copy link

codecov bot commented Dec 30, 2020

Codecov Report

Merging #455 (6f72370) into master (8cfa82d) will increase coverage by 0.00%.
The diff coverage is 100.00%.

Impacted file tree graph

@@           Coverage Diff           @@
##           master     #455   +/-   ##
=======================================
  Coverage   77.13%   77.14%           
=======================================
  Files          42       42           
  Lines        5738     5740    +2     
  Branches     1119     1120    +1     
=======================================
+ Hits         4426     4428    +2     
  Misses       1027     1027           
  Partials      285      285           
Impacted Files Coverage Δ
signac/contrib/job.py 91.20% <100.00%> (+0.06%) ⬆️

Continue to review full report at Codecov.

Legend - Click here to learn more
Δ = absolute <relative> (impact), ø = not affected, ? = missing data
Powered by Codecov. Last update 8cfa82d...6f72370. Read the comment docs.

Comment on lines +155 to +157
non_job = NonJob(job)
assert job != non_job
assert non_job != job
Copy link
Collaborator

Choose a reason for hiding this comment

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

Why does this ensure that we have NotImplemeted as opposed to False?

Copy link
Member Author

@bdice bdice Jan 3, 2021

Choose a reason for hiding this comment

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

I'm testing both a != b and b != a because of the order in which Python resolves __eq__ operators. This section of the Python docs explains why both tests are relevant here, and why we return NotImplemented here: object.__eq__ docs.

Specifically, the order of resolution is controlled by this rule: "If the operands are of different types, and right operand’s type is a direct or indirect subclass of the left operand’s type, the reflected method of the right operand has priority, otherwise the left operand’s method has priority."

Copy link
Member Author

Choose a reason for hiding this comment

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

Also, I used this issue as a reference implementation because it appeared to correct the equality behavior that I was unsure about for subclasses (JobSubclass) and objects of different types with similar APIs (NonJob): googleapis/google-cloud-python#3455

Copy link
Member

@tcmoore3 tcmoore3 left a comment

Choose a reason for hiding this comment

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

Good find, and nice fix!

@bdice bdice merged commit bd3af82 into master Jan 11, 2021
@bdice bdice deleted the fix/job-eq-respect-non-jobs branch January 11, 2021 02:14
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants