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

Implement condition/attr __repr__ #3254

Open
wants to merge 4 commits into
base: develop
Choose a base branch
from

Conversation

rajveerappan
Copy link

@rajveerappan rajveerappan commented May 4, 2022

The main goal of this PR is to implement a human readable __repr__ representation of the boto3 DynamoDB conditions. The primary use case being for debugging and logging. Currently printing/logging a FilterExpression will show something like:

<boto3.dynamodb.conditions.AttributeExists object at 0x1041021f0>
<boto3.dynamodb.conditions.And object at 0x104102370>
<boto3.dynamodb.conditions.In object at 0x1041027f0>

After this change the printing/logging will display a more human readable version like:

attribute_exists(mykey)
(mykey = foo AND myotherkey = bar)
mykey IN foo

The way this is implemented is by turning an expression_format which looks like {0} {operator} {1} into something that looks like {values[0]} {operator} {values[1]} which can be passed to the string format method using named indices.

@rajveerappan rajveerappan changed the title [PoC] Implement condition __str__ [PoC] Implement condition/attr __str__ May 4, 2022
@tim-finnigan
Copy link
Contributor

Linking to the issue you created: #3259

The team is open to improving the logging but mentioned that repr() should be used here rather than str().

@rajveerappan
Copy link
Author

Hi @tim-finnigan, apologies for the late response, not sure why Github didn't notify me of your response. I'll make the change to implement repr() instead and add tests. Please let me know if there are any concerns around performance with using the REGEX the way I am here, if not I'll update this PR in the next day or two.

@rajveerappan rajveerappan force-pushed the dynamodb-conditions-str branch from 79df738 to ed1cfd7 Compare November 23, 2022 21:41
@rajveerappan rajveerappan force-pushed the dynamodb-conditions-str branch from ed1cfd7 to 04ace42 Compare March 7, 2023 14:32
@rajveerappan rajveerappan changed the title [PoC] Implement condition/attr __str__ Implement condition/attr __repr__ Mar 7, 2023
@@ -284,6 +284,10 @@ def test_eq(self):
},
)

def test_eq_repr(self):
actual_repr = str(Equals(self.value, self.value2))
assert actual_repr == 'mykey = foo'
Copy link
Author

Choose a reason for hiding this comment

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

@tim-finnigan / @tibbe I could use your input on these tests before I flesh them out fully. One option is what I'm doing here and adding a bunch of test_op_repr tests. Another option is to change ConditionBase#get_expression so that it returns a new repr key with the string representation and then update the existing test_op test cases to assert on the repr also.

I don't have a preference either way but let me know what you would prefer.

Copy link
Author

Choose a reason for hiding this comment

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

actually I already fleshed them out but happy to revise based on comments. otherwise I think this PR is ready for review.

Copy link

Choose a reason for hiding this comment

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

Sorry for the late reply. We might need to distinguish between what __str__ and __repr__ does. According to the specification of __repr__:

If at all possible, this should look like a valid Python expression that could be used to recreate an object with the same value (given an appropriate environment). If this is not possible, a string of the form <...some useful description...> should be returned. The return value must be a string object. If a class defines __repr__() but not __str__(), then __repr__() is also used when an “informal” string representation of instances of that class is required.

In the case of __repr__ we probably want something that looks like constructor application e.g. And(AttributeExists("x"), AttributeExists("y")). For __str__ we could then have whatever pretty-prints nicely e.g. the syntax actually used in DynamoDB query expressions.

Copy link

Choose a reason for hiding this comment

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

What I ultimately care about is that when the Stubber compares two expressions and they are non-equal it prints something useful. It currently prints using __str__ (which falls back to __repr__ if the condition class doesn't define it): https://github.com/boto/botocore/blob/9f1dd1c843cb938ee96af969045b8a7a0c53deee/botocore/stub.py#L386

Copy link

@tibbe tibbe Mar 8, 2023

Choose a reason for hiding this comment

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

To be concrete, I suggest we implement both __str__ and __repr__, using the implementation you have for __repr__ in this PR for __str__ and adding my suggestion for __repr__ above.

Copy link
Author

Choose a reason for hiding this comment

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

I had originally started with __str__ but was asked to use __repr__ instead here:
#3254 (comment)

tbh I'm not particular as long as we can get this PR merged and have human understandable representations so that debugging is a nicer experience.

@tim-finnigan what can I do to help get this PR merged?

Copy link

Choose a reason for hiding this comment

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

If we go with a textual description the string should be surrounded by <>, as per the documentation of __report__(), or otherwise it will be hard to parse when there is surrounding text.

In this case I think we could actually get a real __repr__() (i.e. one fulfilling "If at all possible, this should look like a valid Python expression that could be used to recreate an object with the same value (given an appropriate environment).") by instead implementing __repr__() along the lines of:

def __repr__(self):
  return f'{self.__class__.__name__}({", ".join(self._values)})'

@rshah-evertz
Copy link

Any updates on this?

@rajveerappan
Copy link
Author

Any updates on this?

@tim-finnigan how can we get this PR merged?

@tim-finnigan
Copy link
Contributor

Thanks for your patience — I don't have any updates here, a maintainer on the team will need to review this when there is bandwidth. I'll try raising this PR again but we can't guarantee any timelines on if or when any PR would get merged.

@rajveerappan rajveerappan force-pushed the dynamodb-conditions-str branch from 3b00c06 to dd9e4fd Compare August 5, 2024 11:56
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.

4 participants