Source code for pyndn.transport.async_tcp_transport

# -*- Mode:python; c-file-style:"gnu"; indent-tabs-mode:nil -*- */
#
# Copyright (C) 2015-2019 Regents of the University of California.
# Author: Jeff Thompson <[email protected]>
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with this program.  If not, see <http://www.gnu.org/licenses/>.
# A copy of the GNU Lesser General Public License is in the file COPYING.

"""
This module defines the AsyncTcpTransport class which extends
AsyncSocketTransport for async communication over TCP using Python's asyncio.
This only uses asyncio for communication. To make this thread-safe, you must
dispatch calls to send(), etc. to the asyncio loop using, e.g.,
call_soon_threadsafe, as is done by ThreadsafeFace. To use this, you do not need
to call processEvents.
"""

from pyndn.transport.transport import Transport
from pyndn.transport.tcp_transport import TcpTransport
from pyndn.transport.async_socket_transport import AsyncSocketTransport

[docs]class AsyncTcpTransport(AsyncSocketTransport): """ Create a new AsyncTcpTransport in the unconnected state. This will use the asyncio loop to create the connection and communicate asynchronously. :param loop: The event loop, for example from asyncio.get_event_loop(). It is the responsibility of the application to start and stop the loop. """ def __init__(self, loop): super(AsyncTcpTransport, self).__init__(loop) self._loop = loop self._connectionInfo = None self._isLocal = False
[docs] class ConnectionInfo(Transport.ConnectionInfo): """ Create a new AsyncTcpTransport.ConnectionInfo which extends Transport.ConnectionInfo to hold the host and port info for the TCP connection. :param str host: The host for the connection. :param int port: (optional) The port number for the connection. If omitted, use 6363. """ def __init__(self, host, port = 6363): self._host = host self._port = port
[docs] def getHost(self): """ Get the host given to the constructor. :return: The host. :rtype: str """ return self._host
[docs] def getPort(self): """ Get the port given to the constructor. :return: The port. :rtype: int """ return self._port
[docs] def isLocal(self, connectionInfo): """ Determine whether this transport connecting according to connectionInfo is to a node on the current machine; results are cached. According to http://redmine.named-data.net/projects/nfd/wiki/ScopeControl#local-face, TCP transports with a loopback address are local. If connectionInfo contains a host name, this will do a blocking DNS lookup; otherwise this will parse the IP address and examine the first octet to determine if it is a loopback address (e.g. the first IPv4 octet is 127 or IPv6 is "::1"). :param TcpTransport.ConnectionInfo connectionInfo: A TcpTransport.ConnectionInfo with the host to check. :return: True if the host is local, False if not. :rtype bool: """ if (self._connectionInfo == None or self._connectionInfo.getHost() != connectionInfo.getHost()): # Cache the result in _isLocal and save _connectionInfo for next time. self._isLocal = TcpTransport.getIsLocal(connectionInfo.getHost()) self._connectionInfo = connectionInfo return self._isLocal
[docs] def isAsync(self): """ Override to return true since connect needs to use the onConnected callback. :return: True :rtype bool: """ return True
[docs] def connect(self, connectionInfo, elementListener, onConnected): """ Connect according to the info in connectionInfo, and use elementListener. To be thread-safe, this must be called from a dispatch to the loop which was given to the constructor, as is done by ThreadsafeFace. :param AsyncTcpTransport.ConnectionInfo connectionInfo: An AsyncTcpTransport.ConnectionInfo. :param elementListener: The elementListener must remain valid during the life of this object. :type elementListener: An object with onReceivedElement :param onConnected: This calls onConnected() when the connection is established. :type onConnected: function object """ self._connectHelper( elementListener, self._loop.create_connection( lambda: AsyncSocketTransport._ReceiveProtocol(self, onConnected), connectionInfo.getHost(), connectionInfo.getPort()))