Skip to content

Commit

Permalink
Merge pull request #557 from HttpRunner/upload_files
Browse files Browse the repository at this point in the history
2.1.1 Upload files
  • Loading branch information
debugtalk committed Apr 16, 2019
2 parents 4b44b31 + 52c3a3d commit 9fd69d8
Show file tree
Hide file tree
Showing 9 changed files with 86 additions and 29 deletions.
9 changes: 9 additions & 0 deletions HISTORY.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,14 @@
# Release History

## 2.1.1 (2019-04-11)

**Features**

refactor upload files mechanism with [requests-toolbelt](https://toolbelt.readthedocs.io/en/latest/user.html#multipart-form-data-encoder):

- simplify usage syntax, detect mimetype with [filetype](https://github.com/h2non/filetype.py).
- support upload multiple fields.

## 2.1.0 (2019-04-10)

**Features**
Expand Down
1 change: 1 addition & 0 deletions Pipfile
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ har2case = "*"
colorama = "*"
colorlog = "*"
requests-toolbelt = "*"
filetype = "*"

[dev-packages]
Flask = "<1.0.0"
Expand Down
2 changes: 1 addition & 1 deletion httprunner/__about__.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
__title__ = 'HttpRunner'
__description__ = 'One-stop solution for HTTP(S) testing.'
__url__ = 'https://github.com/HttpRunner/HttpRunner'
__version__ = '2.1.0'
__version__ = '2.1.1'
__author__ = 'debugtalk'
__author_email__ = 'mail@debugtalk.com'
__license__ = 'Apache-2.0'
Expand Down
75 changes: 62 additions & 13 deletions httprunner/built_in.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,13 +12,18 @@
import string
import time

import filetype
from httprunner.compat import basestring, builtin_str, integer_types, str
from httprunner.exceptions import ParamsError
from requests_toolbelt import MultipartEncoder

PWD = os.getcwd()


###############################################################################
## built-in functions
###############################################################################

""" built-in functions
"""
def gen_random_string(str_len):
""" generate random string with specified length
"""
Expand All @@ -38,24 +43,68 @@ def get_current_date(fmt="%Y-%m-%d"):
"""
return datetime.datetime.now().strftime(fmt)

def multipart_encoder(field_name, file_path, file_type=None, file_headers=None):
if not os.path.isabs(file_path):
file_path = os.path.join(os.getcwd(), file_path)

filename = os.path.basename(file_path)
with open(file_path, 'rb') as f:
fields = {
field_name: (filename, f.read(), file_type)
}
###############################################################################
## upload files with requests-toolbelt
# e.g.
# - test:
# name: upload file
# variables:
# file_path: "data/test.env"
# multipart_encoder: ${multipart_encoder(file=$file_path)}
# request:
# url: /post
# method: POST
# headers:
# Content-Type: ${multipart_content_type($multipart_encoder)}
# data: $multipart_encoder
# validate:
# - eq: ["status_code", 200]
# - startswith: ["content.files.file", "UserName=test"]
###############################################################################

def multipart_encoder(**kwargs):
""" initialize MultipartEncoder with uploading fields.
"""
def get_filetype(file_path):
file_type = filetype.guess(file_path)
if file_type:
return file_type.mime
else:
return "text/html"

fields_dict = {}
for key, value in kwargs.items():

if os.path.isabs(value):
_file_path = value
is_file = True
else:
global PWD
_file_path = os.path.join(PWD, value)
is_file = os.path.isfile(_file_path)

if is_file:
filename = os.path.basename(_file_path)
with open(_file_path, 'rb') as f:
mime_type = get_filetype(_file_path)
fields_dict[key] = (filename, f.read(), mime_type)
else:
fields_dict[key] = value

return MultipartEncoder(fields=fields_dict)

return MultipartEncoder(fields)

def multipart_content_type(multipart_encoder):
""" prepare Content-Type for request headers
"""
return multipart_encoder.content_type


""" built-in comparators
"""
###############################################################################
## built-in comparators
###############################################################################

def equals(check_value, expect_value):
assert check_value == expect_value

Expand Down
5 changes: 2 additions & 3 deletions httprunner/loader.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,8 +8,7 @@
import sys

import yaml
from httprunner import exceptions, logger, parser, utils, validator

from httprunner import built_in, exceptions, logger, parser, utils, validator

###############################################################################
## file loader
Expand Down Expand Up @@ -263,7 +262,6 @@ def load_module_functions(module):
def load_builtin_functions():
""" load built_in module functions
"""
from httprunner import built_in
return load_module_functions(built_in)


Expand Down Expand Up @@ -703,6 +701,7 @@ def load_project_tests(test_path, dot_env_path=None):
# locate PWD and load debugtalk.py functions

project_mapping["PWD"] = project_working_directory
built_in.PWD = project_working_directory
project_mapping["functions"] = debugtalk_functions

# load api
Expand Down
2 changes: 1 addition & 1 deletion httprunner/runner.py
Original file line number Diff line number Diff line change
Expand Up @@ -267,7 +267,7 @@ def _run_test(self, test_dict):
self.session_context.update_session_variables(extracted_variables_mapping)

# validate
validators = test_dict.get("validate", [])
validators = test_dict.get("validate") or test_dict.get("validators") or []
try:
self.session_context.validate(validators, resp_obj)
except (exceptions.ParamsError, exceptions.ValidationFailure, exceptions.ExtractFailure):
Expand Down
3 changes: 2 additions & 1 deletion setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,8 @@
"har2case",
"colorama",
"colorlog",
"requests_toolbelt"
"requests_toolbelt",
"filetype"
]

class UploadCommand(Command):
Expand Down
8 changes: 4 additions & 4 deletions tests/httpbin/load_image.yml
Original file line number Diff line number Diff line change
Expand Up @@ -7,30 +7,30 @@
request:
url: /image/png
method: GET
validators:
validate:
- eq: ["status_code", 200]

- test:
name: get jpeg image
request:
url: /image/jpeg
method: GET
validators:
validate:
- eq: ["status_code", 200]

- test:
name: get webp image
request:
url: /image/webp
method: GET
validators:
validate:
- eq: ["status_code", 200]

- test:
name: get svg image
request:
url: /image/svg
method: GET
validators:
validate:
- eq: ["status_code", 200]

10 changes: 4 additions & 6 deletions tests/httpbin/upload.yml
Original file line number Diff line number Diff line change
Expand Up @@ -5,17 +5,15 @@
- test:
name: upload file
variables:
field_name: "file"
file_path: "LICENSE"
file_type: "text/html"
multipart_encoder: ${multipart_encoder($field_name, $file_path, $file_type)}
file_path: "data/test.env"
multipart_encoder: ${multipart_encoder(file=$file_path)}
request:
url: /post
method: POST
headers:
Content-Type: ${multipart_content_type($multipart_encoder)}
data: $multipart_encoder
validators:
validate:
- eq: ["status_code", 200]
- startswith: ["content.files.file", "MIT License"]
- startswith: ["content.files.file", "UserName=test"]

0 comments on commit 9fd69d8

Please sign in to comment.