2018-08-15 19:08:03 -04:00
|
|
|
#!/usr/bin/env python
|
2020-01-02 15:13:47 -05:00
|
|
|
# Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2018-08-15 19:08:03 -04:00
|
|
|
# Many tests expect there to be an http server on port 4545 servering the deno
|
|
|
|
# root directory.
|
2019-05-30 13:40:40 -07:00
|
|
|
from collections import namedtuple
|
|
|
|
from contextlib import contextmanager
|
2018-08-15 19:08:03 -04:00
|
|
|
import os
|
|
|
|
import SimpleHTTPServer
|
|
|
|
import SocketServer
|
2019-11-19 18:56:37 +01:00
|
|
|
import socket
|
2019-05-30 13:40:40 -07:00
|
|
|
import sys
|
2018-08-15 19:08:03 -04:00
|
|
|
from time import sleep
|
2019-05-30 13:40:40 -07:00
|
|
|
from threading import Thread
|
|
|
|
from util import root_path
|
2020-02-17 11:59:51 -05:00
|
|
|
import ssl
|
|
|
|
import getopt
|
|
|
|
import argparse
|
2018-08-15 19:08:03 -04:00
|
|
|
|
|
|
|
PORT = 4545
|
2018-10-09 17:31:06 -07:00
|
|
|
REDIRECT_PORT = 4546
|
2019-04-01 18:46:40 -07:00
|
|
|
ANOTHER_REDIRECT_PORT = 4547
|
|
|
|
DOUBLE_REDIRECTS_PORT = 4548
|
2019-06-24 06:34:09 -07:00
|
|
|
INF_REDIRECTS_PORT = 4549
|
2020-02-28 06:04:18 -08:00
|
|
|
REDIRECT_ABSOLUTE_PORT = 4550
|
2020-02-17 11:59:51 -05:00
|
|
|
HTTPS_PORT = 5545
|
|
|
|
|
|
|
|
|
|
|
|
def create_http_arg_parser():
|
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
parser.add_argument('--verbose', '-v', action='store_true')
|
|
|
|
return parser
|
|
|
|
|
|
|
|
|
|
|
|
HttpArgParser = create_http_arg_parser()
|
|
|
|
|
|
|
|
args, unknown = HttpArgParser.parse_known_args(sys.argv[1:])
|
2020-02-26 05:50:25 -05:00
|
|
|
CERT_FILE = os.path.join(root_path, "std/http/testdata/tls/localhost.crt")
|
|
|
|
KEY_FILE = os.path.join(root_path, "std/http/testdata/tls/localhost.key")
|
2020-02-17 11:59:51 -05:00
|
|
|
QUIET = not args.verbose
|
|
|
|
|
|
|
|
|
|
|
|
class SSLTCPServer(SocketServer.TCPServer):
|
|
|
|
def __init__(self,
|
|
|
|
server_address,
|
|
|
|
request_handler,
|
|
|
|
certfile,
|
|
|
|
keyfile,
|
|
|
|
ssl_version=ssl.PROTOCOL_TLSv1_2,
|
|
|
|
bind_and_activate=True):
|
|
|
|
SocketServer.TCPServer.__init__(self, server_address, request_handler,
|
|
|
|
bind_and_activate)
|
|
|
|
self.certfile = certfile
|
|
|
|
self.keyfile = keyfile
|
|
|
|
self.ssl_version = ssl_version
|
|
|
|
|
|
|
|
def get_request(self):
|
|
|
|
newsocket, fromaddr = self.socket.accept()
|
|
|
|
connstream = ssl.wrap_socket(
|
|
|
|
newsocket,
|
|
|
|
server_side=True,
|
|
|
|
certfile=self.certfile,
|
|
|
|
keyfile=self.keyfile,
|
|
|
|
ssl_version=self.ssl_version)
|
|
|
|
return connstream, fromaddr
|
|
|
|
|
|
|
|
|
|
|
|
class SSLThreadingTCPServer(SocketServer.ThreadingMixIn, SSLTCPServer):
|
|
|
|
pass
|
2018-08-15 19:08:03 -04:00
|
|
|
|
2019-06-08 04:46:57 -07:00
|
|
|
|
|
|
|
class QuietSimpleHTTPRequestHandler(SimpleHTTPServer.SimpleHTTPRequestHandler):
|
|
|
|
def log_request(self, code='-', size='-'):
|
|
|
|
if not QUIET:
|
|
|
|
SimpleHTTPServer.SimpleHTTPRequestHandler.log_request(
|
|
|
|
self, code, size)
|
|
|
|
|
|
|
|
|
|
|
|
class ContentTypeHandler(QuietSimpleHTTPRequestHandler):
|
2018-12-21 17:09:53 -05:00
|
|
|
def do_GET(self):
|
2020-01-11 04:11:05 -06:00
|
|
|
|
|
|
|
# Check if there is a custom header configuration ending
|
|
|
|
# with ".header" before sending the file
|
|
|
|
maybe_header_file_path = "./" + self.path + ".header"
|
|
|
|
if os.path.exists(maybe_header_file_path):
|
|
|
|
self.protocol_version = 'HTTP/1.1'
|
|
|
|
self.send_response(200, 'OK')
|
|
|
|
|
|
|
|
f = open(maybe_header_file_path)
|
|
|
|
for line in f:
|
|
|
|
kv = line.split(": ")
|
|
|
|
self.send_header(kv[0].strip(), kv[1].strip())
|
|
|
|
f.close()
|
|
|
|
self.end_headers()
|
|
|
|
|
|
|
|
body = open("./" + self.path)
|
|
|
|
self.wfile.write(body.read())
|
|
|
|
body.close()
|
|
|
|
return
|
|
|
|
|
|
|
|
if "etag_script.ts" in self.path:
|
|
|
|
self.protocol_version = 'HTTP/1.1'
|
|
|
|
if_not_match = self.headers.getheader('if-none-match')
|
|
|
|
if if_not_match == "33a64df551425fcc55e":
|
|
|
|
self.send_response(304, 'Not Modified')
|
2020-01-27 05:59:41 +11:00
|
|
|
self.send_header('Content-type', 'application/typescript')
|
2020-01-11 04:11:05 -06:00
|
|
|
self.send_header('ETag', '33a64df551425fcc55e')
|
|
|
|
self.end_headers()
|
|
|
|
else:
|
|
|
|
self.send_response(200, 'OK')
|
2020-01-27 05:59:41 +11:00
|
|
|
self.send_header('Content-type', 'application/typescript')
|
2020-01-11 04:11:05 -06:00
|
|
|
self.send_header('ETag', '33a64df551425fcc55e')
|
|
|
|
self.end_headers()
|
|
|
|
self.wfile.write(bytes("console.log('etag')"))
|
|
|
|
return
|
|
|
|
|
2020-01-27 05:59:41 +11:00
|
|
|
if "xTypeScriptTypes.js" in self.path:
|
|
|
|
self.protocol_version = "HTTP/1.1"
|
|
|
|
self.send_response(200, 'OK')
|
|
|
|
self.send_header('Content-type', 'application/javascript')
|
|
|
|
self.send_header('X-TypeScript-Types', './xTypeScriptTypes.d.ts')
|
|
|
|
self.end_headers()
|
|
|
|
self.wfile.write(bytes("export const foo = 'foo';"))
|
|
|
|
return
|
|
|
|
|
2020-05-22 16:01:00 +02:00
|
|
|
if "type_directives_redirect.js" in self.path:
|
|
|
|
self.protocol_version = "HTTP/1.1"
|
|
|
|
self.send_response(200, 'OK')
|
|
|
|
self.send_header('Content-type', 'application/javascript')
|
|
|
|
self.send_header(
|
|
|
|
'X-TypeScript-Types',
|
|
|
|
'http://localhost:4547/xTypeScriptTypesRedirect.d.ts')
|
|
|
|
self.end_headers()
|
|
|
|
self.wfile.write(bytes("export const foo = 'foo';"))
|
|
|
|
return
|
|
|
|
|
|
|
|
if "xTypeScriptTypesRedirect.d.ts" in self.path:
|
|
|
|
self.protocol_version = "HTTP/1.1"
|
|
|
|
self.send_response(200, 'OK')
|
|
|
|
self.send_header('Content-type', 'application/typescript')
|
|
|
|
self.end_headers()
|
|
|
|
self.wfile.write(
|
|
|
|
bytes("import './xTypeScriptTypesRedirected.d.ts';"))
|
|
|
|
return
|
|
|
|
|
|
|
|
if "xTypeScriptTypesRedirected.d.ts" in self.path:
|
|
|
|
self.protocol_version = "HTTP/1.1"
|
|
|
|
self.send_response(200, 'OK')
|
|
|
|
self.send_header('Content-type', 'application/typescript')
|
|
|
|
self.end_headers()
|
|
|
|
self.wfile.write(bytes("export const foo: 'foo';"))
|
|
|
|
return
|
|
|
|
|
2020-01-27 05:59:41 +11:00
|
|
|
if "xTypeScriptTypes.d.ts" in self.path:
|
|
|
|
self.protocol_version = "HTTP/1.1"
|
|
|
|
self.send_response(200, 'OK')
|
|
|
|
self.send_header('Content-type', 'application/typescript')
|
|
|
|
self.end_headers()
|
|
|
|
self.wfile.write(bytes("export const foo: 'foo';"))
|
|
|
|
return
|
|
|
|
|
|
|
|
if "referenceTypes.js" in self.path:
|
|
|
|
self.protocol_version = "HTTP/1.1"
|
|
|
|
self.send_response(200, 'OK')
|
|
|
|
self.send_header('Content-type', 'application/javascript')
|
|
|
|
self.end_headers()
|
|
|
|
self.wfile.write(
|
|
|
|
bytes('/// <reference types="./xTypeScriptTypes.d.ts" />\r\n'
|
|
|
|
'export const foo = "foo";\r\n'))
|
|
|
|
return
|
|
|
|
|
2018-12-21 17:09:53 -05:00
|
|
|
if "multipart_form_data.txt" in self.path:
|
|
|
|
self.protocol_version = 'HTTP/1.1'
|
|
|
|
self.send_response(200, 'OK')
|
|
|
|
self.send_header('Content-type',
|
|
|
|
'multipart/form-data;boundary=boundary')
|
|
|
|
self.end_headers()
|
|
|
|
self.wfile.write(
|
|
|
|
bytes('Preamble\r\n'
|
|
|
|
'--boundary\t \r\n'
|
|
|
|
'Content-Disposition: form-data; name="field_1"\r\n'
|
|
|
|
'\r\n'
|
|
|
|
'value_1 \r\n'
|
|
|
|
'\r\n--boundary\r\n'
|
|
|
|
'Content-Disposition: form-data; name="field_2"; '
|
|
|
|
'filename="file.js"\r\n'
|
|
|
|
'Content-Type: text/javascript\r\n'
|
|
|
|
'\r\n'
|
|
|
|
'console.log("Hi")'
|
|
|
|
'\r\n--boundary--\r\n'
|
|
|
|
'Epilogue'))
|
|
|
|
return
|
|
|
|
return SimpleHTTPServer.SimpleHTTPRequestHandler.do_GET(self)
|
|
|
|
|
2019-01-03 06:41:20 -05:00
|
|
|
def do_POST(self):
|
|
|
|
# Simple echo server for request reflection
|
|
|
|
if "echo_server" in self.path:
|
2020-06-01 01:21:14 +02:00
|
|
|
status = int(self.headers.getheader('x-status', "200"))
|
2019-01-03 06:41:20 -05:00
|
|
|
self.protocol_version = 'HTTP/1.1'
|
2020-06-01 01:21:14 +02:00
|
|
|
self.send_response(status, 'OK')
|
2019-01-03 06:41:20 -05:00
|
|
|
if self.headers.has_key('content-type'):
|
|
|
|
self.send_header('content-type',
|
|
|
|
self.headers.getheader('content-type'))
|
2019-09-11 13:31:00 +02:00
|
|
|
if self.headers.has_key('user-agent'):
|
|
|
|
self.send_header('user-agent',
|
|
|
|
self.headers.getheader('user-agent'))
|
2019-01-03 06:41:20 -05:00
|
|
|
self.end_headers()
|
|
|
|
data_string = self.rfile.read(int(self.headers['Content-Length']))
|
|
|
|
self.wfile.write(bytes(data_string))
|
|
|
|
return
|
2020-06-01 14:32:08 +02:00
|
|
|
if "echo_multipart_file" in self.path:
|
|
|
|
self.protocol_version = 'HTTP/1.1'
|
|
|
|
self.send_response(200, 'OK')
|
|
|
|
self.send_header('Content-type',
|
|
|
|
'multipart/form-data;boundary=boundary')
|
|
|
|
self.end_headers()
|
|
|
|
file_content = self.rfile.read(int(self.headers['Content-Length']))
|
|
|
|
self.wfile.write(
|
|
|
|
bytes('--boundary\t \r\n'
|
|
|
|
'Content-Disposition: form-data; name="field_1"\r\n'
|
|
|
|
'\r\n'
|
|
|
|
'value_1 \r\n'
|
|
|
|
'\r\n--boundary\r\n'
|
|
|
|
'Content-Disposition: form-data; name="file"; '
|
|
|
|
'filename="file.bin"\r\n'
|
|
|
|
'Content-Type: application/octet-stream\r\n'
|
|
|
|
'\r\n') + bytes(file_content) +
|
|
|
|
bytes('\r\n--boundary--\r\n'))
|
|
|
|
return
|
2019-01-03 06:41:20 -05:00
|
|
|
self.protocol_version = 'HTTP/1.1'
|
|
|
|
self.send_response(501)
|
|
|
|
self.send_header('content-type', 'text/plain')
|
|
|
|
self.end_headers()
|
|
|
|
self.wfile.write(bytes('Server does not support this operation'))
|
|
|
|
|
2018-10-22 13:14:27 +11:00
|
|
|
def guess_type(self, path):
|
|
|
|
if ".t1." in path:
|
|
|
|
return "text/typescript"
|
|
|
|
if ".t2." in path:
|
|
|
|
return "video/vnd.dlna.mpeg-tts"
|
|
|
|
if ".t3." in path:
|
|
|
|
return "video/mp2t"
|
2018-10-27 17:26:42 -07:00
|
|
|
if ".t4." in path:
|
|
|
|
return "application/x-typescript"
|
2018-10-22 13:14:27 +11:00
|
|
|
if ".j1." in path:
|
|
|
|
return "text/javascript"
|
|
|
|
if ".j2." in path:
|
|
|
|
return "application/ecmascript"
|
|
|
|
if ".j3." in path:
|
|
|
|
return "text/ecmascript"
|
|
|
|
if ".j4." in path:
|
|
|
|
return "application/x-javascript"
|
2018-12-21 17:09:53 -05:00
|
|
|
if "form_urlencoded" in path:
|
|
|
|
return "application/x-www-form-urlencoded"
|
2018-12-27 15:40:06 -05:00
|
|
|
if "no_ext" in path:
|
|
|
|
return "text/typescript"
|
|
|
|
if "unknown_ext" in path:
|
|
|
|
return "text/typescript"
|
|
|
|
if "mismatch_ext" in path:
|
|
|
|
return "text/javascript"
|
2018-10-22 13:14:27 +11:00
|
|
|
return SimpleHTTPServer.SimpleHTTPRequestHandler.guess_type(self, path)
|
|
|
|
|
|
|
|
|
2019-05-30 13:40:40 -07:00
|
|
|
RunningServer = namedtuple("RunningServer", ["server", "thread"])
|
|
|
|
|
|
|
|
|
2020-02-17 11:59:51 -05:00
|
|
|
def get_socket(port, handler, use_https):
|
2019-11-19 18:56:37 +01:00
|
|
|
SocketServer.TCPServer.allow_reuse_address = True
|
|
|
|
if os.name != "nt":
|
|
|
|
# We use AF_INET6 to avoid flaky test issue, particularly with
|
|
|
|
# the test 019_media_types. It's not well understood why this fixes the
|
|
|
|
# flaky tests, but it does appear to...
|
|
|
|
# See https://github.com/denoland/deno/issues/3332
|
|
|
|
SocketServer.TCPServer.address_family = socket.AF_INET6
|
2020-02-17 11:59:51 -05:00
|
|
|
|
|
|
|
if use_https:
|
|
|
|
return SSLThreadingTCPServer(("", port), handler, CERT_FILE, KEY_FILE)
|
2019-11-19 18:56:37 +01:00
|
|
|
return SocketServer.TCPServer(("", port), handler)
|
|
|
|
|
|
|
|
|
2018-09-25 18:39:04 -07:00
|
|
|
def server():
|
2018-08-15 19:08:03 -04:00
|
|
|
os.chdir(root_path) # Hopefully the main thread doesn't also chdir.
|
2018-10-22 13:14:27 +11:00
|
|
|
Handler = ContentTypeHandler
|
|
|
|
Handler.extensions_map.update({
|
|
|
|
".ts": "application/typescript",
|
|
|
|
".js": "application/javascript",
|
2019-10-17 02:35:04 +09:00
|
|
|
".tsx": "application/typescript",
|
|
|
|
".jsx": "application/javascript",
|
2018-10-22 13:14:27 +11:00
|
|
|
".json": "application/json",
|
|
|
|
})
|
2020-02-17 11:59:51 -05:00
|
|
|
s = get_socket(PORT, Handler, False)
|
2019-06-08 04:46:57 -07:00
|
|
|
if not QUIET:
|
|
|
|
print "Deno test server http://localhost:%d/" % PORT
|
2019-05-30 13:40:40 -07:00
|
|
|
return RunningServer(s, start(s))
|
2018-08-15 19:08:03 -04:00
|
|
|
|
|
|
|
|
2019-04-01 18:46:40 -07:00
|
|
|
def base_redirect_server(host_port, target_port, extra_path_segment=""):
|
2018-10-09 17:31:06 -07:00
|
|
|
os.chdir(root_path)
|
2019-04-01 18:46:40 -07:00
|
|
|
target_host = "http://localhost:%d" % target_port
|
2018-10-09 17:31:06 -07:00
|
|
|
|
2019-06-08 04:46:57 -07:00
|
|
|
class RedirectHandler(QuietSimpleHTTPRequestHandler):
|
2018-10-09 17:31:06 -07:00
|
|
|
def do_GET(self):
|
|
|
|
self.send_response(301)
|
2019-04-01 18:46:40 -07:00
|
|
|
self.send_header('Location',
|
|
|
|
target_host + extra_path_segment + self.path)
|
2018-10-09 17:31:06 -07:00
|
|
|
self.end_headers()
|
|
|
|
|
2020-02-17 11:59:51 -05:00
|
|
|
s = get_socket(host_port, RedirectHandler, False)
|
2019-06-08 04:46:57 -07:00
|
|
|
if not QUIET:
|
|
|
|
print "redirect server http://localhost:%d/ -> http://localhost:%d/" % (
|
|
|
|
host_port, target_port)
|
2019-05-30 13:40:40 -07:00
|
|
|
return RunningServer(s, start(s))
|
2018-10-09 17:31:06 -07:00
|
|
|
|
|
|
|
|
2019-04-01 18:46:40 -07:00
|
|
|
# redirect server
|
|
|
|
def redirect_server():
|
|
|
|
return base_redirect_server(REDIRECT_PORT, PORT)
|
|
|
|
|
|
|
|
|
|
|
|
# another redirect server pointing to the same port as the one above
|
|
|
|
# BUT with an extra subdir path
|
|
|
|
def another_redirect_server():
|
|
|
|
return base_redirect_server(
|
2020-02-02 22:55:22 +01:00
|
|
|
ANOTHER_REDIRECT_PORT, PORT, extra_path_segment="/cli/tests/subdir")
|
2019-04-01 18:46:40 -07:00
|
|
|
|
|
|
|
|
|
|
|
# redirect server that points to another redirect server
|
|
|
|
def double_redirects_server():
|
|
|
|
return base_redirect_server(DOUBLE_REDIRECTS_PORT, REDIRECT_PORT)
|
|
|
|
|
|
|
|
|
2019-06-24 06:34:09 -07:00
|
|
|
# redirect server that points to itself
|
|
|
|
def inf_redirects_server():
|
|
|
|
return base_redirect_server(INF_REDIRECTS_PORT, INF_REDIRECTS_PORT)
|
|
|
|
|
|
|
|
|
2020-02-28 06:04:18 -08:00
|
|
|
# redirect server that redirect to absolute paths under same host
|
|
|
|
# redirects /REDIRECT/file_name to /file_name
|
|
|
|
def absolute_redirect_server():
|
|
|
|
os.chdir(root_path)
|
|
|
|
|
|
|
|
class AbsoluteRedirectHandler(ContentTypeHandler):
|
|
|
|
def do_GET(self):
|
|
|
|
print(self.path)
|
|
|
|
if (self.path.startswith("/REDIRECT/")):
|
|
|
|
self.send_response(302)
|
|
|
|
self.send_header('Location',
|
|
|
|
self.path.split('/REDIRECT', 1)[1])
|
|
|
|
self.end_headers()
|
|
|
|
else:
|
|
|
|
ContentTypeHandler.do_GET(self)
|
|
|
|
|
|
|
|
s = get_socket(REDIRECT_ABSOLUTE_PORT, AbsoluteRedirectHandler, False)
|
|
|
|
if not QUIET:
|
|
|
|
print("absolute redirect server http://localhost:%d/" %
|
|
|
|
REDIRECT_ABSOLUTE_PORT)
|
|
|
|
return RunningServer(s, start(s))
|
|
|
|
|
|
|
|
|
2020-02-17 11:59:51 -05:00
|
|
|
def https_server():
|
|
|
|
os.chdir(root_path) # Hopefully the main thread doesn't also chdir.
|
|
|
|
Handler = ContentTypeHandler
|
|
|
|
Handler.extensions_map.update({
|
|
|
|
".ts": "application/typescript",
|
|
|
|
".js": "application/javascript",
|
|
|
|
".tsx": "application/typescript",
|
|
|
|
".jsx": "application/javascript",
|
|
|
|
".json": "application/json",
|
|
|
|
})
|
|
|
|
s = get_socket(HTTPS_PORT, Handler, True)
|
|
|
|
if not QUIET:
|
|
|
|
print "Deno https test server https://localhost:%d/" % HTTPS_PORT
|
|
|
|
return RunningServer(s, start(s))
|
|
|
|
|
|
|
|
|
2019-05-30 13:40:40 -07:00
|
|
|
def start(s):
|
|
|
|
thread = Thread(target=s.serve_forever, kwargs={"poll_interval": 0.05})
|
2018-08-15 19:08:03 -04:00
|
|
|
thread.daemon = True
|
|
|
|
thread.start()
|
2018-10-19 19:15:29 -07:00
|
|
|
return thread
|
2018-08-15 19:08:03 -04:00
|
|
|
|
|
|
|
|
2019-05-30 13:40:40 -07:00
|
|
|
@contextmanager
|
|
|
|
def spawn():
|
|
|
|
servers = (server(), redirect_server(), another_redirect_server(),
|
2020-02-28 06:04:18 -08:00
|
|
|
double_redirects_server(), https_server(),
|
2020-05-31 22:13:53 +02:00
|
|
|
absolute_redirect_server(), inf_redirects_server())
|
2020-02-26 05:50:25 -05:00
|
|
|
# In order to wait for each of the servers to be ready, we try connecting to
|
|
|
|
# them with a tcp socket.
|
|
|
|
for running_server in servers:
|
|
|
|
client = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
|
|
|
port = running_server.server.server_address[1]
|
|
|
|
client.connect(("127.0.0.1", port))
|
|
|
|
print "connected", port
|
|
|
|
client.close()
|
|
|
|
assert running_server.thread.is_alive()
|
|
|
|
# The following output "ready" is specificly looked for in cli/test_util.rs
|
|
|
|
# to prevent race conditions.
|
|
|
|
print "ready"
|
2019-05-30 13:40:40 -07:00
|
|
|
try:
|
2019-10-29 13:06:14 -07:00
|
|
|
yield servers
|
2019-05-30 13:40:40 -07:00
|
|
|
finally:
|
|
|
|
for s in servers:
|
2020-03-30 11:04:45 -07:00
|
|
|
# Make sure all servers still running,
|
|
|
|
# if not assume there was an error
|
|
|
|
assert s.thread.is_alive()
|
2019-05-30 13:40:40 -07:00
|
|
|
s.server.shutdown()
|
|
|
|
|
|
|
|
|
2018-11-30 03:27:41 -05:00
|
|
|
def main():
|
2019-10-29 13:06:14 -07:00
|
|
|
with spawn() as servers:
|
|
|
|
try:
|
|
|
|
while all(s.thread.is_alive() for s in servers):
|
|
|
|
sleep(1)
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
pass
|
2018-10-19 19:15:29 -07:00
|
|
|
sys.exit(1)
|
2018-11-30 03:27:41 -05:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|