Merge pull request #372 from ralexstokes/add-py36-compatibility

Add py36 compatibility
This commit is contained in:
Alex Stokes 2019-12-10 17:28:49 -08:00 committed by GitHub
commit 733b1d08b6
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
11 changed files with 60 additions and 54 deletions

View File

@ -2,6 +2,9 @@ language: python
matrix: matrix:
include: include:
- python: 3.6-dev
dist: xenial
env: TOXENV=py36-test
- python: 3.7-dev - python: 3.7-dev
dist: xenial dist: xenial
env: TOXENV=py37-test env: TOXENV=py37-test

View File

@ -1,4 +1,5 @@
import asyncio import asyncio
import sys
from .exceptions import RawConnError from .exceptions import RawConnError
from .raw_connection_interface import IRawConnection from .raw_connection_interface import IRawConnection
@ -52,4 +53,6 @@ class RawConnection(IRawConnection):
async def close(self) -> None: async def close(self) -> None:
self.writer.close() self.writer.close()
if sys.version_info < (3, 7):
return
await self.writer.wait_closed() await self.writer.wait_closed()

View File

@ -149,7 +149,7 @@ class Pubsub:
# Map of topic to topic validator # Map of topic to topic validator
self.topic_validators = {} self.topic_validators = {}
self.counter = time.time_ns() self.counter = int(time.time())
self._tasks = [] self._tasks = []
# Call handle peer to keep waiting for updates to peer queue # Call handle peer to keep waiting for updates to peer queue

View File

@ -1,7 +1,8 @@
import asyncio import asyncio
from contextlib import asynccontextmanager
from typing import Any, AsyncIterator, Dict, Tuple, cast from typing import Any, AsyncIterator, Dict, Tuple, cast
# NOTE: import ``asynccontextmanager`` from ``contextlib`` when support for python 3.6 is dropped.
from async_generator import asynccontextmanager
import factory import factory
from libp2p import generate_new_rsa_identity, generate_peer_id_from from libp2p import generate_new_rsa_identity, generate_peer_id_from
@ -173,7 +174,7 @@ async def host_pair_factory(is_secure: bool) -> Tuple[BasicHost, BasicHost]:
return hosts[0], hosts[1] return hosts[0], hosts[1]
@asynccontextmanager @asynccontextmanager # type: ignore
async def pair_of_connected_hosts( async def pair_of_connected_hosts(
is_secure: bool = True is_secure: bool = True
) -> AsyncIterator[Tuple[BasicHost, BasicHost]]: ) -> AsyncIterator[Tuple[BasicHost, BasicHost]]:

View File

