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

asyncio missing wrap_socket (starttls) #67937

Closed
gc mannequin opened this issue Mar 23, 2015 · 41 comments
Closed

asyncio missing wrap_socket (starttls) #67937

gc mannequin opened this issue Mar 23, 2015 · 41 comments
Assignees
Labels
3.7 (EOL) end of life topic-asyncio type-feature A feature request or enhancement

Comments

@gc
Copy link
Mannequin

gc mannequin commented Mar 23, 2015

BPO 23749
Nosy @warsaw, @vstinner, @tiran, @asvetlov, @agronholm, @1st1, @msornay, @Frzk
PRs
  • bpo-23749: Implement loop.start_tls() #5039
  • bpo-23749: Try fixing start-tls tests on win7 buildbot #5409
  • Files
  • tls1.patch
  • Note: these values reflect the state of the issue at the time it was migrated and might not reflect the current state.

    Show more details

    GitHub fields:

    assignee = 'https://github.com/1st1'
    closed_at = <Date 2018-05-28.16:40:26.762>
    created_at = <Date 2015-03-23.13:59:59.558>
    labels = ['type-feature', '3.7', 'expert-asyncio']
    title = 'asyncio missing wrap_socket (starttls)'
    updated_at = <Date 2018-05-28.22:47:51.658>
    user = 'https://bugs.python.org/gc'

    bugs.python.org fields:

    activity = <Date 2018-05-28.22:47:51.658>
    actor = 'vstinner'
    assignee = 'yselivanov'
    closed = True
    closed_date = <Date 2018-05-28.16:40:26.762>
    closer = 'yselivanov'
    components = ['asyncio']
    creation = <Date 2015-03-23.13:59:59.558>
    creator = 'gc'
    dependencies = []
    files = ['40866']
    hgrepos = []
    issue_num = 23749
    keywords = ['patch']
    message_count = 41.0
    messages = ['239021', '239022', '239026', '242095', '242097', '242136', '242155', '242193', '242194', '242195', '242197', '242198', '242199', '245532', '248048', '248051', '248053', '248054', '252513', '253495', '253496', '253888', '265494', '269417', '269439', '276790', '278161', '278162', '293912', '301628', '309220', '309235', '309236', '309645', '309648', '310577', '311057', '311117', '311150', '317222', '317942']
    nosy_count = 14.0
    nosy_names = ['barry', 'siemer', 'vstinner', 'christian.heimes', 'asvetlov', 'alex.gronholm', 'python-dev', 'yselivanov', 'msornay', 'Elizacat', 'gc', 'sphawk', 'Alex Gr\xc3\xb6nholm', 'Frzk']
    pr_nums = ['5039', '5409']
    priority = 'normal'
    resolution = 'fixed'
    stage = 'resolved'
    status = 'closed'
    superseder = None
    type = 'enhancement'
    url = 'https://bugs.python.org/issue23749'
    versions = ['Python 3.7']

    @gc
    Copy link
    Mannequin Author

    gc mannequin commented Mar 23, 2015

    It's not possible to wrap a socket in tls. The StreamWriter object should have an option to start a tls negotiation using the SSLContext of the server.

    This is needed for protocols the have a "start_tls" feature, for example the ldap protocol.

    In a non async program it's very easy:
    wrapped_socket = ssl_context.wrap_socket(connection.socket, server_side=True, do_handshake_on_connect=True)

    there should be something similar in the StreamWriter interface:
    yield from writer.wrap_socket()

    Bye,
    Giovanni

    @gc gc mannequin added topic-asyncio type-feature A feature request or enhancement labels Mar 23, 2015
    @vstinner
    Copy link
    Member

    Yes, it's not supported yet. It was already requested in this issue:
    https://code.google.com/p/tulip/issues/detail?id=79

    asyncio got a new SSL implementation which makes possible to implement STARTTLS. Are you interested to implement it?

    @gc
    Copy link
    Mannequin Author

    gc mannequin commented Mar 23, 2015

    Thanks, I will look to the new implementation of ssl in 3.5, and try to adapt it for my project (sldap3). I'd like to help, but I'm not skilled in asynchronous programming so I'm not sure if I succeed.

    Bye,
    Giovanni

    @Elizacat
    Copy link
    Mannequin

    Elizacat mannequin commented Apr 27, 2015

    What needs to be done to make this happen? I can try to implement it.

    @Elizacat
    Copy link
    Mannequin

    Elizacat mannequin commented Apr 27, 2015

    For what it's worth, IRC has an optional STARTTLS extension which is implemented by some servers. IMAP and SMTP also have STARTTLS as a fundamental component of their protocols. As gc pointed out, LDAP also supports it.

    IMO this is a pretty glaring omission.

    @gvanrossum
    Copy link
    Member

    We didn't do this originally because the 3.4 SSL module didn't have this functionality (or maybe it was 3.3 that didn't have this) but now that we do have it I'd be very happy if you could implement this!

    I'm not sure what the right interface is, probably coroutine methods on StreamReader and StreamWriter that take an SSLContext object (and from then on the reader/writer is using SSL), but there also would have to be a lower-level way to do the same thing to a Transport. This would probably have to return a new Transport that uses the original, wrapped transport for reading/writing.

    You probably should write a small test app that proves this works for real too. Perhaps start with a synchronous test app that uses the existing wrap_socket() and then work on the async interface until you can reproduce the same thing there.

    Let us know if you need more information.

    @pitrou
    Copy link
    Member

    pitrou commented Apr 27, 2015

    So you need to:

    • have an API to wrap a clear-text protocol in a SSL protocol (see e.g. BaseProactorEventLoop._make_ssl_transport()... note how there's a waiter argument, what should be done with that?)

    • be able to replace a protocol with another on the transport (perhaps using a new, optional Transport API?)

    • perhaps a higher-level API that combines the two

    Also for convenience a protocol should probably be able to inspect whether it has *already* been wrapped.

    @Elizacat
    Copy link
    Mannequin

    Elizacat mannequin commented Apr 28, 2015

    It seems pretty simple to just make a function that returns a new transport, something like "transport = yield from loop.ssl_wrap_transport(transport)". I'm not sure how to handle plaintext data left on the wire, though, unless that's not really a consideration (given most (all?) real-world protocols can (and usually do) wait for the SSL handshake before sending more data when STARTTLS has been requested).

    For the higher-level API, I'm thinking "reader, writer = asyncio.ssl_wrap(reader, writer)" maybe? You can't have half-closed SSL connections, so you would have to pass them both in.

    As for replacing the protocol but keeping the transport, what would be the semantics of that? I can't really think of how to do that one. I do know SMTP clears all state, but some protocols might not (IRC is a key example - this isn't usually a problem since you are supposed to negotiate it early on before you log onto the server), so this shouldn't be mandatory.

    @Elizacat
    Copy link
    Mannequin

    Elizacat mannequin commented Apr 28, 2015

    Reading the source now (just woke up, sorry!), the new protocol thing makes sense. I'm not sure what to do with the waiter argument or how to handle that.

    What I'm really trying to think of here is how to handle copying of state. I guess users will just have to do it by hand if they want to do that? I don't know if keeping the same protocol is practical and just wrapping the transport is a good idea :(.

    @pitrou
    Copy link
    Member

    pitrou commented Apr 28, 2015

    As for replacing the protocol but keeping the transport, what would
    be the semantics of that?

    The protocol is not really replaced, it's wrapped.

    Before:

    SocketTransport <- UserProtocol

    After:

    SocketTransport <- (asyncio.sslproto.SSLProtocol
    <- asyncio.sslproto._SSLProtocolTransport) <- UserProtocol

    That way, the same SocketTransport (but it could be something else, e.g. a pipe transport) is always bound to the event loop; we simply insert a processing layer in the chain between the original transport and the final protocol. There are two distinct objects so that the SocketTransport sees a protocol and the UserProtocol sees a transport; but those two objects work hand in hand.

    @gvanrossum
    Copy link
    Member

    That sounds like a good plan for the top-level APIs.

    But I think we may need to think about low-level APIs that handle Transports and Protocols as well.

    The design I had been thinking of does not do any socket-level manipulation (in fact it doesn't care if the Transport has a socket or some other way to connect to a networking peer) but it does require some cooperation of the Transport and Protocol.

    Let me draw some ASCII art.

    In the initial state (no ssl) we have an App embodied in a Protocol, talking to a Transport which abstracts away a network connection:

                           data_received()
                        [ <--------------- ]
      App <==> Protocol [                  ] Transport <==> Network
                        [ ---------------> ]
                               write()

    (I.e., when the app wants to write to the network, it calls transport.write(); when the network has data for the app, it calls protocol.data_received().)

    In the final state (ssl established) I was thinking the picture would look like this (leaving the method names out):

                   [ \<---- ]                                    [ \<---- ]
    

    App <=> Protocol [ ] HelperTransport <=> HelperProtocol [ ] Transport <=> Network
    [ ----> ] [ ----> ]

    Here the Protocol at the far left and the Transport at the far right are the same objects as in the first diagram; however we've inserted a pair of helpers that handle SSL. Once the SSL connection is flowing, a write() by the app gives the data to the helper, which gives it to the SSL library. When the SSL library wants to send some (encrypted etc.) data to the network it calls write() on the rightmost Transport (the original one, which still owns the network connection). Conversely, when data arrives over the network, it is still given to the rightmost Transport via its data_received() method. This Transport then gives it to the SSL library, which eventually decrypts it (etc.) and gives it to the helper, which then calls data_received() with the decrypted plaintext on the leftmost Protocol (i.e. the App).

    The clever trick here is that the Protocol on the left still talks to a Transport, it's just a different Transport (owned by the helpr); similarly, the Transport on the right still talks to a Protocol, it's just a different one owned by the helper.

    People have proposed general stacks of Protocol/Transport pairs, but so far I haven't seen much of a use case for that. This might be that use case. In order to switch the arrangements, the helper code (which is ultimately invoked by your loop.ssl_wrap_transport() call) must first create the HelperTransport and HelperProtocol halves of the SSL helper layer, and then call set_transport()/set_protocol() on the existing App Protocol and Network Transport to change their respective associated transport and protocol.

    Note that the relationship between HelperTransport and HelperProtocol is not the same as that between associated Transport/Protocol pairs; the interface they use to talk to each other is completely internal to the implementation of the helper (and probably determined mostly by the needs of the underlying SSL BIO interface).

    Hope this helps (and hope others on the issue agree).

    --Guido

    @pitrou
    Copy link
    Member

    pitrou commented Apr 28, 2015

    Reading the source now (just woke up, sorry!), the new protocol thing makes sense

    Good :-)

    I'm not sure what to do with the waiter argument or how to handle that.

    I'm not sure. Apparently it's used for create_connection(), so perhaps it's not necessary here? Perhaps Victor or Guido can give some insight...

    What I'm really trying to think of here is how to handle copying of state

    Well, there shouldn't be any copying necessary. The user just continues using the same protocol instance; it just calls a different transport.

    @gvanrossum
    Copy link
    Member

    Looks like Antoine drew the same diagram but quicker. :-)

    Regarding the waiter arg, you can leave that None if you don't need it. It is intended to give the caller a way to block (using event loop machinery) until the connection is ready. But if your caller doesn't need it that's fine.

    @Elizacat
    Copy link
    Mannequin

    Elizacat mannequin commented Jun 19, 2015

    After giving this a look over, I think this is over my head. Sorry.

    @vstinner vstinner changed the title asyncio missing wrap_socket asyncio missing wrap_socket (starttls) Jun 23, 2015
    @1st1
    Copy link
    Member

    1st1 commented Aug 5, 2015

    I'm working on porting pypostgresql (pure python postgresql driver) library to use asyncio as its underlying IO machinery. And it appears that PQ3 protocol starts as clear text, and then upgrades to use TLS (if server or connection configured so).

    I've been experimenting with various approaches to how we can design an API for this, and below are some of my thoughts:

    1. The API cannot be implemented on event loop. Loops generally know nothing about the internal structure of transports, i.e. what loop or protocol the transport is attached to.

    2. The API cannot be implemented on protocols. Protocols are decoupled from transports (they only receive a reference to the corresponding transport in their connection_made method). Access to the transport is requires to create an SSL proxy transport/protocol pair.

    3. Therefore, the most convenient place to add the new API are *transports*. I propose to add a 'start_ssl' method to transports with the following signature:

            def start_ssl(self, sslcontext=None,
                          server_side=False, server_hostname=None) -> Transport:

    It will only be implemented on Python 3.5 (because of SSL MemoryBIO requirement).

    Protocols can call 'start_ssl' any time after 'connection_made' is called. 'start_ssl' returns a new Transport (ssl proxy) that has to be used from that moment on. In case the SSL handshake fails, protocol's 'connection_lost' method will be called.

    @gvanrossum
    Copy link
    Member

    Why does the start_tls() function need to know the internal structure of the Transport? I'm hesitant to add this API to Transport; it somehow feels wrong to put such an implementation-specific thing there. E.g. I presume you can't do this for an UDP transport. Or perhaps it could be an API on a subclass of Transport -- then only members of that subclass will support this API.

    @1st1
    Copy link
    Member

    1st1 commented Aug 5, 2015

    Why does the start_tls() function need to know the internal structure of the Transport?

    If start_tls() is added to the loop, it will (likely) have the following signature:

        loop.start_tls(transport)

    then I'd want it to check if the transport is on the same event loop, and after that we'll need to patch 'transport._protocol' with an SSLProtocol wrapper. This requires adding 'get_loop()', 'get_protocol()' and 'set_protocol()' methods to transports (the latter one is kind of useless for anything but 'start_tls').

    We can't implement 'loop.start_tls(protocol)', because protocols don't store a reference to their transports.

    I'm hesitant to add this API to Transport; it somehow feels wrong to put such an implementation-specific thing there. E.g. I presume you can't do this for an UDP transport. Or perhaps it could be an API on a subclass of Transport -- then only members of that subclass will support this API.

    We can add a special subclass of Transport -- TLSTransport (that's how it's done in Twisted, btw: http://goo.gl/iAziWY) with 'start_tls' method raising 'NotImplementedError'. We can later inherit _SelectorSocketTransport and _ProactorSocketTransport classes from it, implementing the method in Python 3.5.

    @gvanrossum
    Copy link
    Member

    OK, got it. SGTM.

    @Elizacat
    Copy link
    Mannequin

    Elizacat mannequin commented Oct 8, 2015

    Therefore, the most convenient place to add the new API are *transports*.

    I had an inkling this was the case, but I didn't know how to go about the creation of a new protocol and transport pair.

    I'm hesitant to add this API to Transport; it somehow feels wrong to put such an implementation-specific thing there. E.g. I presume you can't do this for an UDP transport.

    DTLS (basically TLS over any datagram-oriented protocol, including UDP, SCTP, etc.) exists, so this makes sense, although I don't know if asyncio supports it, but the only major protocol I can think of that uses DTLS is WebRTC.

    In any case, it could potentially make sense for other transport types, if not now, then in the future.

    @1st1
    Copy link
    Member

    1st1 commented Oct 26, 2015

    Guido, Victor,

    Please find attached a first draft of the patch. It's a very early attempt (i.e. I'm not including unit tests/docstrings), and its primary purpose is to gather initial feedback.

    Some points:

    1. As discussed earlier, the primary API point is new transports.TLSTransport class with a start_tls(sslcontetx, *, server_side=False, server_hostname=None) method.

    2. While experimenting with the code and unit tests, I thought that it would be great if stream writers could do start_tls too, this patch has that too. I like this new idea -- makes it so much simpler to write protocols.

    @1st1
    Copy link
    Member

    1st1 commented Oct 26, 2015

    Here's an example client implementation with writer.start_tls() (taken from my debug code):

            @asyncio.coroutine
            def client(addr):
                reader, writer = yield from asyncio.open_connection(
                    *addr, loop=loop)
    
                print("CLIENT: ", (yield from reader.readexactly(4)))
                writer.write(b'ehlo')
                yield from writer.start_tls(sslctx)
                # encrypted channel from this point
                print("CLIENT: ", (yield from reader.readexactly(4)))

    @1st1
    Copy link
    Member

    1st1 commented Nov 2, 2015

    Guido, Victor, any thoughts about the (proto-)patch?

    @1st1
    Copy link
    Member

    1st1 commented May 13, 2016

    I'll create a PR on the GitHub for this. I like the proposed design, and I've implemented an SSL test micro-framework that we can use to test starttls in asyncio.

    @sphawk
    Copy link
    Mannequin

    sphawk mannequin commented Jun 28, 2016

    https://bugs.python.org/review/23749/#msg1
    yuri, did you saw guido added review on your patch?

    @1st1
    Copy link
    Member

    1st1 commented Jun 28, 2016

    yuri, did you saw guido added review on your patch?

    Yes. There are few more issues with the patch that I want to resolve before re-submitting it for another review. Will do it soon.

    @tiran tiran added topic-SSL 3.7 (EOL) end of life labels Sep 15, 2016
    @tiran tiran self-assigned this Sep 15, 2016
    @AlexGrnholm
    Copy link
    Mannequin

    AlexGrnholm mannequin commented Sep 17, 2016

    So is this going to make it into 3.6...?

    @python-dev
    Copy link
    Mannequin

    python-dev mannequin commented Oct 5, 2016

    New changeset 3771a6326725 by Yury Selivanov in branch '3.5':
    asyncio: Add "call_connection_made" arg to SSLProtocol.__init__
    https://hg.python.org/cpython/rev/3771a6326725

    New changeset 3e6739e5c2d0 by Yury Selivanov in branch '3.6':
    Merge 3.5 (issue bpo-23749)
    https://hg.python.org/cpython/rev/3e6739e5c2d0

    New changeset f2204eaba685 by Yury Selivanov in branch 'default':
    Merge 3.6 (issue bpo-23749)
    https://hg.python.org/cpython/rev/f2204eaba685

    @1st1
    Copy link
    Member

    1st1 commented Oct 5, 2016

    With the latest change it's possible to implement starttls
    as a separate package on PyPI, or even by copying/pasting a small
    snipped of code in your project.

    It's expected that we'll figure out the API design for starttls
    during 3.6, so that we can add it in 3.7.

    This issue should be kept open until we have a full public API
    for starttls in asyncio.

    @warsaw
    Copy link
    Member

    warsaw commented May 18, 2017

    I'm very interested in this because, even though we do support STARTTLS in aiosmtpd, it's a hack using non-public symbols, and we have a hidden traceback! (I.e. one that doesn't cause the test suite to fail, but only shows up when clients disconnect.)

    Here's our STARTTLS implementation (at least as of this writing):

    https://github.com/aio-libs/aiosmtpd/blob/master/aiosmtpd/smtp.py#L361

    And here's the bug description:

    aio-libs/aiosmtpd#83

    We're getting eof_received() *after* connection_lost()!

    And the "fix":

    https://github.com/aio-libs/aiosmtpd/pull/101/files

    Basically, once we flip the protocol to the SSLProtocol and then munge the transport, we have to keep the original transport around so that we can close that explicitly on connection_lost().

    I don't really know whether this is 1) the right way to implement STARTTLS, and 2) to handle the traceback fix given the APIs we have to work with today (Python 3.4-3.6). But that's the problem right? :)

    @tiran
    Copy link
    Member

    tiran commented Sep 7, 2017

    I'm removing myself and drop the SSL component. It's really a feature request for asyncio.

    @tiran tiran removed the topic-SSL label Sep 7, 2017
    @tiran tiran assigned 1st1 and unassigned tiran Sep 7, 2017
    @1st1
    Copy link
    Member

    1st1 commented Dec 30, 2017

    New changeset f111b3d by Yury Selivanov in branch 'master':
    bpo-23749: Implement loop.start_tls() (bpo-5039)
    f111b3d

    @1st1 1st1 closed this as completed Dec 30, 2017
    @warsaw
    Copy link
    Member

    warsaw commented Dec 30, 2017

    @yselivanov - thanks for adding this, it's a huge win.

    I think the feature is significant enough for a What's New entry.

    @1st1
    Copy link
    Member

    1st1 commented Dec 30, 2017

    I think the feature is significant enough for a What's New entry.

    Sure, Elvis and I will go through all NEWS items when it's time for what's new ;)

    @vstinner
    Copy link
    Member

    vstinner commented Jan 7, 2018

    http://buildbot.python.org/all/#/builders/58/builds/435

    Tests fail on x86 Windows7 3.x:

    ======================================================================
    ERROR: test_start_tls_client_1 (test.test_asyncio.test_sslproto.ProactorStartTLSTests)
    ----------------------------------------------------------------------

    Traceback (most recent call last):
      File "D:\cygwin\home\db3l\buildarea\3.x.bolen-windows7\build\lib\test\test_asyncio\test_sslproto.py", line 225, in test_start_tls_client_1
        asyncio.wait_for(client(srv.addr), loop=self.loop, timeout=10))
      File "D:\cygwin\home\db3l\buildarea\3.x.bolen-windows7\build\lib\asyncio\base_events.py", line 440, in run_until_complete
        return future.result()
      File "D:\cygwin\home\db3l\buildarea\3.x.bolen-windows7\build\lib\asyncio\tasks.py", line 398, in wait_for
        raise futures.TimeoutError()
    concurrent.futures._base.TimeoutError

    ======================================================================
    ERROR: test_start_tls_client_1 (test.test_asyncio.test_sslproto.SelectorStartTLSTests)
    ----------------------------------------------------------------------

    Traceback (most recent call last):
      File "D:\cygwin\home\db3l\buildarea\3.x.bolen-windows7\build\lib\test\test_asyncio\test_sslproto.py", line 225, in test_start_tls_client_1
        asyncio.wait_for(client(srv.addr), loop=self.loop, timeout=10))
      File "D:\cygwin\home\db3l\buildarea\3.x.bolen-windows7\build\lib\asyncio\base_events.py", line 440, in run_until_complete
        return future.result()
      File "D:\cygwin\home\db3l\buildarea\3.x.bolen-windows7\build\lib\asyncio\tasks.py", line 398, in wait_for
        raise futures.TimeoutError()
    concurrent.futures._base.TimeoutError

    @vstinner vstinner reopened this Jan 7, 2018
    @vstinner
    Copy link
    Member

    vstinner commented Jan 7, 2018

    SelectorStartTLSTests failed once on x86 Tiger 3.x in build 453, but then passed, no idea why.

    http://buildbot.python.org/all/#/builders/30/builds/453

    ======================================================================
    ERROR: test_start_tls_client_1 (test.test_asyncio.test_sslproto.SelectorStartTLSTests)
    ----------------------------------------------------------------------

    Traceback (most recent call last):
      File "/Users/db3l/buildarea/3.x.bolen-tiger/build/Lib/test/test_asyncio/test_sslproto.py", line 225, in test_start_tls_client_1
        asyncio.wait_for(client(srv.addr), loop=self.loop, timeout=10))
      File "/Users/db3l/buildarea/3.x.bolen-tiger/build/Lib/asyncio/base_events.py", line 440, in run_until_complete
        return future.result()
      File "/Users/db3l/buildarea/3.x.bolen-tiger/build/Lib/asyncio/tasks.py", line 398, in wait_for
        raise futures.TimeoutError()
    concurrent.futures._base.TimeoutError

    ======================================================================
    ERROR: test_start_tls_server_1 (test.test_asyncio.test_sslproto.SelectorStartTLSTests)
    ----------------------------------------------------------------------

    Traceback (most recent call last):
      File "/Users/db3l/buildarea/3.x.bolen-tiger/build/Lib/test/test_asyncio/test_sslproto.py", line 285, in test_start_tls_server_1
        asyncio.wait_for(main(), loop=self.loop, timeout=10))
      File "/Users/db3l/buildarea/3.x.bolen-tiger/build/Lib/asyncio/base_events.py", line 440, in run_until_complete
        return future.result()
      File "/Users/db3l/buildarea/3.x.bolen-tiger/build/Lib/asyncio/tasks.py", line 398, in wait_for
        raise futures.TimeoutError()
    concurrent.futures._base.TimeoutError

    @vstinner
    Copy link
    Member

    Tests fail on x86 Windows7 3.x:

    I created a more specific issue: bpo-32645, test_asyncio: TLS tests fail on "x86 Windows7" buildbot.

    @1st1
    Copy link
    Member

    1st1 commented Jan 29, 2018

    New changeset 1e5b25b by Yury Selivanov in branch 'master':
    bpo-23749: Make start-tls tests more stable on win7 buildbot (GH-5409)
    1e5b25b

    @vstinner
    Copy link
    Member

    test_start_tls_server_1() still fails randomly. Example on AppVeyor on my PR 5423:

    https://ci.appveyor.com/project/python/cpython/build/3.7build11472

    ERROR: test_start_tls_server_1 (test.test_asyncio.test_sslproto.SelectorStartTLSTests)
    ----------------------------------------------------------------------

    Traceback (most recent call last):
      File "C:\projects\cpython\lib\test\test_asyncio\test_sslproto.py", line 293, in test_start_tls_server_1
        asyncio.wait_for(main(), loop=self.loop, timeout=10))
      File "C:\projects\cpython\lib\asyncio\base_events.py", line 564, in run_until_complete
        raise RuntimeError('Event loop stopped before Future completed.')
    RuntimeError: Event loop stopped before Future completed.

    I also had this failure on my Windows 10 VM when running "python -m test -R 3:3 -v test_asyncio". I skipped the test to be able to debug bpo-32710.

    @1st1
    Copy link
    Member

    1st1 commented Jan 29, 2018

    I have a feeling that using threads+IO+asyncio makes the test too unstable on some Windows buildbots. I'll rewrite start-tls tests without using threads.

    @asvetlov
    Copy link
    Contributor

    Is the issue done?

    @1st1 1st1 closed this as completed May 28, 2018
    @vstinner
    Copy link
    Member

    I found a race condition in START TLS: bpo-33674. I'm fixing it (I'm just waiting to merge my PR which has already been approved).

    @ezio-melotti ezio-melotti transferred this issue from another repository Apr 10, 2022
    Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
    Labels
    3.7 (EOL) end of life topic-asyncio type-feature A feature request or enhancement
    Projects
    None yet
    Development

    No branches or pull requests

    7 participants