blob: 59a06ad2ada6151e4dd7a945e5b648bc65248038 [file] [log] [blame]
Mark Slee57cc25e2007-02-28 21:43:54 +00001#!/usr/bin/env python
Mark Sleec9676562006-09-05 17:34:52 +00002
David Reissea2cba82009-03-30 21:35:00 +00003#
4# Licensed to the Apache Software Foundation (ASF) under one
5# or more contributor license agreements. See the NOTICE file
6# distributed with this work for additional information
7# regarding copyright ownership. The ASF licenses this file
8# to you under the Apache License, Version 2.0 (the
9# "License"); you may not use this file except in compliance
10# with the License. You may obtain a copy of the License at
11#
12# http://www.apache.org/licenses/LICENSE-2.0
13#
14# Unless required by applicable law or agreed to in writing,
15# software distributed under the License is distributed on an
16# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
17# KIND, either express or implied. See the License for the
18# specific language governing permissions and limitations
19# under the License.
20#
Jens Geyerd629ea02015-09-23 21:16:50 +020021import logging
22import os
James E. King III6f8c99e2018-03-24 16:32:02 -040023import signal
Dmytro Shteflyuk0b682282026-02-04 16:26:46 -050024import ssl
Jens Geyerd629ea02015-09-23 21:16:50 +020025import sys
26import time
Bryan Duxbury59d4efd2011-03-21 17:38:22 +000027from optparse import OptionParser
Mark Sleec9676562006-09-05 17:34:52 +000028
Nobuaki Sukegawa7af189a2016-02-11 16:21:01 +090029from util import local_libpath
James E. King III9804ab92019-02-07 16:59:05 -050030sys.path.insert(0, local_libpath())
31from thrift.protocol import TProtocol, TProtocolDecorator
Gregg Donovan62ec9292026-01-29 16:51:37 -050032from thrift.Thrift import TException
Nobuaki Sukegawa7af189a2016-02-11 16:21:01 +090033
Nobuaki Sukegawaa185d7e2015-11-06 21:24:24 +090034SCRIPT_DIR = os.path.abspath(os.path.dirname(__file__))
Roger Meierf4eec7a2011-09-11 18:16:21 +000035
Mark Sleec9676562006-09-05 17:34:52 +000036
Jens Geyerd629ea02015-09-23 21:16:50 +020037class TestHandler(object):
Carel Combrink5abe53f2025-11-13 06:16:41 +010038 def __init__(self, options):
39 self.options = options
40
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090041 def testVoid(self):
Carel Combrink5abe53f2025-11-13 06:16:41 +010042 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090043 logging.info('testVoid()')
Mark Sleec9676562006-09-05 17:34:52 +000044
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090045 def testString(self, str):
Carel Combrink5abe53f2025-11-13 06:16:41 +010046 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090047 logging.info('testString(%s)' % str)
48 return str
Mark Sleec9676562006-09-05 17:34:52 +000049
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090050 def testBool(self, boolean):
Carel Combrink5abe53f2025-11-13 06:16:41 +010051 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090052 logging.info('testBool(%s)' % str(boolean).lower())
53 return boolean
Nobuaki Sukegawaa649e742015-09-21 13:53:25 +090054
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090055 def testByte(self, byte):
Carel Combrink5abe53f2025-11-13 06:16:41 +010056 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090057 logging.info('testByte(%d)' % byte)
58 return byte
Mark Sleec9676562006-09-05 17:34:52 +000059
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090060 def testI16(self, i16):
Carel Combrink5abe53f2025-11-13 06:16:41 +010061 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090062 logging.info('testI16(%d)' % i16)
63 return i16
Mark Sleec98d0502006-09-06 02:42:25 +000064
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090065 def testI32(self, i32):
Carel Combrink5abe53f2025-11-13 06:16:41 +010066 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090067 logging.info('testI32(%d)' % i32)
68 return i32
Mark Sleec98d0502006-09-06 02:42:25 +000069
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090070 def testI64(self, i64):
Carel Combrink5abe53f2025-11-13 06:16:41 +010071 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090072 logging.info('testI64(%d)' % i64)
73 return i64
Mark Sleec98d0502006-09-06 02:42:25 +000074
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090075 def testDouble(self, dub):
Carel Combrink5abe53f2025-11-13 06:16:41 +010076 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090077 logging.info('testDouble(%f)' % dub)
78 return dub
Mark Sleec98d0502006-09-06 02:42:25 +000079
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090080 def testBinary(self, thing):
Carel Combrink5abe53f2025-11-13 06:16:41 +010081 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090082 logging.info('testBinary()') # TODO: hex output
83 return thing
Jens Geyerd629ea02015-09-23 21:16:50 +020084
Carel Combrinka715bdf2025-10-30 07:44:21 +010085 def testUuid(self, thing):
86 if self.options.verbose > 1:
87 logging.info('testUuid(%s)' % thing)
88 return thing
89
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090090 def testStruct(self, thing):
Carel Combrink5abe53f2025-11-13 06:16:41 +010091 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090092 logging.info('testStruct({%s, %s, %s, %s})' % (thing.string_thing, thing.byte_thing, thing.i32_thing, thing.i64_thing))
93 return thing
Mark Sleec98d0502006-09-06 02:42:25 +000094
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090095 def testException(self, arg):
Carel Combrink5abe53f2025-11-13 06:16:41 +010096 from ThriftTest.ttypes import Xception
97 # if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +090098 logging.info('testException(%s)' % arg)
99 if arg == 'Xception':
100 raise Xception(errorCode=1001, message=arg)
101 elif arg == 'TException':
102 raise TException(message='This is a TException')
Roger Meier1f554e12013-01-05 20:38:35 +0100103
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900104 def testMultiException(self, arg0, arg1):
Carel Combrink5abe53f2025-11-13 06:16:41 +0100105 from ThriftTest.ttypes import Xtruct, Xception, Xception2
106
107 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900108 logging.info('testMultiException(%s, %s)' % (arg0, arg1))
109 if arg0 == 'Xception':
110 raise Xception(errorCode=1001, message='This is an Xception')
111 elif arg0 == 'Xception2':
112 raise Xception2(
113 errorCode=2002,
114 struct_thing=Xtruct(string_thing='This is an Xception2'))
115 return Xtruct(string_thing=arg1)
Mark Sleec9676562006-09-05 17:34:52 +0000116
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900117 def testOneway(self, seconds):
Carel Combrink5abe53f2025-11-13 06:16:41 +0100118 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900119 logging.info('testOneway(%d) => sleeping...' % seconds)
120 time.sleep(seconds / 3) # be quick
Carel Combrink5abe53f2025-11-13 06:16:41 +0100121 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900122 logging.info('done sleeping')
David Reissdb893b62008-02-18 02:11:48 +0000123
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900124 def testNest(self, thing):
Carel Combrink5abe53f2025-11-13 06:16:41 +0100125 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900126 logging.info('testNest(%s)' % thing)
127 return thing
David Reiss74421272008-11-07 23:09:31 +0000128
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900129 def testMap(self, thing):
Carel Combrink5abe53f2025-11-13 06:16:41 +0100130 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900131 logging.info('testMap(%s)' % thing)
132 return thing
Jens Geyerd629ea02015-09-23 21:16:50 +0200133
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900134 def testStringMap(self, thing):
Carel Combrink5abe53f2025-11-13 06:16:41 +0100135 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900136 logging.info('testStringMap(%s)' % thing)
137 return thing
David Reiss74421272008-11-07 23:09:31 +0000138
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900139 def testSet(self, thing):
Carel Combrink5abe53f2025-11-13 06:16:41 +0100140 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900141 logging.info('testSet(%s)' % thing)
142 return thing
David Reiss74421272008-11-07 23:09:31 +0000143
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900144 def testList(self, thing):
Carel Combrink5abe53f2025-11-13 06:16:41 +0100145 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900146 logging.info('testList(%s)' % thing)
147 return thing
David Reiss74421272008-11-07 23:09:31 +0000148
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900149 def testEnum(self, thing):
Carel Combrink5abe53f2025-11-13 06:16:41 +0100150 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900151 logging.info('testEnum(%s)' % thing)
152 return thing
David Reiss74421272008-11-07 23:09:31 +0000153
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900154 def testTypedef(self, thing):
Carel Combrink5abe53f2025-11-13 06:16:41 +0100155 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900156 logging.info('testTypedef(%s)' % thing)
157 return thing
David Reiss74421272008-11-07 23:09:31 +0000158
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900159 def testMapMap(self, thing):
Carel Combrink5abe53f2025-11-13 06:16:41 +0100160 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900161 logging.info('testMapMap(%s)' % thing)
162 return {
163 -4: {
164 -4: -4,
165 -3: -3,
166 -2: -2,
167 -1: -1,
168 },
169 4: {
170 4: 4,
171 3: 3,
172 2: 2,
173 1: 1,
174 },
175 }
Roger Meier1f554e12013-01-05 20:38:35 +0100176
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900177 def testInsanity(self, argument):
Carel Combrink5abe53f2025-11-13 06:16:41 +0100178 from ThriftTest.ttypes import Insanity
179
180 if self.options.verbose > 1:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900181 logging.info('testInsanity(%s)' % argument)
182 return {
183 1: {
184 2: argument,
185 3: argument,
186 },
187 2: {6: Insanity()},
188 }
Bryan Duxbury59d4efd2011-03-21 17:38:22 +0000189
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900190 def testMulti(self, arg0, arg1, arg2, arg3, arg4, arg5):
Carel Combrink5abe53f2025-11-13 06:16:41 +0100191 from ThriftTest.ttypes import Xtruct
192
193 if self.options.verbose > 1:
Jens Geyer506e3112020-05-17 22:48:51 +0200194 logging.info('testMulti(%s, %s, %s, %s, %s, %s)' % (arg0, arg1, arg2, arg3, arg4, arg5))
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900195 return Xtruct(string_thing='Hello2',
196 byte_thing=arg0, i32_thing=arg1, i64_thing=arg2)
Roger Meier1f554e12013-01-05 20:38:35 +0100197
Bryan Duxbury59d4efd2011-03-21 17:38:22 +0000198
James E. King III9804ab92019-02-07 16:59:05 -0500199class SecondHandler(object):
200 def secondtestString(self, argument):
201 return "testString(\"" + argument + "\")"
202
203
204# LAST_SEQID is a global because we have one transport and multiple protocols
205# running on it (when multiplexed)
206LAST_SEQID = None
207
208
209class TPedanticSequenceIdProtocolWrapper(TProtocolDecorator.TProtocolDecorator):
210 """
211 Wraps any protocol with sequence ID checking: looks for outbound
212 uniqueness as well as request/response alignment.
213 """
Dmytro Shteflyukacbcf102026-02-13 18:25:55 -0500214
James E. King III9804ab92019-02-07 16:59:05 -0500215 def __init__(self, protocol):
216 # TProtocolDecorator.__new__ does all the heavy lifting
217 pass
218
219 def readMessageBegin(self):
220 global LAST_SEQID
221 (name, type, seqid) =\
222 super(TPedanticSequenceIdProtocolWrapper, self).readMessageBegin()
223 if LAST_SEQID is not None and LAST_SEQID == seqid:
224 raise TProtocol.TProtocolException(
225 TProtocol.TProtocolException.INVALID_DATA,
226 "We received the same seqid {0} twice in a row".format(seqid))
227 LAST_SEQID = seqid
228 return (name, type, seqid)
229
230
231def make_pedantic(proto):
232 """ Wrap a protocol in the pedantic sequence ID wrapper. """
233 # NOTE: this is disabled for now as many clients send seqid
234 # of zero and that is okay, need a way to identify
235 # clients that MUST send seqid unique to function right
236 # or just force all implementations to send unique seqids (preferred)
237 return proto # TPedanticSequenceIdProtocolWrapper(proto)
238
239
240class TPedanticSequenceIdProtocolFactory(TProtocol.TProtocolFactory):
241 def __init__(self, encapsulated):
242 super(TPedanticSequenceIdProtocolFactory, self).__init__()
243 self.encapsulated = encapsulated
244
245 def getProtocol(self, trans):
246 return make_pedantic(self.encapsulated.getProtocol(trans))
247
248
Nobuaki Sukegawaa185d7e2015-11-06 21:24:24 +0900249def main(options):
James E. King III9804ab92019-02-07 16:59:05 -0500250 # common header allowed client types
251 allowed_client_types = [
252 THeaderTransport.THeaderClientType.HEADERS,
253 THeaderTransport.THeaderClientType.FRAMED_BINARY,
254 THeaderTransport.THeaderClientType.UNFRAMED_BINARY,
255 THeaderTransport.THeaderClientType.FRAMED_COMPACT,
256 THeaderTransport.THeaderClientType.UNFRAMED_COMPACT,
257 ]
258
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900259 # set up the protocol factory form the --protocol option
260 prot_factories = {
Neil Williams66a44c52018-08-13 16:12:24 -0700261 'accel': TBinaryProtocol.TBinaryProtocolAcceleratedFactory(),
James E. King III9804ab92019-02-07 16:59:05 -0500262 'multia': TBinaryProtocol.TBinaryProtocolAcceleratedFactory(),
Neil Williams66a44c52018-08-13 16:12:24 -0700263 'accelc': TCompactProtocol.TCompactProtocolAcceleratedFactory(),
James E. King III9804ab92019-02-07 16:59:05 -0500264 'multiac': TCompactProtocol.TCompactProtocolAcceleratedFactory(),
265 'binary': TPedanticSequenceIdProtocolFactory(TBinaryProtocol.TBinaryProtocolFactory()),
266 'multi': TPedanticSequenceIdProtocolFactory(TBinaryProtocol.TBinaryProtocolFactory()),
Neil Williams66a44c52018-08-13 16:12:24 -0700267 'compact': TCompactProtocol.TCompactProtocolFactory(),
James E. King III9804ab92019-02-07 16:59:05 -0500268 'multic': TCompactProtocol.TCompactProtocolFactory(),
269 'header': THeaderProtocol.THeaderProtocolFactory(allowed_client_types),
270 'multih': THeaderProtocol.THeaderProtocolFactory(allowed_client_types),
Neil Williams66a44c52018-08-13 16:12:24 -0700271 'json': TJSONProtocol.TJSONProtocolFactory(),
James E. King III9804ab92019-02-07 16:59:05 -0500272 'multij': TJSONProtocol.TJSONProtocolFactory(),
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900273 }
Neil Williams66a44c52018-08-13 16:12:24 -0700274 pfactory = prot_factories.get(options.proto, None)
275 if pfactory is None:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900276 raise AssertionError('Unknown --protocol option: %s' % options.proto)
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900277 try:
278 pfactory.string_length_limit = options.string_limit
279 pfactory.container_length_limit = options.container_limit
James E. King, III350fe752017-10-25 09:57:18 -0400280 except Exception:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900281 # Ignore errors for those protocols that does not support length limit
282 pass
Bryan Duxbury16066592011-03-22 18:06:04 +0000283
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900284 # get the server type (TSimpleServer, TNonblockingServer, etc...)
285 if len(args) > 1:
286 raise AssertionError('Only one server type may be specified, not multiple types.')
287 server_type = args[0]
James E. King III6f8c99e2018-03-24 16:32:02 -0400288 if options.trans == 'http':
289 server_type = 'THttpServer'
David Reissbcaa2ad2008-06-10 22:55:26 +0000290
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900291 # Set up the handler and processor objects
Carel Combrink5abe53f2025-11-13 06:16:41 +0100292 handler = TestHandler(options)
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900293 processor = ThriftTest.Processor(handler)
Bryan Duxbury16066592011-03-22 18:06:04 +0000294
James E. King III9804ab92019-02-07 16:59:05 -0500295 if options.proto.startswith('multi'):
296 secondHandler = SecondHandler()
297 secondProcessor = SecondService.Processor(secondHandler)
298
299 multiplexedProcessor = TMultiplexedProcessor()
300 multiplexedProcessor.registerDefault(processor)
301 multiplexedProcessor.registerProcessor('ThriftTest', processor)
302 multiplexedProcessor.registerProcessor('SecondService', secondProcessor)
303 processor = multiplexedProcessor
304
James E. King III6f8c99e2018-03-24 16:32:02 -0400305 global server
306
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900307 # Handle THttpServer as a special case
308 if server_type == 'THttpServer':
James E. King III6f8c99e2018-03-24 16:32:02 -0400309 if options.ssl:
310 __certfile = os.path.join(os.path.dirname(SCRIPT_DIR), "keys", "server.crt")
311 __keyfile = os.path.join(os.path.dirname(SCRIPT_DIR), "keys", "server.key")
312 server = THttpServer.THttpServer(processor, ('', options.port), pfactory, cert_file=__certfile, key_file=__keyfile)
313 else:
314 server = THttpServer.THttpServer(processor, ('', options.port), pfactory)
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900315 server.serve()
316 sys.exit(0)
317
318 # set up server transport and transport factory
319
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900320 host = None
321 if options.ssl:
322 from thrift.transport import TSSLSocket
Dmytro Shteflyuk0b682282026-02-04 16:26:46 -0500323 keys_dir = os.path.join(os.path.dirname(SCRIPT_DIR), 'keys')
324 ca_certs = os.path.join(keys_dir, 'client.pem')
325 certfile = os.path.join(keys_dir, 'server.crt')
326 keyfile = os.path.join(keys_dir, 'server.key')
327 ssl_version = getattr(ssl, 'PROTOCOL_TLS_SERVER', ssl.PROTOCOL_TLSv1)
328 transport = TSSLSocket.TSSLServerSocket(
329 host,
330 options.port,
331 certfile=certfile,
332 keyfile=keyfile,
333 ca_certs=ca_certs,
334 cert_reqs=ssl.CERT_REQUIRED,
335 ssl_version=ssl_version,
336 )
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900337 else:
Kengo Sekif1c53412019-12-13 08:09:36 +0900338 transport = TSocket.TServerSocket(host, options.port, options.domain_socket)
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900339 tfactory = TTransport.TBufferedTransportFactory()
340 if options.trans == 'buffered':
341 tfactory = TTransport.TBufferedTransportFactory()
342 elif options.trans == 'framed':
343 tfactory = TTransport.TFramedTransportFactory()
344 elif options.trans == '':
345 raise AssertionError('Unknown --transport option: %s' % options.trans)
346 else:
347 tfactory = TTransport.TBufferedTransportFactory()
348 # if --zlib, then wrap server transport, and use a different transport factory
349 if options.zlib:
Gregg Donovan62ec9292026-01-29 16:51:37 -0500350 if server_type != "TProcessPoolServer":
351 transport = TZlibTransport.TZlibTransport(transport) # wrap with zlib
352 # Avoid wrapping the server transport for process pools; TZlibTransport isn't picklable on spawn.
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900353 tfactory = TZlibTransport.TZlibTransportFactory()
354
355 # do server-specific setup here:
356 if server_type == "TNonblockingServer":
357 server = TNonblockingServer.TNonblockingServer(processor, transport, inputProtocolFactory=pfactory)
358 elif server_type == "TProcessPoolServer":
359 import signal
360 from thrift.server import TProcessPoolServer
361 server = TProcessPoolServer.TProcessPoolServer(processor, transport, tfactory, pfactory)
362 server.setNumWorkers(5)
363
364 def set_alarm():
365 def clean_shutdown(signum, frame):
366 for worker in server.workers:
367 if options.verbose > 0:
368 logging.info('Terminating worker: %s' % worker)
369 worker.terminate()
370 if options.verbose > 0:
371 logging.info('Requesting server to stop()')
372 try:
373 server.stop()
James E. King, III350fe752017-10-25 09:57:18 -0400374 except Exception:
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900375 pass
376 signal.signal(signal.SIGALRM, clean_shutdown)
377 signal.alarm(4)
378 set_alarm()
379 else:
380 # look up server class dynamically to instantiate server
381 ServerClass = getattr(TServer, server_type)
382 server = ServerClass(processor, transport, tfactory, pfactory)
383 # enter server main loop
Nobuaki Sukegawaa185d7e2015-11-06 21:24:24 +0900384 server.serve()
Nobuaki Sukegawaa185d7e2015-11-06 21:24:24 +0900385
James E. King, III0ad20bd2017-09-30 15:44:16 -0700386
James E. King III6f8c99e2018-03-24 16:32:02 -0400387def exit_gracefully(signum, frame):
388 print("SIGINT received\n")
389 server.shutdown() # doesn't work properly, yet
390 sys.exit(0)
391
392
Nobuaki Sukegawaa185d7e2015-11-06 21:24:24 +0900393if __name__ == '__main__':
James E. King III6f8c99e2018-03-24 16:32:02 -0400394 signal.signal(signal.SIGINT, exit_gracefully)
395
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900396 parser = OptionParser()
397 parser.add_option('--libpydir', type='string', dest='libpydir',
398 help='include this directory to sys.path for locating library code')
399 parser.add_option('--genpydir', type='string', dest='genpydir',
400 default='gen-py',
401 help='include this directory to sys.path for locating generated code')
402 parser.add_option("--port", type="int", dest="port",
403 help="port number for server to listen on")
404 parser.add_option("--zlib", action="store_true", dest="zlib",
405 help="use zlib wrapper for compressed transport")
406 parser.add_option("--ssl", action="store_true", dest="ssl",
407 help="use SSL for encrypted transport")
408 parser.add_option('-v', '--verbose', action="store_const",
409 dest="verbose", const=2,
410 help="verbose output")
411 parser.add_option('-q', '--quiet', action="store_const",
412 dest="verbose", const=0,
413 help="minimal output")
414 parser.add_option('--protocol', dest="proto", type="string",
James E. King III9804ab92019-02-07 16:59:05 -0500415 help="protocol to use, one of: accel, accelc, binary, compact, json, multi, multia, multiac, multic, multih, multij")
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900416 parser.add_option('--transport', dest="trans", type="string",
James E. King III6f8c99e2018-03-24 16:32:02 -0400417 help="transport to use, one of: buffered, framed, http")
Kengo Sekif1c53412019-12-13 08:09:36 +0900418 parser.add_option('--domain-socket', dest="domain_socket", type="string",
419 help="Unix domain socket path")
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900420 parser.add_option('--container-limit', dest='container_limit', type='int', default=None)
421 parser.add_option('--string-limit', dest='string_limit', type='int', default=None)
James E. King III6f8c99e2018-03-24 16:32:02 -0400422 parser.set_defaults(port=9090, verbose=1, proto='binary', transport='buffered')
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900423 options, args = parser.parse_args()
Nobuaki Sukegawaa185d7e2015-11-06 21:24:24 +0900424
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900425 # Print TServer log to stdout so that the test-runner can redirect it to log files
426 logging.basicConfig(level=options.verbose)
Nobuaki Sukegawa7b545b52016-01-11 13:46:04 +0900427
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900428 sys.path.insert(0, os.path.join(SCRIPT_DIR, options.genpydir))
Nobuaki Sukegawaa185d7e2015-11-06 21:24:24 +0900429
James E. King III9804ab92019-02-07 16:59:05 -0500430 from ThriftTest import ThriftTest, SecondService
James E. King III9804ab92019-02-07 16:59:05 -0500431 from thrift.TMultiplexedProcessor import TMultiplexedProcessor
Neil Williams66a44c52018-08-13 16:12:24 -0700432 from thrift.transport import THeaderTransport
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900433 from thrift.transport import TTransport
434 from thrift.transport import TSocket
435 from thrift.transport import TZlibTransport
436 from thrift.protocol import TBinaryProtocol
437 from thrift.protocol import TCompactProtocol
Neil Williams66a44c52018-08-13 16:12:24 -0700438 from thrift.protocol import THeaderProtocol
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900439 from thrift.protocol import TJSONProtocol
440 from thrift.server import TServer, TNonblockingServer, THttpServer
Nobuaki Sukegawaa185d7e2015-11-06 21:24:24 +0900441
Nobuaki Sukegawa10308cb2016-02-03 01:57:03 +0900442 sys.exit(main(options))