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

bpo-32622: Implement loop.sendfile() #5271

Merged
merged 36 commits into from
Jan 27, 2018
Merged

Conversation

asvetlov
Copy link
Contributor

@asvetlov asvetlov commented Jan 22, 2018


def connection_made(self, transport):
raise RuntimeError("Broken state, "
"connection should be established already.")
Copy link
Member

Choose a reason for hiding this comment

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

"Invalid state: connection should have been established already"

self._paused = None

def data_received(self, data):
raise RuntimeError("Broken state, reading should be paused")
Copy link
Member

Choose a reason for hiding this comment

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

ditto

self._transport = transp
self._proto = transp.get_protocol()
self._resume_reading = transp.is_reading()
self._write_paused = transp._protocol_paused
Copy link
Member

Choose a reason for hiding this comment

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

I'd call these '_should_resume_reading' and '_should_resume_writing'

if mode is constants._SendfileMode.UNSUPPORTED:
raise RuntimeError(
f"sendfile is not supported for transport {transport!r}")
if mode is constants._SendfileMode.NATIVE:
Copy link
Member

Choose a reason for hiding this comment

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

I'd call it TRY_NATIVE

except events.SendfileNotAvailableError as exc:
if not fallback:
raise
# the mode is FALLBACK or fallback is True
Copy link
Member

Choose a reason for hiding this comment

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

I'd put an assert check here.

break
fut = proto._paused
if fut is not None:
if await fut:
Copy link
Member

Choose a reason for hiding this comment

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

Let's use another enum for the values of proto._paused future; that will improve the readability a great deal.


def __init__(self, loop, sock, protocol, waiter=None,
extra=None, server=None):
super().__init__(loop, sock, protocol, extra, server)
self._eof = False
self._paused = False
self._empty_waiter = None
Copy link
Member

Choose a reason for hiding this comment

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

_empty_waiter should handle a case when it exists and a connection is aborted/closed. In which case we want to cancel it.

Copy link
Member

Choose a reason for hiding this comment

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

This case is still isn't handled.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Cancelling is confusing.
I prefer another exception -- and OSError or derived type is explicitly set

Copy link
Member

Choose a reason for hiding this comment

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

Whatever, just make sure that if something awaits on _empty_writer does not wait forever because we don't cancel the future.


def connection_made(self, transport):
raise RuntimeError("Invalid state, "
"connection should be established already.")
Copy link
Member

Choose a reason for hiding this comment

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

"should be" -> "should have been"


class _SendfileProtocol(protocols.Protocol):
def __init__(self, transp):
# transport should be _FlowControlMixin instance
Copy link
Member

Choose a reason for hiding this comment

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

Can you add an if not isinstance(transp, _FlowControlMixin): raise TypeError. I know that it shouldn't be possible to get this exception, but in case we made a mistake and some user hits this error message, it would be more informative than some random AttributeError.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Perform the check in debug mode only maybe?

Copy link
Member

Choose a reason for hiding this comment

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

sendfile isn't a frequent operation so one isinstance call won't slow it down, so let's check it always.

@@ -765,6 +782,9 @@ def write(self, data):
f'not {type(data).__name__!r}')
if self._eof:
raise RuntimeError('Cannot call write() after write_eof()')
if self._empty_waiter is not None:
raise RuntimeError('Cannot call write() when loop.sendfile() '
'is not finished')
Copy link
Member

Choose a reason for hiding this comment

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

unable to write; sendfile is in progress

"""Send a file through a transport.

Return amount of sent bytes.
"""
Copy link
Member

Choose a reason for hiding this comment

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

Copy the relevant snippet from the documentation to make it a proper docstring.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Did it but I feel that docstring is too long.

transp.pause_reading()
fut = transp._make_empty_waiter()
await fut
try:
Copy link
Member

Choose a reason for hiding this comment

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

await transp._make_empty_waiter() seems to be a bit shorter/readable.

@1st1
Copy link
Member

1st1 commented Jan 26, 2018

Overall LGTM, left a couple minor comments.

@asvetlov asvetlov merged commit 7c68407 into python:master Jan 27, 2018
@bedevere-bot
Copy link

@asvetlov: Please replace # with GH- in the commit message next time. Thanks!

@asvetlov asvetlov deleted the sendfile branch January 27, 2018 19:22
@asvetlov
Copy link
Contributor Author

Uhhh. Done, finally.

@1st1
Copy link
Member

1st1 commented Jan 27, 2018

Thank you, Andrew! I'm super happy about asyncio finally getting first class sendfile support!

@asvetlov
Copy link
Contributor Author

It means I can drop ugly sendfile hacks from aiohttp :)

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

Successfully merging this pull request may close these issues.

4 participants