Skip to content

Commit

Permalink
Updated cli parser to handle @file sytanx and added --data-binary option
Browse files Browse the repository at this point in the history
Arguments which start with @ will be loaded from file for the --data
and --data-binary cli options.
  • Loading branch information
tomdottom committed May 25, 2018
1 parent 1e5d194 commit 0ed92aa
Show file tree
Hide file tree
Showing 2 changed files with 89 additions and 6 deletions.
56 changes: 56 additions & 0 deletions tests/test_curl.py
Original file line number Diff line number Diff line change
@@ -1,11 +1,23 @@
from copy import deepcopy
import json
import socket
import tempfile
import unittest

from uwsgi_tools.curl import cli
from tests.utils import server


class CurlTests(unittest.TestCase):
def setUp(self):
self.datafile = tempfile.NamedTemporaryFile(delete=True)
with open(self.datafile.name, 'w') as fh:
json.dump({'foo': 'bar'}, fh)

self.binary_datafile = tempfile.NamedTemporaryFile(delete=True)
with open(self.binary_datafile.name, 'wb') as fh:
fh.write(b'binary-fooooo')

def test_cli(self):
with server():
self.assertFalse(cli('127.0.0.1', 'host.name/'))
Expand Down Expand Up @@ -43,3 +55,47 @@ def request_sniper(x):
self.assertIn(b'CONTENT_LENGTH\x02\x0017', requests[0])
self.assertIn(b'CONTENT_TYPE', requests[0])
self.assertIn(b'application/json', requests[0])

def test_at_prefixed_data(self):
requests = []

def request_sniper(x):
requests.append(deepcopy(x))

with server(callback=request_sniper):
self.assertFalse(cli(
'--method', 'POST',
'--header', 'Content-Type: application/json',
'--data', '@{}'.format(self.datafile.name),
'127.0.0.1',
'host.name/',
))

self.assertIn(b'POST', requests[0])
# Magic number is the val_size + val
self.assertIn(b'CONTENT_LENGTH\x02\x0014', requests[0])
self.assertIn(b'CONTENT_TYPE', requests[0])
self.assertIn(b'application/json', requests[0])
self.assertIn(b'{"foo": "bar"}', requests[0])

def test_binary_data(self):
requests = []

def request_sniper(x):
requests.append(deepcopy(x))

with server(callback=request_sniper):
self.assertFalse(cli(
'--method', 'POST',
'--header', 'Content-Type: application/json',
'--data-binary', '@{}'.format(self.binary_datafile.name),
'127.0.0.1',
'host.name/',
))

self.assertIn(b'POST', requests[0])
# Magic number is the val_size + val
self.assertIn(b'CONTENT_LENGTH\x02\x0013', requests[0])
self.assertIn(b'CONTENT_TYPE', requests[0])
self.assertIn(b'application/json', requests[0])
self.assertIn(b'binary-fooooo', requests[0])
39 changes: 33 additions & 6 deletions uwsgi_tools/curl.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import os
import socket
import sys
from .compat import urlsplit
Expand Down Expand Up @@ -26,10 +27,19 @@ def ask_uwsgi(uwsgi_addr, var, body='', timeout=0, udp=False):
break
response.append(data)
s.close()
return b''.join(response).decode('utf8')

response_lines = [i for r in response for i in r.splitlines()]

def curl(uwsgi_addr, url, method='GET', body='', timeout=0, headers=(),
def try_decode(b):
try:
return b.decode('utf-8')
except UnicodeDecodeError:
return str(b)

return os.linesep.join(try_decode(r) for r in response_lines)


def curl(uwsgi_addr, url, method='GET', body='', body_binary=b'', timeout=0, headers=(),
udp=False):
host, uri = get_host_from_url(url)
parts_uri = urlsplit(uri)
Expand All @@ -42,7 +52,10 @@ def curl(uwsgi_addr, url, method='GET', body='', timeout=0, headers=(),
else:
port = None

body = (body or '').encode('utf-8')
if body_binary:
body = body_binary
else:
body = (body or '').encode('utf-8')

var = {
'SERVER_PROTOCOL': 'HTTP/1.1',
Expand Down Expand Up @@ -78,6 +91,18 @@ def curl(uwsgi_addr, url, method='GET', body='', timeout=0, headers=(),
def cli(*args):
import argparse

class LoadFile(argparse.Action):
def __call__(self, parser, namespace, values, option_string=None):
if values.startswith('@'):
if option_string == '--data' or option_string == '-d':
with open(values[1:]) as fh:
namespace.data = fh.read()
elif option_string == '--data-binary':
with open(values[1:], 'rb') as fh:
namespace.data_binary = fh.read()
else:
namespace.data = values

parser = argparse.ArgumentParser()

parser.add_argument('uwsgi_addr', nargs=1,
Expand All @@ -92,7 +117,9 @@ def cli(*args):
parser.add_argument('-H', '--header', action='append', dest='headers',
help="Request header. It can be used multiple times")

parser.add_argument('-d', '--data', help="Request body")
parser.add_argument('-d', '--data', action=LoadFile, help="Request body")

parser.add_argument('--data-binary', action=LoadFile, help="Request body")

parser.add_argument('-t', '--timeout', default=0.0, type=float,
help="Socket timeout")
Expand All @@ -103,8 +130,8 @@ def cli(*args):
args = parser.parse_args(args or sys.argv[1:])

response = curl(uwsgi_addr=args.uwsgi_addr[0], method=args.method,
url=args.url, body=args.data, timeout=args.timeout,
headers=args.headers, udp=args.udp)
url=args.url, body=args.data, body_binary=args.data_binary,
timeout=args.timeout, headers=args.headers, udp=args.udp)
print(response)

status = int(response.split(' ', 2)[1])
Expand Down

0 comments on commit 0ed92aa

Please sign in to comment.