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

Make CommandResult a dataclass #722

Merged
merged 1 commit into from
Nov 13, 2023

Conversation

amaslenn
Copy link
Contributor

Following discussion for #717, make CommandResult a dataclass.

@amaslenn
Copy link
Contributor Author

@philpep I don't really like how BaseBackend.result() looks now, what do you think about changing its signature from result(self, *args: Any, **kwargs: Any) to something like this:

    def result(
        self,
        exit_code: int,
        command: bytes,
        stdout_bytes: bytes,
        stderr_bytes: bytes,
        stdout: Optional[str] = None,
        stderr: Optional[str] = None,
    ) -> CommandResult:

The code that uses it should be updated of course.

Let me know what do you think.

@amaslenn amaslenn marked this pull request as ready for review July 26, 2023 16:07
if not out and out_bytes:
out = self.decode(out_bytes)
elif out and not out_bytes:
out_bytes = self.encode(out)
Copy link
Contributor

Choose a reason for hiding this comment

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

I think we should keep lazy encoding or decoding, i.e. using properties, because sometime we may want to get binary data which cannot be converted to text.

I think the dataclass should have _stdout and _stdout_bytes and check at __init__ that at least one of them are set. Or make subclasses BinaryCommandResult and TextCommandResult.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Why is lazy encoding and decoding are needed? My way of thoughts here was to remove dependency on Backend, which looks redundant to me. So why is it needed here?

With dataclass implementation we always know that parameters are set (at least as None, but that is the task for mypy to detect), so this verification is kinda included.

Current solution doesn't break backward compatibility, at least to the extend it is tested now.

Another solution here would be to do similar to what subprocess.CompletedProcess does: it is generic and can be either bytes of str. We can borrow the approach, but that will change the API.

Copy link
Contributor

@philpep philpep Aug 2, 2023

Choose a reason for hiding this comment

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

Lazy encoding/decoding is needed because commands like run("cat /usr/bin/cat") will raise an error if trying to decode. Also IIRC either salt and/or ansible might have weird encoding behavior while returning command output in a json field. Lazy encoding/decoding avoid some error, as long as user doesn't explicitly ask for encoding/decoding.

Yes something like this could be fine:

@dataclass
class CommandResult:
    _stdout: AnyStr
    _stderr: AnyStr

    @property
    def stdout(self) -> str:
        if isinstance(self._stdout, bytes):
            return self._stdout.decode(...)
        return self._stdout

    [...]

Copy link
Contributor Author

Choose a reason for hiding this comment

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

(took longer than I expected, I was busy)

Applied changes to keep lazy encoding/decoding.
@philpep do you think it is time to change result(self, *args: Any, **kwargs: Any) signature in this PR?

@philpep philpep added the typing label Nov 13, 2023
@philpep philpep force-pushed the commandresult-dataclass branch from dff3b38 to e1064e4 Compare November 13, 2023 17:05
@philpep philpep force-pushed the commandresult-dataclass branch from e1064e4 to 145817e Compare November 13, 2023 17:08
@philpep philpep merged commit dc48cd9 into pytest-dev:main Nov 13, 2023
7 checks passed
@philpep
Copy link
Contributor

philpep commented Nov 13, 2023

I modified result() to make it simple and changed calls instead. Merged, thanks!

@amaslenn amaslenn deleted the commandresult-dataclass branch November 13, 2023 17:35
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants