blob: 751174dce8e7664e8b61740024fecdfc9a8bcbd3 [file] [log] [blame]
#!/usr/bin/python3
#
# Copyright 2007 Google Inc. All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""Pure python code for finding unused ports on a host.
This module provides a pure python implementation of PickUnusedPort.
It can also be called via the command line for use in shell scripts.
When called from the command line, it takes one optional argument, which,
if given, is sent to portserver instead of portpicker's PID.
To reserve a port for the lifetime of a bash script, use $BASHPID as this
argument.
There is a race condition between picking a port and your application code
binding to it. The use of a port server to prevent that is recommended on
loaded test hosts running many tests at a time.
If your code can accept a bound socket as input rather than being handed a
port number consider using socket.bind(('localhost', 0)) to bind to an
available port without a race condition rather than using this library.
Typical usage:
test_port = portpicker.PickUnusedPort()
"""
from __future__ import print_function
import os
import random
import socket
import sys
_PROTOS = [(socket.SOCK_STREAM, socket.IPPROTO_TCP),
(socket.SOCK_DGRAM, socket.IPPROTO_UDP)]
def Bind(port, socket_type, socket_proto):
"""Try to bind to a socket of the specified type, protocol, and port.
This is primarily a helper function for PickUnusedPort, used to see
if a particular port number is available.
Args:
port: The port number to bind to, or 0 to have the OS pick a free port.
socket_type: The type of the socket (ex: socket.SOCK_STREAM).
socket_proto: The protocol of the socket (ex: socket.IPPROTO_TCP).
Returns:
The port number on success or None on failure.
"""
s = socket.socket(socket.AF_INET, socket_type, socket_proto)
try:
try:
s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
s.bind(('', port))
return s.getsockname()[1]
except socket.error:
return None
finally:
s.close()
def IsPortFree(port):
"""Check if specified port is free.
Args:
port: integer, port to check
Returns:
boolean, whether it is free to use for both TCP and UDP
"""
return (Bind(port, _PROTOS[0][0], _PROTOS[0][1]) and
Bind(port, _PROTOS[1][0], _PROTOS[1][1]))
def PickUnusedPort(pid=None):
"""A pure python implementation of PickUnusedPort.
Args:
pid: PID to tell the portserver to associate the reservation with. If None,
the current process's PID is used.
Returns:
A port number that is unused on both TCP and UDP.
"""
port = None
# Provide access to the portserver on an opt-in basis.
if 'PORTSERVER_ADDRESS' in os.environ:
port = _GetPortFromPortServer(os.environ['PORTSERVER_ADDRESS'], pid=pid)
if not port:
return _PickUnusedPortWithoutServer()
return port
def _PickUnusedPortWithoutServer():
"""A pure python implementation of PickUnusedPort_NoServer().
This code ensures that the port is available on both TCP and UDP.
This function is an implementation detail of PickUnusedPort(), and
should not be called by code outside of this module.
Returns:
A port number that is unused on both TCP and UDP. None on error.
"""
# Try random ports first.
r = random.Random()
for _ in range(10):
port = int(r.randrange(32768, 60000))
if IsPortFree(port):
return port
# Try OS-assigned ports next.
# Ambrose discovered that on the 2.6 kernel, calling Bind() on UDP socket
# returns the same port over and over. So always try TCP first.
while True:
# Ask the OS for an unused port.
port = Bind(0, _PROTOS[0][0], _PROTOS[0][1])
# Check if this port is unused on the other protocol.
if port and Bind(port, _PROTOS[1][0], _PROTOS[1][1]):
return port
def _GetPortFromPortServer(portserver_address, pid=None):
"""Request a free a port from a system-wide portserver.
This follows a very simple portserver protocol:
The request consists of our pid (in ASCII) followed by a newline.
The response is a port number and a newline, 0 on failure.
This function is an implementation detail of PickUnusedPort(), and
should not be called by code outside of this module.
Args:
portserver_address: The address (path) of a unix domain socket
with which to connect to the portserver. A leading '@'
character indicates an address in the "abstract namespace."
pid: The PID to tell the portserver to associate the reservation with.
If None, the current process's PID is used.
Returns:
The port number on success or None on failure.
"""
if not portserver_address:
return None
# An AF_UNIX address may start with a zero byte, in which case it is in the
# "abstract namespace", and doesn't have any filesystem representation.
# See 'man 7 unix' for details.
# The convention is to write '@' in the address to represent this zero byte.
if portserver_address[0] == '@':
portserver_address = '\0' + portserver_address[1:]
if pid is None:
pid = os.getpid()
try:
# Create socket.
sock = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)
try:
# Connect to portserver.
sock.connect(portserver_address)
# Write request.
sock.sendall(('%d\n' % pid).encode('ascii'))
# Read response.
# 1K should be ample buffer space.
buf = sock.recv(1024)
finally:
sock.close()
except socket.error:
print ('Socket error when connecting to portserver.', file=sys.stderr)
return None
try:
return int(buf.split(b'\n')[0])
except ValueError:
print ('Portserver failed to find a port.', file=sys.stderr)
return None
if __name__ == '__main__':
# If passed an argument, cast it to int and treat it as a PID, otherwise pass
# pid=None to use portpicker's PID.
port = PickUnusedPort(pid=int(sys.argv[1]) if len(sys.argv) > 1 else None)
if not port:
sys.exit(1)
print (port)