@ -143,6 +143,14 @@ floodsub_protocol_pytest_params = [
] ]
def _collect_node_ids(adj_list):
node_ids = set()
for node, neighbors in adj_list.items():
node_ids.add(node)
node_ids.update(set(neighbors))
return node_ids
async def perform_test_from_obj(obj, router_factory) -> None: async def perform_test_from_obj(obj, router_factory) -> None:
""" """
Perform pubsub tests from a test object, which is composed as follows: Perform pubsub tests from a test object, which is composed as follows:
@ -180,59 +188,43 @@ async def perform_test_from_obj(obj, router_factory) -> None:
node_map = {} node_map = {}
pubsub_map = {} pubsub_map = {}
async def add_node(node_id_str: str) -> None: async def add_node(node_id_str: str):
pubsub_router = router_factory(protocols=obj["supported_protocols"]) pubsub_router = router_factory(protocols=obj["supported_protocols"])
pubsub = PubsubFactory(router=pubsub_router) pubsub = PubsubFactory(router=pubsub_router)
await pubsub.host.get_network().listen(LISTEN_MADDR) await pubsub.host.get_network().listen(LISTEN_MADDR)
node_map[node_id_str] = pubsub.host node_map[node_id_str] = pubsub.host
pubsub_map[node_id_str] = pubsub pubsub_map[node_id_str] = pubsub
tasks_connect = [] all_node_ids = _collect_node_ids(adj_list)
for start_node_id in adj_list:
# Create node if node does not yet exist
if start_node_id not in node_map:
await add_node(start_node_id)
# For each neighbor of start_node, create if does not yet exist, for node in all_node_ids:
# then connect start_node to neighbor await add_node(node)
for neighbor_id in adj_list[start_node_id]:
# Create neighbor if neighbor does not yet exist for node, neighbors in adj_list.items():
if neighbor_id not in node_map: for neighbor_id in neighbors:
await add_node(neighbor_id) await connect(node_map[node], node_map[neighbor_id])
tasks_connect.append(
connect(node_map[start_node_id], node_map[neighbor_id]) # NOTE: the test using this routine will fail w/o these sleeps...
) await asyncio.sleep(1)
# Connect nodes and wait at least for 2 seconds
await asyncio.gather(*tasks_connect, asyncio.sleep(2))
# Step 2) Subscribe to topics # Step 2) Subscribe to topics
queues_map = {} queues_map = {}
topic_map = obj["topic_map"] topic_map = obj["topic_map"]
tasks_topic = []
tasks_topic_data = []
for topic, node_ids in topic_map.items(): for topic, node_ids in topic_map.items():
for node_id in node_ids: for node_id in node_ids:
tasks_topic.append(pubsub_map[node_id].subscribe(topic)) queue = await pubsub_map[node_id].subscribe(topic)
tasks_topic_data.append((node_id, topic))
tasks_topic.append(asyncio.sleep(2))
# Gather is like Promise.all
responses = await asyncio.gather(*tasks_topic)
for i in range(len(responses) - 1):
node_id, topic = tasks_topic_data[i]
if node_id not in queues_map: if node_id not in queues_map:
queues_map[node_id] = {} queues_map[node_id] = {}
# Store queue in topic-queue map for node # Store queue in topic-queue map for node
queues_map[node_id][topic] = responses[i] queues_map[node_id][topic] = queue
# Allow time for subscribing before continuing # NOTE: the test using this routine will fail w/o these sleeps...
await asyncio.sleep(0.01) await asyncio.sleep(1)
# Step 3) Publish messages # Step 3) Publish messages
topics_in_msgs_ordered = [] topics_in_msgs_ordered = []
messages = obj["messages"] messages = obj["messages"]
tasks_publish = []
for msg in messages: for msg in messages:
topics = msg["topics"] topics = msg["topics"]
@ -242,21 +234,17 @@ async def perform_test_from_obj(obj, router_factory) -> None:
# Publish message # Publish message
# TODO: Should be single RPC package with several topics # TODO: Should be single RPC package with several topics
for topic in topics: for topic in topics:
tasks_publish.append(pubsub_map[node_id].publish(topic, data)) await pubsub_map[node_id].publish(topic, data)
# For each topic in topics, add (topic, node_id, data) tuple to ordered test list # For each topic in topics, add (topic, node_id, data) tuple to ordered test list
for topic in topics:
topics_in_msgs_ordered.append((topic, node_id, data)) topics_in_msgs_ordered.append((topic, node_id, data))
# Allow time for publishing before continuing
await asyncio.gather(*tasks_publish, asyncio.sleep(2))
# Step 4) Check that all messages were received correctly. # Step 4) Check that all messages were received correctly.
for topic, origin_node_id, data in topics_in_msgs_ordered: for topic, origin_node_id, data in topics_in_msgs_ordered:
# Look at each node in each topic # Look at each node in each topic
for node_id in topic_map[topic]: for node_id in topic_map[topic]:
# Get message from subscription queue # Get message from subscription queue
msg = await queues_map[node_id][topic].get() queue = queues_map[node_id][topic]
msg = await queue.get()
assert data == msg.data assert data == msg.data
# Check the message origin # Check the message origin
assert node_map[origin_node_id].get_id().to_bytes() == msg.from_id assert node_map[origin_node_id].get_id().to_bytes() == msg.from_id

View File

@ -1,5 +1,6 @@
import asyncio import asyncio
from socket import socket from socket import socket
import sys
from typing import List from typing import List
from multiaddr import Multiaddr from multiaddr import Multiaddr
@ -53,8 +54,11 @@ class TCPListener(IListener):
if self.server is None: if self.server is None:
return return
self.server.close() self.server.close()
await self.server.wait_closed() server = self.server
self.server = None self.server = None
if sys.version_info < (3, 7):
return
await server.wait_closed()
class TCP(ITransport): class TCP(ITransport):

View File

@ -58,6 +58,8 @@ install_requires = [
"protobuf>=3.10.0,<4.0.0", "protobuf>=3.10.0,<4.0.0",
"coincurve>=10.0.0,<11.0.0", "coincurve>=10.0.0,<11.0.0",
"pynacl==1.3.0", "pynacl==1.3.0",
"dataclasses>=0.7, <1;python_version<'3.7'",
"async_generator==1.10",
] ]
@ -80,7 +82,7 @@ setup(
url="https://github.com/libp2p/py-libp2p", url="https://github.com/libp2p/py-libp2p",
include_package_data=True, include_package_data=True,
install_requires=install_requires, install_requires=install_requires,
python_requires=">=3.7,<4", python_requires=">=3.6,<4",
extras_require=extras_require, extras_require=extras_require,
py_modules=["libp2p"], py_modules=["libp2p"],
license="MIT/APACHE2.0", license="MIT/APACHE2.0",
@ -94,6 +96,7 @@ setup(
"License :: OSI Approved :: Apache Software License", "License :: OSI Approved :: Apache Software License",
"Natural Language :: English", "Natural Language :: English",
"Programming Language :: Python :: 3", "Programming Language :: Python :: 3",
"Programming Language :: Python :: 3.6",
"Programming Language :: Python :: 3.7", "Programming Language :: Python :: 3.7",
], ],
platforms=["unix", "linux", "osx"], platforms=["unix", "linux", "osx"],

