Skip to content

Commit a43c2f8

Browse files
committed
Patch #401196: Use getaddrinfo and AF_INET6 in TCP servers and clients.
1 parent d61e3ea commit a43c2f8

7 files changed

Lines changed: 160 additions & 49 deletions

File tree

β€ŽLib/BaseHTTPServer.pyβ€Ž

Lines changed: 4 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -68,8 +68,10 @@
6868
import sys
6969
import time
7070
import socket # For gethostbyaddr()
71+
import string
7172
import mimetools
7273
import SocketServer
74+
import re
7375

7476
# Default error message
7577
DEFAULT_ERROR_MESSAGE = """\
@@ -474,7 +476,8 @@ def test(HandlerClass = BaseHTTPRequestHandler,
474476

475477
httpd = ServerClass(server_address, HandlerClass)
476478

477-
print "Serving HTTP on port", port, "..."
479+
sa = httpd.socket.getsockname()
480+
print "Serving HTTP on", sa[0], "port", sa[1], "..."
478481
httpd.serve_forever()
479482

480483

β€ŽLib/SocketServer.pyβ€Ž

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@
55
For socket-based servers:
66
77
- address family:
8-
- AF_INET: IP (Internet Protocol) sockets (default)
8+
- AF_INET{,6}: IP (Internet Protocol) sockets (default)
99
- AF_UNIX: Unix domain sockets
1010
- others, e.g. AF_DECNET are conceivable (see <socket.h>
1111
- socket type:

β€ŽLib/ftplib.pyβ€Ž

Lines changed: 90 additions & 22 deletions
Original file line numberDiff line numberDiff line change
@@ -108,17 +108,29 @@ def __init__(self, host='', user='', passwd='', acct=''):
108108
self.connect(host)
109109
if user: self.login(user, passwd, acct)
110110

111-
def connect(self, host='', port=0):
112-
'''Connect to host. Arguments are:
113-
- host: hostname to connect to (string, default previous host)
114-
- port: port to connect to (integer, default previous port)'''
115-
if host: self.host = host
116-
if port: self.port = port
117-
self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
118-
self.sock.connect((self.host, self.port))
119-
self.file = self.sock.makefile('rb')
120-
self.welcome = self.getresp()
121-
return self.welcome
111+
def connect(self, host = '', port = 0):
112+
'''Connect to host. Arguments are:
113+
- host: hostname to connect to (string, default previous host)
114+
- port: port to connect to (integer, default previous port)'''
115+
if host: self.host = host
116+
if port: self.port = port
117+
self.passiveserver = 0
118+
for res in socket.getaddrinfo(self.host, self.port, 0, socket.SOCK_STREAM):
119+
af, socktype, proto, canonname, sa = res
120+
try:
121+
self.sock = socket.socket(af, socktype, proto)
122+
self.sock.connect(sa)
123+
except socket.error, msg:
124+
self.sock.close()
125+
self.sock = None
126+
continue
127+
break
128+
if not self.sock:
129+
raise socket.error, msg
130+
self.af = af
131+
self.file = self.sock.makefile('rb')
132+
self.welcome = self.getresp()
133+
return self.welcome
122134

123135
def getwelcome(self):
124136
'''Get the welcome message from the server.
@@ -243,15 +255,48 @@ def sendport(self, host, port):
243255
cmd = 'PORT ' + ','.join(bytes)
244256
return self.voidcmd(cmd)
245257

258+
def sendeprt(self, host, port):
259+
'''Send a EPRT command with the current host and the given port number.'''
260+
af = 0
261+
if self.af == socket.AF_INET:
262+
af = 1
263+
if self.af == socket.AF_INET6:
264+
af = 2
265+
if af == 0:
266+
raise error_proto, 'unsupported address family'
267+
fields = ['', `af`, host, `port`, '']
268+
cmd = 'EPRT ' + string.joinfields(fields, '|')
269+
return self.voidcmd(cmd)
270+
246271
def makeport(self):
247-
'''Create a new socket and send a PORT command for it.'''
248-
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
249-
sock.bind(('', 0))
250-
sock.listen(1)
251-
dummyhost, port = sock.getsockname() # Get proper port
252-
host, dummyport = self.sock.getsockname() # Get proper host
253-
resp = self.sendport(host, port)
254-
return sock
272+
'''Create a new socket and send a PORT command for it.'''
273+
for res in socket.getaddrinfo(None, 0, self.af, socket.SOCK_STREAM, 0, socket.AI_PASSIVE):
274+
af, socktype, proto, canonname, sa = res
275+
try:
276+
sock = socket.socket(af, socktype, proto)
277+
sock.bind(sa)
278+
except socket.error, msg:
279+
sock.close()
280+
sock = None
281+
continue
282+
break
283+
if not sock:
284+
raise socket.error, msg
285+
sock.listen(1)
286+
port = sock.getsockname()[1] # Get proper port
287+
host = self.sock.getsockname()[0] # Get proper host
288+
if self.af == socket.AF_INET:
289+
resp = self.sendport(host, port)
290+
else:
291+
resp = self.sendeprt(host, port)
292+
return sock
293+
294+
def makepasv(self):
295+
if self.af == socket.AF_INET:
296+
host, port = parse227(self.sendcmd('PASV'))
297+
else:
298+
host, port = parse229(self.sendcmd('EPSV'), self.sock.getpeername())
299+
return host, port
255300

256301
def ntransfercmd(self, cmd, rest=None):
257302
"""Initiate a transfer over the data connection.
@@ -270,9 +315,10 @@ def ntransfercmd(self, cmd, rest=None):
270315
"""
271316
size = None
272317
if self.passiveserver:
273-
host, port = parse227(self.sendcmd('PASV'))
274-
conn=socket.socket(socket.AF_INET, socket.SOCK_STREAM)
275-
conn.connect((host, port))
318+
host, port = self.makepasv()
319+
af, socktype, proto, canon, sa = socket.getaddrinfo(host, port, 0, socket.SOCK_STREAM)[0]
320+
conn = socket.socket(af, socktype, proto)
321+
conn.connect(sa)
276322
if rest is not None:
277323
self.sendcmd("REST %s" % rest)
278324
resp = self.sendcmd(cmd)
@@ -523,6 +569,28 @@ def parse227(resp):
523569
return host, port
524570

525571

572+
def parse229(resp, peer):
573+
'''Parse the '229' response for a EPSV request.
574+
Raises error_proto if it does not contain '(|||port|)'
575+
Return ('host.addr.as.numbers', port#) tuple.'''
576+
577+
if resp[:3] <> '229':
578+
raise error_reply, resp
579+
left = string.find(resp, '(')
580+
if left < 0: raise error_proto, resp
581+
right = string.find(resp, ')', left + 1)
582+
if right < 0:
583+
raise error_proto, resp # should contain '(|||port|)'
584+
if resp[left + 1] <> resp[right - 1]:
585+
raise error_proto, resp
586+
parts = string.split(resp[left + 1:right], resp[left+1])
587+
if len(parts) <> 5:
588+
raise error_proto, resp
589+
host = peer[0]
590+
port = string.atoi(parts[3])
591+
return host, port
592+
593+
526594
def parse257(resp):
527595
'''Parse the '257' response for a MKD or PWD request.
528596
This is a response to a MKD or PWD request: a directory name.

β€ŽLib/httplib.pyβ€Ž

Lines changed: 16 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -357,10 +357,22 @@ def set_debuglevel(self, level):
357357

358358
def connect(self):
359359
"""Connect to the host and port specified in __init__."""
360-
self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
361-
if self.debuglevel > 0:
362-
print "connect: (%s, %s)" % (self.host, self.port)
363-
self.sock.connect((self.host, self.port))
360+
for res in socket.getaddrinfo(self.host, self.port, 0, socket.SOCK_STREAM):
361+
af, socktype, proto, canonname, sa = res
362+
try:
363+
self.sock = socket.socket(af, socktype, proto)
364+
if self.debuglevel > 0:
365+
print "connect: (%s, %s)" % (self.host, self.port)
366+
self.sock.connect(sa)
367+
except socket.error, msg:
368+
if self.debuglevel > 0:
369+
print 'connect fail:', (self.host, self.port)
370+
self.sock.close()
371+
self.sock = None
372+
continue
373+
break
374+
if not self.sock:
375+
raise socket.error, msg
364376

365377
def close(self):
366378
"""Close the connection to the HTTP server."""

β€ŽLib/poplib.pyβ€Ž

Lines changed: 17 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -73,13 +73,23 @@ class POP3:
7373

7474

7575
def __init__(self, host, port = POP3_PORT):
76-
self.host = host
77-
self.port = port
78-
self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
79-
self.sock.connect((self.host, self.port))
80-
self.file = self.sock.makefile('rb')
81-
self._debugging = 0
82-
self.welcome = self._getresp()
76+
self.host = host
77+
self.port = port
78+
for res in socket.getaddrinfo(self.host, self.port, 0, socket.SOCK_STREAM):
79+
af, socktype, proto, canonname, sa = res
80+
try:
81+
self.sock = socket.socket(af, socktype, proto)
82+
self.sock.connect(sa)
83+
except socket.error, msg:
84+
self.sock.close()
85+
self.sock = None
86+
continue
87+
break
88+
if not self.sock:
89+
raise socket.error, msg
90+
self.file = self.sock.makefile('rb')
91+
self._debugging = 0
92+
self.welcome = self._getresp()
8393

8494

8595
def _putline(self, line):

β€ŽLib/smtplib.pyβ€Ž

Lines changed: 20 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -208,24 +208,32 @@ def connect(self, host='localhost', port = 0):
208208
specified during instantiation.
209209
210210
"""
211-
if not port:
212-
i = host.find(':')
211+
if not port and (host.find(':') == host.rfind(':')):
212+
i = host.rfind(':')
213213
if i >= 0:
214214
host, port = host[:i], host[i+1:]
215215
try: port = int(port)
216216
except ValueError:
217217
raise socket.error, "nonnumeric port"
218218
if not port: port = SMTP_PORT
219-
self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
220-
if self.debuglevel > 0: print 'connect:', (host, port)
221-
try:
222-
self.sock.connect((host, port))
223-
except socket.error:
224-
self.close()
225-
raise
226-
(code,msg)=self.getreply()
227-
if self.debuglevel >0 : print "connect:", msg
228-
return (code,msg)
219+
if self.debuglevel > 0: print 'connect:', (host, port)
220+
for res in socket.getaddrinfo(host, port, 0, socket.SOCK_STREAM):
221+
af, socktype, proto, canonname, sa = res
222+
try:
223+
self.sock = socket.socket(af, socktype, proto)
224+
if self.debuglevel > 0: print 'connect:', (host, port)
225+
self.sock.connect(sa)
226+
except socket.error, msg:
227+
if self.debuglevel > 0: print 'connect fail:', (host, port)
228+
self.sock.close()
229+
self.sock = None
230+
continue
231+
break
232+
if not self.sock:
233+
raise socket.error, msg
234+
(code, msg) = self.getreply()
235+
if self.debuglevel > 0: print "connect:", msg
236+
return (code, msg)
229237

230238
def send(self, str):
231239
"""Send `str' to the server."""

β€ŽLib/telnetlib.pyβ€Ž

Lines changed: 12 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -136,8 +136,18 @@ def open(self, host, port=0):
136136
port = TELNET_PORT
137137
self.host = host
138138
self.port = port
139-
self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
140-
self.sock.connect((self.host, self.port))
139+
for res in socket.getaddrinfo(host, port, 0, socket.SOCK_STREAM):
140+
af, socktype, proto, canonname, sa = res
141+
try:
142+
self.sock = socket.socket(af, socktype, proto)
143+
self.sock.connect(sa)
144+
except socket.error, msg:
145+
self.sock.close()
146+
self.sock = None
147+
continue
148+
break
149+
if not self.sock:
150+
raise socket.error, msg
141151

142152
def __del__(self):
143153
"""Destructor -- close the connection."""

0 commit comments

Comments
 (0)