blob: a8ed4b7dc72da5f6a193ce418aacd81ed5ed5367 [file] [log] [blame]
Mark Slee89e2bb82007-03-01 00:20:36 +00001#
David Reissea2cba82009-03-30 21:35:00 +00002# Licensed to the Apache Software Foundation (ASF) under one
3# or more contributor license agreements. See the NOTICE file
4# distributed with this work for additional information
5# regarding copyright ownership. The ASF licenses this file
6# to you under the Apache License, Version 2.0 (the
7# "License"); you may not use this file except in compliance
8# with the License. You may obtain a copy of the License at
9#
10# http://www.apache.org/licenses/LICENSE-2.0
11#
12# Unless required by applicable law or agreed to in writing,
13# software distributed under the License is distributed on an
14# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15# KIND, either express or implied. See the License for the
16# specific language governing permissions and limitations
17# under the License.
18#
Mark Slee89e2bb82007-03-01 00:20:36 +000019
David Reisse29995e2008-07-31 20:15:17 +000020import errno
Bryan Duxbury69720412012-01-03 17:32:30 +000021import os
Mark Sleecde2b612006-09-03 21:13:07 +000022import socket
David Reiss73af3b72010-08-30 21:57:07 +000023import sys
Mark Sleecde2b612006-09-03 21:13:07 +000024
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090025from .TTransport import TTransportBase, TTransportException, TServerTransportBase
Bryan Duxbury69720412012-01-03 17:32:30 +000026
27
David Reisse29995e2008-07-31 20:15:17 +000028class TSocketBase(TTransportBase):
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090029 def _resolveAddr(self):
30 if self._unix_socket is not None:
31 return [(socket.AF_UNIX, socket.SOCK_STREAM, None, None,
32 self._unix_socket)]
33 else:
34 return socket.getaddrinfo(self.host,
35 self.port,
36 self._socket_family,
37 socket.SOCK_STREAM,
38 0,
39 socket.AI_PASSIVE | socket.AI_ADDRCONFIG)
Mark Sleecde2b612006-09-03 21:13:07 +000040
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090041 def close(self):
42 if self.handle:
43 self.handle.close()
44 self.handle = None
David Reisse29995e2008-07-31 20:15:17 +000045
Bryan Duxbury69720412012-01-03 17:32:30 +000046
David Reisse29995e2008-07-31 20:15:17 +000047class TSocket(TSocketBase):
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090048 """Socket implementation of TTransport base."""
Mark Sleecde2b612006-09-03 21:13:07 +000049
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090050 def __init__(self, host='localhost', port=9090, unix_socket=None, socket_family=socket.AF_UNSPEC):
51 """Initialize a TSocket
David Reissc16a8f62007-12-14 23:46:47 +000052
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090053 @param host(str) The host to connect to.
54 @param port(int) The (TCP) port to connect to.
55 @param unix_socket(str) The filename of a unix socket to connect to.
56 (host and port will be ignored.)
57 @param socket_family(int) The socket family to use with this socket.
58 """
59 self.host = host
60 self.port = port
61 self.handle = None
62 self._unix_socket = unix_socket
63 self._timeout = None
64 self._socket_family = socket_family
David Reiss0c90f6f2008-02-06 22:18:40 +000065
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090066 def setHandle(self, h):
67 self.handle = h
Mark Sleec9676562006-09-05 17:34:52 +000068
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090069 def isOpen(self):
70 return self.handle is not None
Aditya Agarwalf954f972007-02-06 01:26:12 +000071
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090072 def setTimeout(self, ms):
73 if ms is None:
74 self._timeout = None
75 else:
76 self._timeout = ms / 1000.0
David Reiss0c90f6f2008-02-06 22:18:40 +000077
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090078 if self.handle is not None:
79 self.handle.settimeout(self._timeout)
Mark Sleecde2b612006-09-03 21:13:07 +000080
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090081 def open(self):
Mark Slee22974602007-07-06 22:20:19 +000082 try:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090083 res0 = self._resolveAddr()
84 for res in res0:
85 self.handle = socket.socket(res[0], res[1])
86 self.handle.settimeout(self._timeout)
87 try:
88 self.handle.connect(res[4])
89 except socket.error as e:
90 if res is not res0[-1]:
91 continue
92 else:
93 raise e
94 break
jfarrelld565e2f2015-03-18 21:02:47 -040095 except socket.error as e:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090096 if self._unix_socket:
97 message = 'Could not connect to socket %s' % self._unix_socket
98 else:
99 message = 'Could not connect to %s:%d' % (self.host, self.port)
100 raise TTransportException(type=TTransportException.NOT_OPEN,
101 message=message)
Mark Sleecde2b612006-09-03 21:13:07 +0000102
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900103 def read(self, sz):
104 try:
105 buff = self.handle.recv(sz)
106 except socket.error as e:
107 if (e.args[0] == errno.ECONNRESET and
108 (sys.platform == 'darwin' or sys.platform.startswith('freebsd'))):
109 # freebsd and Mach don't follow POSIX semantic of recv
110 # and fail with ECONNRESET if peer performed shutdown.
111 # See corresponding comment and code in TSocket::read()
112 # in lib/cpp/src/transport/TSocket.cpp.
113 self.close()
114 # Trigger the check to raise the END_OF_FILE exception below.
115 buff = ''
116 else:
117 raise
118 if len(buff) == 0:
119 raise TTransportException(type=TTransportException.END_OF_FILE,
120 message='TSocket read 0 bytes')
121 return buff
Mark Sleecde2b612006-09-03 21:13:07 +0000122
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900123 def write(self, buff):
124 if not self.handle:
125 raise TTransportException(type=TTransportException.NOT_OPEN,
126 message='Transport not open')
127 sent = 0
128 have = len(buff)
129 while sent < have:
130 plus = self.handle.send(buff)
131 if plus == 0:
132 raise TTransportException(type=TTransportException.END_OF_FILE,
133 message='TSocket sent 0 bytes')
134 sent += plus
135 buff = buff[plus:]
Mark Sleecde2b612006-09-03 21:13:07 +0000136
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900137 def flush(self):
138 pass
Mark Sleec9676562006-09-05 17:34:52 +0000139
Bryan Duxbury69720412012-01-03 17:32:30 +0000140
David Reissd73255d2009-03-24 22:51:02 +0000141class TServerSocket(TSocketBase, TServerTransportBase):
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900142 """Socket implementation of TServerTransport base."""
Mark Sleec9676562006-09-05 17:34:52 +0000143
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900144 def __init__(self, host=None, port=9090, unix_socket=None, socket_family=socket.AF_UNSPEC):
145 self.host = host
146 self.port = port
147 self._unix_socket = unix_socket
148 self._socket_family = socket_family
149 self.handle = None
Mark Slee256bdc42007-11-27 08:42:19 +0000150
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900151 def listen(self):
152 res0 = self._resolveAddr()
153 socket_family = self._socket_family == socket.AF_UNSPEC and socket.AF_INET6 or self._socket_family
154 for res in res0:
155 if res[0] is socket_family or res is res0[-1]:
156 break
Mark Slee22974602007-07-06 22:20:19 +0000157
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900158 # We need remove the old unix socket if the file exists and
159 # nobody is listening on it.
160 if self._unix_socket:
161 tmp = socket.socket(res[0], res[1])
162 try:
163 tmp.connect(res[4])
164 except socket.error as err:
165 eno, message = err.args
166 if eno == errno.ECONNREFUSED:
167 os.unlink(res[4])
David Reisse29995e2008-07-31 20:15:17 +0000168
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900169 self.handle = socket.socket(res[0], res[1])
170 self.handle.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
171 if hasattr(self.handle, 'settimeout'):
172 self.handle.settimeout(None)
173 self.handle.bind(res[4])
174 self.handle.listen(128)
Mark Sleec9676562006-09-05 17:34:52 +0000175
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900176 def accept(self):
177 client, addr = self.handle.accept()
178 result = TSocket()
179 result.setHandle(client)
180 return result