View File

@ -58,11 +58,11 @@ async def test_peers_subscribe(pubsubs_fsub):
await connect(pubsubs_fsub[0].host, pubsubs_fsub[1].host) await connect(pubsubs_fsub[0].host, pubsubs_fsub[1].host)
await pubsubs_fsub[0].subscribe(TESTING_TOPIC) await pubsubs_fsub[0].subscribe(TESTING_TOPIC)
# Yield to let 0 notify 1 # Yield to let 0 notify 1
await asyncio.sleep(0.1) await asyncio.sleep(1)
assert pubsubs_fsub[0].my_id in pubsubs_fsub[1].peer_topics[TESTING_TOPIC] assert pubsubs_fsub[0].my_id in pubsubs_fsub[1].peer_topics[TESTING_TOPIC]
await pubsubs_fsub[0].unsubscribe(TESTING_TOPIC) await pubsubs_fsub[0].unsubscribe(TESTING_TOPIC)
# Yield to let 0 notify 1 # Yield to let 0 notify 1
await asyncio.sleep(0.1) await asyncio.sleep(1)
assert pubsubs_fsub[0].my_id not in pubsubs_fsub[1].peer_topics[TESTING_TOPIC] assert pubsubs_fsub[0].my_id not in pubsubs_fsub[1].peer_topics[TESTING_TOPIC]

View File

@ -76,8 +76,8 @@ async def test_create_secure_session():
local_conn = InMemoryConnection(local_peer, is_initiator=True) local_conn = InMemoryConnection(local_peer, is_initiator=True)
remote_conn = InMemoryConnection(remote_peer) remote_conn = InMemoryConnection(remote_peer)
local_pipe_task = asyncio.create_task(create_pipe(local_conn, remote_conn)) local_pipe_task = asyncio.ensure_future(create_pipe(local_conn, remote_conn))
remote_pipe_task = asyncio.create_task(create_pipe(remote_conn, local_conn)) remote_pipe_task = asyncio.ensure_future(create_pipe(remote_conn, local_conn))
local_session_builder = create_secure_session( local_session_builder = create_secure_session(
local_nonce, local_peer, local_key_pair.private_key, local_conn, remote_peer local_nonce, local_peer, local_key_pair.private_key, local_conn, remote_peer

View File

@ -151,6 +151,8 @@ class DaemonStream(ReadWriteCloser):
async def close(self) -> None: async def close(self) -> None:
self.writer.close() self.writer.close()
if sys.version_info < (3, 7):
return
await self.writer.wait_closed() await self.writer.wait_closed()
async def read(self, n: int = -1) -> bytes: async def read(self, n: int = -1) -> bytes:
@ -196,6 +198,7 @@ async def py_to_daemon_stream_pair(hosts, p2pds, is_to_fail_daemon_stream):
# some day. # some day.
listener = p2pds[0].control.control.listener listener = p2pds[0].control.control.listener
listener.close() listener.close()
if sys.version_info[0:2] > (3, 6):
await listener.wait_closed() await listener.wait_closed()
stream_py = await host.new_stream(p2pd.peer_id, [protocol_id]) stream_py = await host.new_stream(p2pd.peer_id, [protocol_id])
if not is_to_fail_daemon_stream: if not is_to_fail_daemon_stream:

View File

@ -1,9 +1,9 @@
# Reference: https://github.com/ethereum/ethereum-python-project-template/blob/master/tox.ini # Reference: https://github.com/ethereum/ethereum-python-project-template/blob/master/tox.ini
# TODO: consider py36 and pypy3 support # TODO: consider pypy3 support
[tox] [tox]
envlist = envlist =
py37-test py{36,37}-test
py37-interop py37-interop
lint lint
docs docs
@ -37,6 +37,7 @@ commands =
basepython = basepython =
docs: python docs: python
py37: python3.7 py37: python3.7
py36: python3.6
extras = extras =
test test
docs: doc docs: doc