Initial Cookiecutter Commit.
Change-Id: I58c1914ca033c2c40ba2b63a3c07e0a9a8397ba4
This commit is contained in:
parent
c0809470d1
commit
403102e8e9
7
.coveragerc
Normal file
7
.coveragerc
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
[run]
|
||||||
|
branch = True
|
||||||
|
source = storlets
|
||||||
|
omit = storlets/openstack/*
|
||||||
|
|
||||||
|
[report]
|
||||||
|
ignore-errors = True
|
54
.gitignore
vendored
Normal file
54
.gitignore
vendored
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
*.py[cod]
|
||||||
|
|
||||||
|
# C extensions
|
||||||
|
*.so
|
||||||
|
|
||||||
|
# Packages
|
||||||
|
*.egg
|
||||||
|
*.egg-info
|
||||||
|
dist
|
||||||
|
build
|
||||||
|
.eggs
|
||||||
|
eggs
|
||||||
|
parts
|
||||||
|
bin
|
||||||
|
var
|
||||||
|
sdist
|
||||||
|
develop-eggs
|
||||||
|
.installed.cfg
|
||||||
|
lib
|
||||||
|
lib64
|
||||||
|
|
||||||
|
# Installer logs
|
||||||
|
pip-log.txt
|
||||||
|
|
||||||
|
# Unit test / coverage reports
|
||||||
|
.coverage
|
||||||
|
.tox
|
||||||
|
nosetests.xml
|
||||||
|
.testrepository
|
||||||
|
.venv
|
||||||
|
|
||||||
|
# Translations
|
||||||
|
*.mo
|
||||||
|
|
||||||
|
# Mr Developer
|
||||||
|
.mr.developer.cfg
|
||||||
|
.project
|
||||||
|
.pydevproject
|
||||||
|
|
||||||
|
# Complexity
|
||||||
|
output/*.html
|
||||||
|
output/*/index.html
|
||||||
|
|
||||||
|
# Sphinx
|
||||||
|
doc/build
|
||||||
|
|
||||||
|
# pbr generates these
|
||||||
|
AUTHORS
|
||||||
|
ChangeLog
|
||||||
|
|
||||||
|
# Editors
|
||||||
|
*~
|
||||||
|
.*.swp
|
||||||
|
.*sw?
|
4
.gitreview
Normal file
4
.gitreview
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
[gerrit]
|
||||||
|
host=review.openstack.org
|
||||||
|
port=29418
|
||||||
|
project=openstack/storlets.git
|
3
.mailmap
Normal file
3
.mailmap
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
# Format is:
|
||||||
|
# <preferred e-mail> <other e-mail 1>
|
||||||
|
# <preferred e-mail> <other e-mail 2>
|
7
.testr.conf
Normal file
7
.testr.conf
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
[DEFAULT]
|
||||||
|
test_command=OS_STDOUT_CAPTURE=${OS_STDOUT_CAPTURE:-1} \
|
||||||
|
OS_STDERR_CAPTURE=${OS_STDERR_CAPTURE:-1} \
|
||||||
|
OS_TEST_TIMEOUT=${OS_TEST_TIMEOUT:-60} \
|
||||||
|
${PYTHON:-python} -m subunit.run discover -t ./ . $LISTOPT $IDOPTION
|
||||||
|
test_id_option=--load-list $IDFILE
|
||||||
|
test_list_option=--list
|
17
CONTRIBUTING.rst
Normal file
17
CONTRIBUTING.rst
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
If you would like to contribute to the development of OpenStack, you must
|
||||||
|
follow the steps in this page:
|
||||||
|
|
||||||
|
http://docs.openstack.org/infra/manual/developers.html
|
||||||
|
|
||||||
|
If you already have a good understanding of how the system works and your
|
||||||
|
OpenStack accounts are set up, you can skip to the development workflow
|
||||||
|
section of this documentation to learn how changes to OpenStack should be
|
||||||
|
submitted for review via the Gerrit tool:
|
||||||
|
|
||||||
|
http://docs.openstack.org/infra/manual/developers.html#development-workflow
|
||||||
|
|
||||||
|
Pull requests submitted through GitHub will be ignored.
|
||||||
|
|
||||||
|
Bugs should be filed on Launchpad, not GitHub:
|
||||||
|
|
||||||
|
https://bugs.launchpad.net/storlets
|
@ -1,6 +1,6 @@
|
|||||||
ibm_container_install_dir: opt/ibm
|
ibm_container_install_dir: opt/ibm
|
||||||
lxc_device: /home/docker_device
|
lxc_device: /home/docker_device
|
||||||
storlet_source_dir: ~/swift-storlets
|
storlet_source_dir: ~/storlets
|
||||||
python_dist_packages_dir: usr/local/lib/python2.7/dist-packages
|
python_dist_packages_dir: usr/local/lib/python2.7/dist-packages
|
||||||
storlet_gateway_conf_file: /etc/swift/storlet_docker_gateway.conf
|
storlet_gateway_conf_file: /etc/swift/storlet_docker_gateway.conf
|
||||||
|
|
||||||
|
@ -19,35 +19,38 @@ Limitations under the License.
|
|||||||
@author: cdoron
|
@author: cdoron
|
||||||
'''
|
'''
|
||||||
|
|
||||||
import sys
|
|
||||||
import subprocess
|
|
||||||
import json
|
import json
|
||||||
|
import subprocess
|
||||||
|
import sys
|
||||||
|
|
||||||
|
|
||||||
def extractId(tar_file_name, repository, tag):
|
def extractId(tar_file_name, repository, tag):
|
||||||
subprocess.call(['tar', 'xf', tar_file_name, 'repositories'])
|
subprocess.call(['tar', 'xf', tar_file_name, 'repositories'])
|
||||||
repository_file = open('repositories')
|
repository_file = open('repositories')
|
||||||
j = json.loads(repository_file.read())
|
j = json.loads(repository_file.read())
|
||||||
|
|
||||||
if not repository in j:
|
if repository not in j:
|
||||||
print "Not Found"
|
print("Not Found")
|
||||||
else:
|
else:
|
||||||
pairs = j[repository]
|
pairs = j[repository]
|
||||||
if tag:
|
if tag:
|
||||||
if tag not in pairs:
|
if tag not in pairs:
|
||||||
print "Not Found"
|
print("Not Found")
|
||||||
else:
|
else:
|
||||||
print pairs[tag]
|
print(pairs[tag])
|
||||||
else:
|
else:
|
||||||
if len(pairs) != 1:
|
if len(pairs) != 1:
|
||||||
print "No tag supplied. Ambiguous"
|
print("No tag supplied. Ambiguous")
|
||||||
else:
|
else:
|
||||||
print pairs.values()[0]
|
print(pairs.values()[0])
|
||||||
|
|
||||||
repository_file.close()
|
repository_file.close()
|
||||||
subprocess.call(['rm', '-f', 'repositories'])
|
subprocess.call(['rm', '-f', 'repositories'])
|
||||||
|
|
||||||
|
|
||||||
def usage(argv):
|
def usage(argv):
|
||||||
print argv[0] + " <tar_file> <repository> [tag]"
|
print(argv[0] + " <tar_file> <repository> [tag]")
|
||||||
|
|
||||||
|
|
||||||
def main(argv):
|
def main(argv):
|
||||||
if len(argv) < 3 or len(argv) > 4:
|
if len(argv) < 3 or len(argv) > 4:
|
||||||
@ -64,5 +67,3 @@ def main(argv):
|
|||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
main(sys.argv)
|
main(sys.argv)
|
||||||
|
|
||||||
|
|
||||||
|
@ -21,5 +21,5 @@
|
|||||||
command: docker pull {{ hostvars[groups['docker'][0]]['inventory_hostname'] }}:{{ docker_registry_port }}/{{ tenant_id.stdout_lines[0] }}
|
command: docker pull {{ hostvars[groups['docker'][0]]['inventory_hostname'] }}:{{ docker_registry_port }}/{{ tenant_id.stdout_lines[0] }}
|
||||||
|
|
||||||
- name: shutdown_container
|
- name: shutdown_container
|
||||||
shell: "{{ lxc_device }}/scripts/send_halt_cmd_to_daemon_factory.py
|
shell: "/usr/bin/python {{ lxc_device }}/scripts/send_halt_cmd_to_daemon_factory.py
|
||||||
{{ lxc_device }}/pipes/scopes/AUTH_{{ tenant_id.stdout_lines[0] }}/factory_pipe"
|
{{ lxc_device }}/pipes/scopes/AUTH_{{ tenant_id.stdout_lines[0] }}/factory_pipe"
|
||||||
|
@ -13,21 +13,22 @@ See the License for the specific language governing permissions and
|
|||||||
Limitations under the License.
|
Limitations under the License.
|
||||||
-------------------------------------------------------------------------'''
|
-------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
'''
|
'''
|
||||||
@author: cdoron
|
@author: cdoron
|
||||||
'''
|
'''
|
||||||
|
import ConfigParser
|
||||||
|
import fileinput
|
||||||
import os
|
import os
|
||||||
import sys
|
|
||||||
import pwd
|
import pwd
|
||||||
import shutil
|
import shutil
|
||||||
import fileinput
|
import sys
|
||||||
import ConfigParser
|
|
||||||
|
|
||||||
def _chown_to_swift(path):
|
def _chown_to_swift(path):
|
||||||
uc = pwd.getpwnam('swift')
|
uc = pwd.getpwnam('swift')
|
||||||
os.chown(path, uc.pw_uid, uc.pw_gid)
|
os.chown(path, uc.pw_uid, uc.pw_gid)
|
||||||
|
|
||||||
|
|
||||||
def _unpatch_pipeline_line(orig_line, storlet_middleware):
|
def _unpatch_pipeline_line(orig_line, storlet_middleware):
|
||||||
mds = list()
|
mds = list()
|
||||||
for md in orig_line.split():
|
for md in orig_line.split():
|
||||||
@ -44,6 +45,7 @@ def _unpatch_pipeline_line(orig_line, storlet_middleware):
|
|||||||
|
|
||||||
return new_line + '\n'
|
return new_line + '\n'
|
||||||
|
|
||||||
|
|
||||||
def _patch_proxy_pipeline_line(orig_line, storlet_middleware):
|
def _patch_proxy_pipeline_line(orig_line, storlet_middleware):
|
||||||
mds = list()
|
mds = list()
|
||||||
for md in orig_line.split():
|
for md in orig_line.split():
|
||||||
@ -71,7 +73,8 @@ def _patch_proxy_pipeline_line(orig_line, storlet_middleware):
|
|||||||
|
|
||||||
return new_line + '\n'
|
return new_line + '\n'
|
||||||
|
|
||||||
def _patch_object_pipeline_line(orig_line,storlet_middleware):
|
|
||||||
|
def _patch_object_pipeline_line(orig_line, storlet_middleware):
|
||||||
mds = list()
|
mds = list()
|
||||||
for md in orig_line.split():
|
for md in orig_line.split():
|
||||||
if md == 'pipeline' or md == '=':
|
if md == 'pipeline' or md == '=':
|
||||||
@ -90,100 +93,119 @@ def _patch_object_pipeline_line(orig_line,storlet_middleware):
|
|||||||
|
|
||||||
return new_line + '\n'
|
return new_line + '\n'
|
||||||
|
|
||||||
def unpatch_swift_config_file(conf, conf_file):
|
|
||||||
storlet_middleware = conf.get('common-confs','storlet_middleware')
|
|
||||||
filter_block_first_line = '[filter:%s]\n' % storlet_middleware
|
|
||||||
|
|
||||||
for line in fileinput.input(conf_file, inplace = 1):
|
def unpatch_swift_config_file(conf, conf_file):
|
||||||
|
storlet_middleware = conf.get('common-confs', 'storlet_middleware')
|
||||||
|
|
||||||
|
for line in fileinput.input(conf_file, inplace=1):
|
||||||
if line.startswith('pipeline'):
|
if line.startswith('pipeline'):
|
||||||
new_line = _unpatch_pipeline_line(line, storlet_middleware)
|
new_line = _unpatch_pipeline_line(line, storlet_middleware)
|
||||||
line = new_line
|
line = new_line
|
||||||
print line,
|
sys.stdout.write(line)
|
||||||
|
|
||||||
_chown_to_swift(conf_file)
|
_chown_to_swift(conf_file)
|
||||||
|
|
||||||
|
|
||||||
def patch_swift_config_file(conf, conf_file, service):
|
def patch_swift_config_file(conf, conf_file, service):
|
||||||
storlet_middleware = conf.get('common-confs','storlet_middleware')
|
storlet_middleware = conf.get('common-confs', 'storlet_middleware')
|
||||||
storlet_gateway_implementation_class = conf.get('common-confs','storlet_gateway_module')
|
|
||||||
filter_block_first_line = '[filter:%s]\n' % storlet_middleware
|
filter_block_first_line = '[filter:%s]\n' % storlet_middleware
|
||||||
|
|
||||||
filter_in_file = False
|
filter_in_file = False
|
||||||
for line in fileinput.input(conf_file, inplace = 1):
|
for line in fileinput.input(conf_file, inplace=1):
|
||||||
if line.startswith('pipeline'):
|
if line.startswith('pipeline'):
|
||||||
if service == 'proxy':
|
if service == 'proxy':
|
||||||
new_line = _patch_proxy_pipeline_line(line,storlet_middleware)
|
new_line = _patch_proxy_pipeline_line(line, storlet_middleware)
|
||||||
else:
|
else:
|
||||||
new_line = _patch_object_pipeline_line(line,storlet_middleware)
|
new_line = _patch_object_pipeline_line(line,
|
||||||
|
storlet_middleware)
|
||||||
line = new_line
|
line = new_line
|
||||||
if filter_block_first_line in line:
|
if filter_block_first_line in line:
|
||||||
filter_in_file = True
|
filter_in_file = True
|
||||||
print line,
|
sys.stdout.write(line)
|
||||||
|
|
||||||
if filter_in_file == False:
|
if filter_in_file is False:
|
||||||
with open(conf_file, 'a') as f:
|
with open(conf_file, 'a') as f:
|
||||||
f.write('\n')
|
f.write('\n')
|
||||||
f.write(filter_block_first_line)
|
f.write(filter_block_first_line)
|
||||||
f.write('use = egg:storlets#%s\n' % storlet_middleware)
|
f.write('use = egg:storlets#%s\n' % storlet_middleware)
|
||||||
f.write('storlet_container = %s\n' % conf.get('common-confs','storlet_container'))
|
f.write('storlet_container = %s\n' %
|
||||||
f.write('storlet_dependency = %s\n' % conf.get('common-confs','storlet_dependency'))
|
conf.get('common-confs', 'storlet_container'))
|
||||||
f.write('storlet_timeout = %s\n' % conf.get('common-confs','storlet_timeout'))
|
f.write('storlet_dependency = %s\n' %
|
||||||
f.write('storlet_gateway_module = %s\n' % conf.get('common-confs','storlet_gateway_module'))
|
conf.get('common-confs', 'storlet_dependency'))
|
||||||
f.write('storlet_gateway_conf = %s\n' % conf.get('common-confs','storlet_gateway_conf'))
|
f.write('storlet_timeout = %s\n' %
|
||||||
f.write('storlet_execute_on_proxy_only = %s\n' % conf.get('common-confs','storlet_proxy_execution'))
|
conf.get('common-confs', 'storlet_timeout'))
|
||||||
|
f.write('storlet_gateway_module = %s\n' %
|
||||||
|
conf.get('common-confs', 'storlet_gateway_module'))
|
||||||
|
f.write('storlet_gateway_conf = %s\n' %
|
||||||
|
conf.get('common-confs', 'storlet_gateway_conf'))
|
||||||
|
f.write('storlet_execute_on_proxy_only = %s\n' % conf.get(
|
||||||
|
'common-confs', 'storlet_proxy_execution'))
|
||||||
f.write('execution_server = %s\n' % service)
|
f.write('execution_server = %s\n' % service)
|
||||||
|
|
||||||
_chown_to_swift(conf_file)
|
_chown_to_swift(conf_file)
|
||||||
|
|
||||||
|
|
||||||
def unpatch_swift_storlet_proxy_file(conf):
|
def unpatch_swift_storlet_proxy_file(conf):
|
||||||
storlet_proxy_server_conf_file = conf.get('proxy-confs','storlet_proxy_server_conf_file')
|
storlet_proxy_server_conf_file = conf.get('proxy-confs',
|
||||||
|
'storlet_proxy_server_conf_file')
|
||||||
if os.path.exists(storlet_proxy_server_conf_file):
|
if os.path.exists(storlet_proxy_server_conf_file):
|
||||||
os.remove(storlet_proxy_server_conf_file)
|
os.remove(storlet_proxy_server_conf_file)
|
||||||
|
|
||||||
|
|
||||||
def patch_swift_storlet_proxy_file(conf):
|
def patch_swift_storlet_proxy_file(conf):
|
||||||
storlet_proxy_server_conf_file = conf.get('proxy-confs','storlet_proxy_server_conf_file')
|
storlet_proxy_server_conf_file = conf.get('proxy-confs',
|
||||||
|
'storlet_proxy_server_conf_file')
|
||||||
proxy_server_conf_file = conf.get('proxy-confs', 'proxy_server_conf_file')
|
proxy_server_conf_file = conf.get('proxy-confs', 'proxy_server_conf_file')
|
||||||
|
|
||||||
source_file = proxy_server_conf_file
|
source_file = proxy_server_conf_file
|
||||||
target_file = storlet_proxy_server_conf_file
|
target_file = storlet_proxy_server_conf_file
|
||||||
shutil.copyfile(source_file, target_file)
|
shutil.copyfile(source_file, target_file)
|
||||||
|
|
||||||
for line in fileinput.input(storlet_proxy_server_conf_file, inplace = 1):
|
for line in fileinput.input(storlet_proxy_server_conf_file, inplace=1):
|
||||||
if line.startswith('pipeline'):
|
if line.startswith('pipeline'):
|
||||||
line= 'pipeline = proxy-logging cache storlet_handler slo proxy-logging proxy-server\n'
|
line = 'pipeline = proxy-logging cache storlet_handler slo ' + \
|
||||||
print line,
|
'proxy-logging proxy-server\n'
|
||||||
|
sys.stdout.write(line)
|
||||||
|
|
||||||
_chown_to_swift(storlet_proxy_server_conf_file)
|
_chown_to_swift(storlet_proxy_server_conf_file)
|
||||||
|
|
||||||
|
|
||||||
def remove_gateway_conf_file(conf):
|
def remove_gateway_conf_file(conf):
|
||||||
gateway_conf_file = conf.get('common-confs', 'storlet_gateway_conf')
|
gateway_conf_file = conf.get('common-confs', 'storlet_gateway_conf')
|
||||||
if os.path.exists(gateway_conf_file):
|
if os.path.exists(gateway_conf_file):
|
||||||
os.remove(gateway_conf_file)
|
os.remove(gateway_conf_file)
|
||||||
|
|
||||||
|
|
||||||
def remove(conf):
|
def remove(conf):
|
||||||
object_server_conf_files = conf.get('object-confs', 'object_server_conf_files').split(',')
|
object_server_conf_files = conf.get('object-confs',
|
||||||
|
'object_server_conf_files').split(',')
|
||||||
for f in object_server_conf_files:
|
for f in object_server_conf_files:
|
||||||
if os.path.exists(f):
|
if os.path.exists(f):
|
||||||
unpatch_swift_config_file(conf, f)
|
unpatch_swift_config_file(conf, f)
|
||||||
|
|
||||||
proxy_server_conf_file = conf.get('proxy-confs','proxy_server_conf_file')
|
proxy_server_conf_file = conf.get('proxy-confs', 'proxy_server_conf_file')
|
||||||
unpatch_swift_config_file(conf, proxy_server_conf_file)
|
unpatch_swift_config_file(conf, proxy_server_conf_file)
|
||||||
|
|
||||||
unpatch_swift_storlet_proxy_file(conf)
|
unpatch_swift_storlet_proxy_file(conf)
|
||||||
remove_gateway_conf_file(conf)
|
remove_gateway_conf_file(conf)
|
||||||
|
|
||||||
|
|
||||||
def install(conf):
|
def install(conf):
|
||||||
object_server_conf_files = conf.get('object-confs', 'object_server_conf_files').split(',')
|
object_server_conf_files = conf.get('object-confs',
|
||||||
|
'object_server_conf_files').split(',')
|
||||||
for f in object_server_conf_files:
|
for f in object_server_conf_files:
|
||||||
if os.path.exists(f):
|
if os.path.exists(f):
|
||||||
patch_swift_config_file(conf, f, 'object')
|
patch_swift_config_file(conf, f, 'object')
|
||||||
|
|
||||||
proxy_server_conf_file = conf.get('proxy-confs','proxy_server_conf_file')
|
proxy_server_conf_file = conf.get('proxy-confs', 'proxy_server_conf_file')
|
||||||
patch_swift_config_file(conf, proxy_server_conf_file, 'proxy')
|
patch_swift_config_file(conf, proxy_server_conf_file, 'proxy')
|
||||||
|
|
||||||
patch_swift_storlet_proxy_file(conf)
|
patch_swift_storlet_proxy_file(conf)
|
||||||
|
|
||||||
|
|
||||||
def usage(argv):
|
def usage(argv):
|
||||||
print "Usage: " + argv[0] + " install/remove conf_file"
|
print("Usage: " + argv[0] + " install/remove conf_file")
|
||||||
|
|
||||||
|
|
||||||
def main(argv):
|
def main(argv):
|
||||||
if len(argv) != 3:
|
if len(argv) != 3:
|
||||||
|
@ -29,16 +29,17 @@ from ctypes import POINTER
|
|||||||
|
|
||||||
|
|
||||||
class SBus(object):
|
class SBus(object):
|
||||||
'''
|
'''@summary: This class wraps low level C-API for SBus functionality
|
||||||
@summary: This class wraps low level C-API for SBus functionality
|
|
||||||
to be used with Python
|
to be used with Python
|
||||||
'''
|
'''
|
||||||
SBUS_SO_NAME = '/usr/local/lib/python2.7/dist-packages/sbus.so'
|
SBUS_SO_NAME = '/usr/local/lib/python2.7/dist-packages/sbus.so'
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
'''
|
'''@summary: CTOR
|
||||||
@summary: CTOR
|
|
||||||
Setup argument types mappings.
|
Setup argument types mappings.
|
||||||
'''
|
'''
|
||||||
|
|
||||||
@ -46,12 +47,12 @@ class SBus(object):
|
|||||||
self.sbus_back_ = ctypes.CDLL(SBus.SBUS_SO_NAME)
|
self.sbus_back_ = ctypes.CDLL(SBus.SBUS_SO_NAME)
|
||||||
|
|
||||||
# create SBus
|
# create SBus
|
||||||
self.sbus_back_.sbus_create.argtypes = [c_char_p]
|
self.sbus_back_.sbus_create.argtypes = [c_char_p]
|
||||||
self.sbus_back_.sbus_create.restype = c_int
|
self.sbus_back_.sbus_create.restype = c_int
|
||||||
|
|
||||||
# listen to SBus
|
# listen to SBus
|
||||||
self.sbus_back_.sbus_listen.argtypes = [c_int]
|
self.sbus_back_.sbus_listen.argtypes = [c_int]
|
||||||
self.sbus_back_.sbus_listen.restype = c_int
|
self.sbus_back_.sbus_listen.restype = c_int
|
||||||
|
|
||||||
# send message
|
# send message
|
||||||
self.sbus_back_.sbus_send_msg.argtypes = [c_char_p,
|
self.sbus_back_.sbus_send_msg.argtypes = [c_char_p,
|
||||||
@ -61,7 +62,7 @@ class SBus(object):
|
|||||||
c_int,
|
c_int,
|
||||||
c_char_p,
|
c_char_p,
|
||||||
c_int]
|
c_int]
|
||||||
self.sbus_back_.sbus_send_msg.restype = c_int
|
self.sbus_back_.sbus_send_msg.restype = c_int
|
||||||
|
|
||||||
# receive message
|
# receive message
|
||||||
self.sbus_back_.sbus_recv_msg.argtypes = [c_int,
|
self.sbus_back_.sbus_recv_msg.argtypes = [c_int,
|
||||||
@ -71,13 +72,13 @@ class SBus(object):
|
|||||||
POINTER(c_int),
|
POINTER(c_int),
|
||||||
POINTER(c_char_p),
|
POINTER(c_char_p),
|
||||||
POINTER(c_int)]
|
POINTER(c_int)]
|
||||||
self.sbus_back_.sbus_recv_msg.restype = c_int
|
self.sbus_back_.sbus_recv_msg.restype = c_int
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def start_logger(str_log_level='DEBUG', container_id=None):
|
def start_logger(str_log_level='DEBUG', container_id=None):
|
||||||
'''
|
'''@summary: Start logger.
|
||||||
@summary: Start logger.
|
|
||||||
|
|
||||||
@param str_log_level: The level of verbosity in log records.
|
@param str_log_level: The level of verbosity in log records.
|
||||||
Default value - 'DEBUG'.
|
Default value - 'DEBUG'.
|
||||||
@ -92,10 +93,10 @@ class SBus(object):
|
|||||||
sbus_back_.sbus_start_logger(str_log_level, container_id)
|
sbus_back_.sbus_start_logger(str_log_level, container_id)
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def stop_logger():
|
def stop_logger():
|
||||||
'''
|
'''@summary: Stop logger.
|
||||||
@summary: Stop logger.
|
|
||||||
|
|
||||||
@rtype: void
|
@rtype: void
|
||||||
'''
|
'''
|
||||||
@ -104,9 +105,9 @@ class SBus(object):
|
|||||||
sbus_back_.sbus_stop_logger()
|
sbus_back_.sbus_stop_logger()
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def create(self, sbus_name):
|
def create(self, sbus_name):
|
||||||
'''
|
'''@summary: Instantiate an SBus. A wrapper for C function.
|
||||||
@summary: Instantiate an SBus. A wrapper for C function.
|
|
||||||
|
|
||||||
@param sbus_name: Path to domain socket "file".
|
@param sbus_name: Path to domain socket "file".
|
||||||
@type sbus_name: string
|
@type sbus_name: string
|
||||||
@ -117,9 +118,10 @@ class SBus(object):
|
|||||||
return self.sbus_back_.sbus_create(sbus_name)
|
return self.sbus_back_.sbus_create(sbus_name)
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def listen(self, sbus_handler):
|
def listen(self, sbus_handler):
|
||||||
'''
|
'''@summary: Listen to the SBus.
|
||||||
@summary: Listen to the SBus.
|
|
||||||
Suspend the executing thread.
|
Suspend the executing thread.
|
||||||
|
|
||||||
@param sbus_handler: Handler to SBus to listen.
|
@param sbus_handler: Handler to SBus to listen.
|
||||||
@ -131,9 +133,10 @@ class SBus(object):
|
|||||||
return self.sbus_back_.sbus_listen(sbus_handler)
|
return self.sbus_back_.sbus_listen(sbus_handler)
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def receive(self, sbus_handler):
|
def receive(self, sbus_handler):
|
||||||
'''
|
'''@summary: Read the data from SBus.
|
||||||
@summary: Read the data from SBus.
|
|
||||||
Create a datagram.
|
Create a datagram.
|
||||||
|
|
||||||
@param sbus_handler: Handler to SBus to read data from.
|
@param sbus_handler: Handler to SBus to read data from.
|
||||||
@ -142,12 +145,12 @@ class SBus(object):
|
|||||||
@return: An object with the obtained data. Null-able.
|
@return: An object with the obtained data. Null-able.
|
||||||
@rtype: SBusDatagram
|
@rtype: SBusDatagram
|
||||||
'''
|
'''
|
||||||
ph_files = POINTER(c_int)()
|
ph_files = POINTER(c_int)()
|
||||||
pp_metadata = (c_char_p)()
|
pp_metadata = (c_char_p)()
|
||||||
pp_params = (c_char_p)()
|
pp_params = (c_char_p)()
|
||||||
pn_files = (c_int)()
|
pn_files = (c_int)()
|
||||||
pn_metadata = (c_int)()
|
pn_metadata = (c_int)()
|
||||||
pn_params = (c_int)()
|
pn_params = (c_int)()
|
||||||
|
|
||||||
# Invoke C function
|
# Invoke C function
|
||||||
n_status = self.sbus_back_.sbus_recv_msg(sbus_handler,
|
n_status = self.sbus_back_.sbus_recv_msg(sbus_handler,
|
||||||
@ -157,7 +160,7 @@ class SBus(object):
|
|||||||
pn_metadata,
|
pn_metadata,
|
||||||
pp_params,
|
pp_params,
|
||||||
pn_params)
|
pn_params)
|
||||||
result_dtg = None
|
result_dtg = None
|
||||||
if 0 <= n_status:
|
if 0 <= n_status:
|
||||||
# The invocation was successful.
|
# The invocation was successful.
|
||||||
# De-serialize the data
|
# De-serialize the data
|
||||||
@ -169,10 +172,10 @@ class SBus(object):
|
|||||||
h_files.append(ph_files[i])
|
h_files.append(ph_files[i])
|
||||||
|
|
||||||
# Extract Python strings
|
# Extract Python strings
|
||||||
n_metadata = pn_metadata.value
|
n_metadata = pn_metadata.value
|
||||||
str_metadata = pp_metadata.value
|
str_metadata = pp_metadata.value
|
||||||
n_params = pn_params.value
|
n_params = pn_params.value
|
||||||
str_params = pp_params.value
|
str_params = pp_params.value
|
||||||
|
|
||||||
# Trim the junk out
|
# Trim the junk out
|
||||||
if 0 < n_metadata:
|
if 0 < n_metadata:
|
||||||
@ -187,10 +190,11 @@ class SBus(object):
|
|||||||
return result_dtg
|
return result_dtg
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def send(sbus_name, datagram):
|
def send(sbus_name, datagram):
|
||||||
'''
|
'''@summary: Send the datagram through SBus.
|
||||||
@summary: Send the datagram through SBus.
|
|
||||||
Serialize dictionaries into JSON strings.
|
Serialize dictionaries into JSON strings.
|
||||||
|
|
||||||
@param sbus_name: Path to domain socket "file".
|
@param sbus_name: Path to domain socket "file".
|
||||||
@ -204,30 +208,30 @@ class SBus(object):
|
|||||||
|
|
||||||
# Serialize the datagram into JSON strings and C integer array
|
# Serialize the datagram into JSON strings and C integer array
|
||||||
str_json_params = datagram.get_params_and_cmd_as_json()
|
str_json_params = datagram.get_params_and_cmd_as_json()
|
||||||
p_params = c_char_p(str_json_params)
|
p_params = c_char_p(str_json_params)
|
||||||
n_params = c_int(len(str_json_params))
|
n_params = c_int(len(str_json_params))
|
||||||
|
|
||||||
n_files = c_int(0)
|
n_files = c_int(0)
|
||||||
h_files = None
|
h_files = None
|
||||||
n_metadata = c_int(0)
|
n_metadata = c_int(0)
|
||||||
p_metadata = None
|
p_metadata = None
|
||||||
|
|
||||||
if datagram.get_num_files() > 0:
|
if datagram.get_num_files() > 0:
|
||||||
str_json_metadata = datagram.get_files_metadata_as_json()
|
str_json_metadata = datagram.get_files_metadata_as_json()
|
||||||
p_metadata = c_char_p(str_json_metadata)
|
p_metadata = c_char_p(str_json_metadata)
|
||||||
n_metadata = c_int(len(str_json_metadata))
|
n_metadata = c_int(len(str_json_metadata))
|
||||||
|
|
||||||
n_fds = datagram.get_num_files()
|
n_fds = datagram.get_num_files()
|
||||||
n_files = c_int(n_fds)
|
n_files = c_int(n_fds)
|
||||||
|
|
||||||
file_fds = datagram.get_files()
|
file_fds = datagram.get_files()
|
||||||
h_files = (c_int * n_fds)()
|
h_files = (c_int * n_fds)()
|
||||||
|
|
||||||
for i in range(n_fds):
|
for i in range(n_fds):
|
||||||
h_files[i] = file_fds[i]
|
h_files[i] = file_fds[i]
|
||||||
|
|
||||||
# Invoke C function
|
# Invoke C function
|
||||||
sbus = SBus()
|
sbus = SBus()
|
||||||
n_status = sbus.sbus_back_.sbus_send_msg(sbus_name,
|
n_status = sbus.sbus_back_.sbus_send_msg(sbus_name,
|
||||||
h_files,
|
h_files,
|
||||||
n_files,
|
n_files,
|
||||||
@ -237,5 +241,4 @@ class SBus(object):
|
|||||||
n_params)
|
n_params)
|
||||||
return n_status
|
return n_status
|
||||||
|
|
||||||
|
|
||||||
'''============================ END OF FILE ==============================='''
|
'''============================ END OF FILE ==============================='''
|
||||||
|
@ -21,19 +21,19 @@ Limitations under the License.
|
|||||||
dictionary of dictionaries
|
dictionary of dictionaries
|
||||||
==========================================================================='''
|
==========================================================================='''
|
||||||
|
|
||||||
import os
|
|
||||||
import json
|
import json
|
||||||
|
import os
|
||||||
import syslog
|
import syslog
|
||||||
|
|
||||||
from SBusStorletCommand import SBUS_CMD_NOP
|
|
||||||
from SBusFileDescription import SBUS_FD_OUTPUT_OBJECT
|
from SBusFileDescription import SBUS_FD_OUTPUT_OBJECT
|
||||||
|
from SBusStorletCommand import SBUS_CMD_NOP
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
class SBusDatagram(object):
|
class SBusDatagram(object):
|
||||||
'''
|
'''@summary: This class aggregates data to be transferred
|
||||||
@summary: This class aggregates data to be transferred
|
|
||||||
using SBus functionality.
|
using SBus functionality.
|
||||||
'''
|
'''
|
||||||
|
|
||||||
@ -41,9 +41,9 @@ class SBusDatagram(object):
|
|||||||
task_id_dict_key_name_ = 'taskId'
|
task_id_dict_key_name_ = 'taskId'
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
'''
|
'''@summary: CTOR
|
||||||
@summary: CTOR
|
|
||||||
|
|
||||||
@ivar e_command_ : A command to Storlet Daemon.
|
@ivar e_command_ : A command to Storlet Daemon.
|
||||||
@type e_command_ : Integer. SBusStorletCommand enumerated value.
|
@type e_command_ : Integer. SBusStorletCommand enumerated value.
|
||||||
@ -59,19 +59,20 @@ class SBusDatagram(object):
|
|||||||
@invariant: Quantity of entries in files_metadata_ list
|
@invariant: Quantity of entries in files_metadata_ list
|
||||||
is the same as in h_files_, i.e. n_files_.
|
is the same as in h_files_, i.e. n_files_.
|
||||||
'''
|
'''
|
||||||
self.e_command_ = SBUS_CMD_NOP
|
self.e_command_ = SBUS_CMD_NOP
|
||||||
self.task_id_ = None
|
self.task_id_ = None
|
||||||
self.h_files_ = None
|
self.h_files_ = None
|
||||||
self.n_files_ = 0
|
self.n_files_ = 0
|
||||||
self.files_metadata_ = None
|
self.files_metadata_ = None
|
||||||
self.exec_params_ = None
|
self.exec_params_ = None
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def create_service_datagram(command,
|
def create_service_datagram(command,
|
||||||
outfd):
|
outfd):
|
||||||
'''
|
'''@summary: Datagram static factory.
|
||||||
@summary: Datagram static factory.
|
|
||||||
Create "service" datagram, i.e.
|
Create "service" datagram, i.e.
|
||||||
- command shall be one of
|
- command shall be one of
|
||||||
{PING, START/STOP/STATUS-DAEMON}
|
{PING, START/STOP/STATUS-DAEMON}
|
||||||
@ -99,12 +100,13 @@ class SBusDatagram(object):
|
|||||||
return dtg
|
return dtg
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def from_raw_data(self,
|
def from_raw_data(self,
|
||||||
h_files,
|
h_files,
|
||||||
str_json_metadata,
|
str_json_metadata,
|
||||||
str_json_params):
|
str_json_params):
|
||||||
'''
|
'''@summary: CTOR
|
||||||
@summary: CTOR
|
|
||||||
Construct object from file list and
|
Construct object from file list and
|
||||||
two JSON-encoded strings.
|
two JSON-encoded strings.
|
||||||
|
|
||||||
@ -122,10 +124,11 @@ class SBusDatagram(object):
|
|||||||
self.extract_params(str_json_params)
|
self.extract_params(str_json_params)
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def extract_metadata(self,
|
def extract_metadata(self,
|
||||||
str_json_metadata):
|
str_json_metadata):
|
||||||
'''
|
'''@summary: Extract files_metadata array
|
||||||
@summary: Extract files_metadata array
|
|
||||||
of dictionaries form a JSON string
|
of dictionaries form a JSON string
|
||||||
@requires: n_files_ has to be se
|
@requires: n_files_ has to be se
|
||||||
|
|
||||||
@ -142,9 +145,10 @@ class SBusDatagram(object):
|
|||||||
self.files_metadata_.append(json.loads(str_curr_metadata))
|
self.files_metadata_.append(json.loads(str_curr_metadata))
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def extract_params(self, str_json_params):
|
def extract_params(self, str_json_params):
|
||||||
'''
|
'''@summary: Extract command field and exec_params
|
||||||
@summary: Extract command field and exec_params
|
|
||||||
dictionary form a JSON string
|
dictionary form a JSON string
|
||||||
@param str_json_params: JSON encoding for the execution parameters.
|
@param str_json_params: JSON encoding for the execution parameters.
|
||||||
@type str_json_params: string.
|
@type str_json_params: string.
|
||||||
@ -169,9 +173,10 @@ class SBusDatagram(object):
|
|||||||
self.exec_params_ = None
|
self.exec_params_ = None
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def get_params_and_cmd_as_json(self):
|
def get_params_and_cmd_as_json(self):
|
||||||
'''
|
'''@summary: Convert command field and execution parameters
|
||||||
@summary: Convert command field and execution parameters
|
|
||||||
dictionary into JSON as the following -
|
dictionary into JSON as the following -
|
||||||
1. Copy exec_params_. Initialize the combined dictionary.
|
1. Copy exec_params_. Initialize the combined dictionary.
|
||||||
2. Push the next pair into the combined dictionary
|
2. Push the next pair into the combined dictionary
|
||||||
@ -193,9 +198,10 @@ class SBusDatagram(object):
|
|||||||
return str_result
|
return str_result
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def get_files_metadata_as_json(self):
|
def get_files_metadata_as_json(self):
|
||||||
'''
|
'''@summary: Encode the list of dictionaries into JSON as the following
|
||||||
@summary: Encode the list of dictionaries into JSON as the following -
|
|
||||||
1. Create a combined dictionary (Integer-to-String)
|
1. Create a combined dictionary (Integer-to-String)
|
||||||
Key - index in the original list
|
Key - index in the original list
|
||||||
Value - JSON encoding of the certain dictionary
|
Value - JSON encoding of the certain dictionary
|
||||||
@ -213,9 +219,9 @@ class SBusDatagram(object):
|
|||||||
return str_result
|
return str_result
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def get_num_files(self):
|
def get_num_files(self):
|
||||||
'''
|
'''@summary: Getter.
|
||||||
@summary: Getter.
|
|
||||||
|
|
||||||
@return: The quantity of file descriptors.
|
@return: The quantity of file descriptors.
|
||||||
@rtype: integer
|
@rtype: integer
|
||||||
@ -223,9 +229,9 @@ class SBusDatagram(object):
|
|||||||
return self.n_files_
|
return self.n_files_
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def get_files(self):
|
def get_files(self):
|
||||||
'''
|
'''@summary: Getter.
|
||||||
@summary: Getter.
|
|
||||||
|
|
||||||
@return: The list of file descriptors.
|
@return: The list of file descriptors.
|
||||||
@rtype: List of integers
|
@rtype: List of integers
|
||||||
@ -233,9 +239,10 @@ class SBusDatagram(object):
|
|||||||
return self.h_files_
|
return self.h_files_
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def set_files(self, h_files):
|
def set_files(self, h_files):
|
||||||
'''
|
'''@summary: Setter.
|
||||||
@summary: Setter.
|
|
||||||
Assign file handlers list and update n_files_ field
|
Assign file handlers list and update n_files_ field
|
||||||
|
|
||||||
@param h_files: File descriptors.
|
@param h_files: File descriptors.
|
||||||
@ -257,9 +264,10 @@ class SBusDatagram(object):
|
|||||||
self.h_files_.append(h_files[i])
|
self.h_files_.append(h_files[i])
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def get_first_file_of_type(self, file_type):
|
def get_first_file_of_type(self, file_type):
|
||||||
'''
|
'''@summary: Iterate through file list and metadata.
|
||||||
@summary: Iterate through file list and metadata.
|
|
||||||
Find the first file with the required type
|
Find the first file with the required type
|
||||||
|
|
||||||
@param file_type: The file type to look for
|
@param file_type: The file type to look for
|
||||||
@ -273,15 +281,15 @@ class SBusDatagram(object):
|
|||||||
if (self.get_metadata()[i])['type'] == file_type:
|
if (self.get_metadata()[i])['type'] == file_type:
|
||||||
try:
|
try:
|
||||||
required_file = os.fdopen(self.get_files()[i], 'w')
|
required_file = os.fdopen(self.get_files()[i], 'w')
|
||||||
except IOError, err:
|
except IOError as err:
|
||||||
syslog.syslog(syslog.LOG_DEBUG,
|
syslog.syslog(syslog.LOG_DEBUG,
|
||||||
'Failed to open file: %s' % err.strerror)
|
'Failed to open file: %s' % err.strerror)
|
||||||
return required_file
|
return required_file
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def get_metadata(self):
|
def get_metadata(self):
|
||||||
'''
|
'''@summary: Getter.
|
||||||
@summary: Getter.
|
|
||||||
|
|
||||||
@return: The list of meta-data dictionaries.
|
@return: The list of meta-data dictionaries.
|
||||||
@rtype: List of dictionaries
|
@rtype: List of dictionaries
|
||||||
@ -289,9 +297,10 @@ class SBusDatagram(object):
|
|||||||
return self.files_metadata_
|
return self.files_metadata_
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def set_metadata(self, metadata):
|
def set_metadata(self, metadata):
|
||||||
'''
|
'''@summary: Setter.
|
||||||
@summary: Setter.
|
|
||||||
Assign file_metadata_ field
|
Assign file_metadata_ field
|
||||||
|
|
||||||
@param metadata: File descriptors meta-data dictionaries.
|
@param metadata: File descriptors meta-data dictionaries.
|
||||||
@ -302,9 +311,9 @@ class SBusDatagram(object):
|
|||||||
self.files_metadata_ = metadata
|
self.files_metadata_ = metadata
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def get_exec_params(self):
|
def get_exec_params(self):
|
||||||
'''
|
'''@summary: Getter.
|
||||||
@summary: Getter.
|
|
||||||
|
|
||||||
@return: The execution parameters dictionary.
|
@return: The execution parameters dictionary.
|
||||||
@rtype: Dictionary
|
@rtype: Dictionary
|
||||||
@ -312,9 +321,10 @@ class SBusDatagram(object):
|
|||||||
return self.exec_params_
|
return self.exec_params_
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def set_exec_params(self, params):
|
def set_exec_params(self, params):
|
||||||
'''
|
'''@summary: Setter.
|
||||||
@summary: Setter.
|
|
||||||
Assign execution parameters dictionary.
|
Assign execution parameters dictionary.
|
||||||
|
|
||||||
@param params: Execution parameters to assign
|
@param params: Execution parameters to assign
|
||||||
@ -326,9 +336,10 @@ class SBusDatagram(object):
|
|||||||
self.exec_params_ = params
|
self.exec_params_ = params
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def add_exec_param(self, param_name, param_value):
|
def add_exec_param(self, param_name, param_value):
|
||||||
'''
|
'''@summary: Add a single pair to the exec_params_ dictionary
|
||||||
@summary: Add a single pair to the exec_params_ dictionary
|
|
||||||
Don't change if the parameter exists already
|
Don't change if the parameter exists already
|
||||||
|
|
||||||
@param param_name: Execution parameter name to be added
|
@param param_name: Execution parameter name to be added
|
||||||
@ -351,9 +362,9 @@ class SBusDatagram(object):
|
|||||||
return b_status
|
return b_status
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def get_command(self):
|
def get_command(self):
|
||||||
'''
|
'''@summary: Getter.
|
||||||
@summary: Getter.
|
|
||||||
|
|
||||||
@return: The Storlet Daemon command.
|
@return: The Storlet Daemon command.
|
||||||
@rtype: SBusStorletCommand
|
@rtype: SBusStorletCommand
|
||||||
@ -361,9 +372,10 @@ class SBusDatagram(object):
|
|||||||
return self.e_command_
|
return self.e_command_
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def set_command(self, cmd):
|
def set_command(self, cmd):
|
||||||
'''
|
'''@summary: Setter.
|
||||||
@summary: Setter.
|
|
||||||
Assign Storlet Daemon command.
|
Assign Storlet Daemon command.
|
||||||
|
|
||||||
@param cmd: Command to assign
|
@param cmd: Command to assign
|
||||||
@ -374,9 +386,9 @@ class SBusDatagram(object):
|
|||||||
self.e_command_ = cmd
|
self.e_command_ = cmd
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def get_task_id(self):
|
def get_task_id(self):
|
||||||
'''
|
'''@summary: Getter.
|
||||||
@summary: Getter.
|
|
||||||
|
|
||||||
@return: The task id.
|
@return: The task id.
|
||||||
@rtype: string
|
@rtype: string
|
||||||
@ -384,9 +396,10 @@ class SBusDatagram(object):
|
|||||||
return self.task_id_
|
return self.task_id_
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def set_task_id(self, taskId):
|
def set_task_id(self, taskId):
|
||||||
'''
|
'''@summary: Setter.
|
||||||
@summary: Setter.
|
|
||||||
Assign task id
|
Assign task id
|
||||||
|
|
||||||
@param taskId: Command to assign
|
@param taskId: Command to assign
|
||||||
@ -397,10 +410,11 @@ class SBusDatagram(object):
|
|||||||
self.task_id_ = taskId
|
self.task_id_ = taskId
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def dictionaies_equal(d1, d2):
|
def dictionaies_equal(d1, d2):
|
||||||
'''
|
'''@summary: Check whether two dictionaries has the same content.
|
||||||
@summary: Check whether two dictionaries has the same content.
|
|
||||||
The order of the entries is not considered.
|
The order of the entries is not considered.
|
||||||
|
|
||||||
@return: The answer to the above
|
@return: The answer to the above
|
||||||
|
@ -17,7 +17,6 @@ Limitations under the License.
|
|||||||
21-Jul-2014 evgenyl Initial implementation.
|
21-Jul-2014 evgenyl Initial implementation.
|
||||||
==========================================================================='''
|
==========================================================================='''
|
||||||
|
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
'''
|
'''
|
||||||
@summary: Enumerate file usage intents.
|
@summary: Enumerate file usage intents.
|
||||||
@ -25,12 +24,12 @@ Limitations under the License.
|
|||||||
with its Java counterpart.
|
with its Java counterpart.
|
||||||
'''
|
'''
|
||||||
|
|
||||||
SBUS_FD_INPUT_OBJECT = 0
|
SBUS_FD_INPUT_OBJECT = 0
|
||||||
SBUS_FD_OUTPUT_OBJECT = 1
|
SBUS_FD_OUTPUT_OBJECT = 1
|
||||||
SBUS_FD_OUTPUT_OBJECT_METADATA = 2
|
SBUS_FD_OUTPUT_OBJECT_METADATA = 2
|
||||||
SBUS_FD_OUTPUT_OBJECT_AND_METADATA = 3
|
SBUS_FD_OUTPUT_OBJECT_AND_METADATA = 3
|
||||||
SBUS_FD_LOGGER = 4
|
SBUS_FD_LOGGER = 4
|
||||||
SBUS_FD_OUTPUT_CONTAINER = 5
|
SBUS_FD_OUTPUT_CONTAINER = 5
|
||||||
SBUS_FD_OUTPUT_TASK_ID = 6
|
SBUS_FD_OUTPUT_TASK_ID = 6
|
||||||
|
|
||||||
'''============================ END OF FILE ==============================='''
|
'''============================ END OF FILE ==============================='''
|
||||||
|
@ -17,7 +17,6 @@ Limitations under the License.
|
|||||||
21-Jul-2014 evgenyl Initial implementation.
|
21-Jul-2014 evgenyl Initial implementation.
|
||||||
==========================================================================='''
|
==========================================================================='''
|
||||||
|
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
'''
|
'''
|
||||||
@summary: Enumerate Storlet Daemon commands.
|
@summary: Enumerate Storlet Daemon commands.
|
||||||
@ -25,15 +24,15 @@ Limitations under the License.
|
|||||||
with its Java counterpart.
|
with its Java counterpart.
|
||||||
'''
|
'''
|
||||||
|
|
||||||
SBUS_CMD_HALT = 0
|
SBUS_CMD_HALT = 0
|
||||||
SBUS_CMD_EXECUTE = 1
|
SBUS_CMD_EXECUTE = 1
|
||||||
SBUS_CMD_START_DAEMON = 2
|
SBUS_CMD_START_DAEMON = 2
|
||||||
SBUS_CMD_STOP_DAEMON = 3
|
SBUS_CMD_STOP_DAEMON = 3
|
||||||
SBUS_CMD_DAEMON_STATUS = 4
|
SBUS_CMD_DAEMON_STATUS = 4
|
||||||
SBUS_CMD_STOP_DAEMONS = 5
|
SBUS_CMD_STOP_DAEMONS = 5
|
||||||
SBUS_CMD_PING = 6
|
SBUS_CMD_PING = 6
|
||||||
SBUS_CMD_DESCRIPTOR = 7
|
SBUS_CMD_DESCRIPTOR = 7
|
||||||
SBUS_CMD_CANCEL = 8
|
SBUS_CMD_CANCEL = 8
|
||||||
SBUS_CMD_NOP = 9
|
SBUS_CMD_NOP = 9
|
||||||
|
|
||||||
'''============================ END OF FILE ==============================='''
|
'''============================ END OF FILE ==============================='''
|
||||||
|
@ -1,5 +1,4 @@
|
|||||||
#!/usr/bin/python
|
#!/usr/bin/python
|
||||||
#-----------------------------------------------------------------------------------------------
|
|
||||||
'''-------------------------------------------------------------------------
|
'''-------------------------------------------------------------------------
|
||||||
Copyright IBM Corp. 2015, 2015 All Rights Reserved
|
Copyright IBM Corp. 2015, 2015 All Rights Reserved
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
@ -16,8 +15,7 @@ Limitations under the License.
|
|||||||
-------------------------------------------------------------------------'''
|
-------------------------------------------------------------------------'''
|
||||||
|
|
||||||
from setuptools import setup
|
from setuptools import setup
|
||||||
setup( name = 'SBusPythonFacade',
|
setup(name='SBusPythonFacade',
|
||||||
version = '1.0',
|
version='1.0',
|
||||||
package_dir={'SBusPythonFacade':''},
|
package_dir={'SBusPythonFacade': ''},
|
||||||
packages=['SBusPythonFacade'] )
|
packages=['SBusPythonFacade'])
|
||||||
|
|
||||||
|
@ -1,54 +1,58 @@
|
|||||||
#!/usr/bin/python
|
'''-------------------------------------------------------------------------
|
||||||
#-----------------------------------------------------------------------------------------------
|
Copyright IBM Corp. 2015, 2015 All Rights Reserved
|
||||||
# Copyright IBM Corp. 2015, 2015 All Rights Reserved
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
you may not use this file except in compliance with the License.
|
||||||
# you may not use this file except in compliance with the License.
|
You may obtain a copy of the License at
|
||||||
# You may obtain a copy of the License at
|
|
||||||
#
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
# http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
#
|
Unless required by applicable law or agreed to in writing, software
|
||||||
# Unless required by applicable law or agreed to in writing, software
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
See the License for the specific language governing permissions and
|
||||||
# See the License for the specific language governing permissions and
|
Limitations under the License.
|
||||||
# Limitations under the License.
|
-------------------------------------------------------------------------'''
|
||||||
#-----------------------------------------------------------------------------------------------
|
|
||||||
|
|
||||||
'''===========================================================================
|
'''===========================================================================
|
||||||
02-Dec-2014 evgenyl Initial implementation.
|
02-Dec-2014 evgenyl Initial implementation.
|
||||||
==========================================================================='''
|
==========================================================================='''
|
||||||
|
|
||||||
import sys
|
|
||||||
import os
|
import os
|
||||||
|
import sys
|
||||||
|
|
||||||
from SBusPythonFacade.SBus import SBus
|
from SBusPythonFacade.SBus import SBus
|
||||||
from SBusPythonFacade.SBusDatagram import SBusDatagram
|
from SBusPythonFacade.SBusDatagram import SBusDatagram
|
||||||
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_HALT
|
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_HALT
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def print_usage(argv):
|
def print_usage(argv):
|
||||||
print argv[0] + ' /path/to/daemon/factory_pipe'
|
print(argv[0] + ' /path/to/daemon/factory_pipe')
|
||||||
print 'Example:'
|
print('Example:')
|
||||||
print argv[0] + ' ',
|
sys.stdout.write(argv[0] + ' ')
|
||||||
print '/home/lxc_device/pipes/scopes/'\
|
print('/home/lxc_device/pipes/scopes/'
|
||||||
'AUTH_fb8b63c579054c48816ca8acd090b3d9/factory_pipe'
|
'AUTH_fb8b63c579054c48816ca8acd090b3d9/factory_pipe')
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def main(argv):
|
def main(argv):
|
||||||
if 2 > len(argv):
|
if 2 > len(argv):
|
||||||
print_usage(argv)
|
print_usage(argv)
|
||||||
return
|
return
|
||||||
|
|
||||||
daemon_factory_pipe_name = argv[1]
|
daemon_factory_pipe_name = argv[1]
|
||||||
fi,fo = os.pipe()
|
fi, fo = os.pipe()
|
||||||
halt_dtg = SBusDatagram.create_service_datagram(SBUS_CMD_HALT, fo)
|
halt_dtg = SBusDatagram.create_service_datagram(SBUS_CMD_HALT, fo)
|
||||||
n_status = SBus.send(daemon_factory_pipe_name, halt_dtg)
|
n_status = SBus.send(daemon_factory_pipe_name, halt_dtg)
|
||||||
if 0 > n_status:
|
if 0 > n_status:
|
||||||
print 'Sending failed'
|
print('Sending failed')
|
||||||
else:
|
else:
|
||||||
print 'Sending succeeded'
|
print('Sending succeeded')
|
||||||
cmd_response = os.read( fi, 256 )
|
cmd_response = os.read(fi, 256)
|
||||||
print cmd_response
|
print(cmd_response)
|
||||||
os.close(fi)
|
os.close(fi)
|
||||||
os.close(fo)
|
os.close(fo)
|
||||||
|
|
||||||
|
@ -20,38 +20,42 @@ XX-XXX-2014 eranr Initial implementation.
|
|||||||
01-Dec-2014 evgenyl Dropping multi-threaded monitoring
|
01-Dec-2014 evgenyl Dropping multi-threaded monitoring
|
||||||
==========================================================================='''
|
==========================================================================='''
|
||||||
|
|
||||||
|
import errno
|
||||||
|
import logging
|
||||||
|
from logging.handlers import SysLogHandler
|
||||||
import os
|
import os
|
||||||
import pwd
|
import pwd
|
||||||
|
import signal
|
||||||
|
import subprocess
|
||||||
import sys
|
import sys
|
||||||
import time
|
import time
|
||||||
import errno
|
|
||||||
import signal
|
|
||||||
import logging
|
|
||||||
import subprocess
|
|
||||||
from logging.handlers import SysLogHandler
|
|
||||||
|
|
||||||
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_START_DAEMON,\
|
|
||||||
SBUS_CMD_STOP_DAEMON, SBUS_CMD_DAEMON_STATUS, SBUS_CMD_STOP_DAEMONS,\
|
|
||||||
SBUS_CMD_PING, SBUS_CMD_HALT
|
|
||||||
from SBusPythonFacade.SBusFileDescription import SBUS_FD_OUTPUT_OBJECT
|
|
||||||
from SBusPythonFacade.SBus import SBus
|
from SBusPythonFacade.SBus import SBus
|
||||||
from SBusPythonFacade.SBusDatagram import *
|
from SBusPythonFacade.SBusDatagram import SBusDatagram
|
||||||
|
from SBusPythonFacade.SBusFileDescription import SBUS_FD_OUTPUT_OBJECT
|
||||||
|
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_DAEMON_STATUS
|
||||||
|
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_HALT
|
||||||
|
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_PING
|
||||||
|
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_START_DAEMON
|
||||||
|
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_STOP_DAEMON
|
||||||
|
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_STOP_DAEMONS
|
||||||
|
|
||||||
'''========================================================================'''
|
'''========================================================================'''
|
||||||
|
|
||||||
|
|
||||||
class daemon_factory():
|
class daemon_factory(object):
|
||||||
'''
|
'''@summary: This class acts as the manager for storlet daemons.
|
||||||
@summary: This class acts as the manager for storlet daemons.
|
|
||||||
It listens to commands and reacts on them in an internal loop.
|
It listens to commands and reacts on them in an internal loop.
|
||||||
As for now (01-Dec-2014) it is a single thread, synchronous
|
As for now (01-Dec-2014) it is a single thread, synchronous
|
||||||
processing.
|
processing.
|
||||||
'''
|
'''
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def __init__(self, path, logger):
|
def __init__(self, path, logger):
|
||||||
'''
|
'''@summary: CTOR
|
||||||
@summary: CTOR
|
|
||||||
Prepare the auxiliary data structures
|
Prepare the auxiliary data structures
|
||||||
|
|
||||||
@param path: Path to the pipe file internal SBus listens to
|
@param path: Path to the pipe file internal SBus listens to
|
||||||
@ -69,6 +73,7 @@ class daemon_factory():
|
|||||||
self.NUM_OF_TRIES_PINGING_STARTING_DAEMON = 5
|
self.NUM_OF_TRIES_PINGING_STARTING_DAEMON = 5
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def get_jvm_args(self,
|
def get_jvm_args(self,
|
||||||
daemon_language,
|
daemon_language,
|
||||||
storlet_path,
|
storlet_path,
|
||||||
@ -77,8 +82,8 @@ class daemon_factory():
|
|||||||
uds_path,
|
uds_path,
|
||||||
log_level,
|
log_level,
|
||||||
container_id):
|
container_id):
|
||||||
'''
|
'''@summary: get_jvm_args
|
||||||
@summary: get_jvm_args
|
|
||||||
Check the input parameters, produce the list
|
Check the input parameters, produce the list
|
||||||
of arguments for JVM process launch
|
of arguments for JVM process launch
|
||||||
|
|
||||||
@ -136,8 +141,8 @@ class daemon_factory():
|
|||||||
pargs = []
|
pargs = []
|
||||||
if daemon_language == "java":
|
if daemon_language == "java":
|
||||||
self.logger.debug('START_DAEMON:preparing arguments')
|
self.logger.debug('START_DAEMON:preparing arguments')
|
||||||
#Setting two environmental variables
|
# Setting two environmental variables
|
||||||
#The path strings are corrupted if passed is pargs list below
|
# The path strings are corrupted if passed is pargs list below
|
||||||
os.environ['CLASSPATH'] = str_dmn_clspth
|
os.environ['CLASSPATH'] = str_dmn_clspth
|
||||||
os.environ['LD_LIBRARY_PATH'] = str_library_path
|
os.environ['LD_LIBRARY_PATH'] = str_library_path
|
||||||
pargs = [str('/usr/bin/java'),
|
pargs = [str('/usr/bin/java'),
|
||||||
@ -157,9 +162,10 @@ class daemon_factory():
|
|||||||
return n_error_id, error_text, pargs
|
return n_error_id, error_text, pargs
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def spawn_subprocess(self, pargs):
|
def spawn_subprocess(self, pargs):
|
||||||
'''
|
'''@summary: spawn_subprocess
|
||||||
@summary: spawn_subprocess
|
|
||||||
Launch a JVM process for some storlet daemon
|
Launch a JVM process for some storlet daemon
|
||||||
|
|
||||||
@param pargs: Arguments for the JVM
|
@param pargs: Arguments for the JVM
|
||||||
@ -192,8 +198,8 @@ class daemon_factory():
|
|||||||
format(storlet_name, jvm_pid))
|
format(storlet_name, jvm_pid))
|
||||||
# Keep JVM PID
|
# Keep JVM PID
|
||||||
self.storlet_name_to_pid[storlet_name] = jvm_pid
|
self.storlet_name_to_pid[storlet_name] = jvm_pid
|
||||||
b_status, error_text = self.wait_for_daemon_to_initialize(
|
b_status, error_text = \
|
||||||
storlet_name)
|
self.wait_for_daemon_to_initialize(storlet_name)
|
||||||
if not b_status:
|
if not b_status:
|
||||||
raise 'No response from Daemon'
|
raise 'No response from Daemon'
|
||||||
self.logger.debug('START_DAEMON: just occurred')
|
self.logger.debug('START_DAEMON: just occurred')
|
||||||
@ -207,9 +213,10 @@ class daemon_factory():
|
|||||||
return b_status, error_text
|
return b_status, error_text
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def wait_for_daemon_to_initialize(self, storlet_name):
|
def wait_for_daemon_to_initialize(self, storlet_name):
|
||||||
'''
|
'''@summary: wait_for_daemon_to_initialize
|
||||||
@summary: wait_for_daemon_to_initialize
|
|
||||||
Send a Ping service datagram. Validate that
|
Send a Ping service datagram. Validate that
|
||||||
Daemon response is correct. Give up after the
|
Daemon response is correct. Give up after the
|
||||||
predefined number of attempts (5)
|
predefined number of attempts (5)
|
||||||
@ -223,8 +230,8 @@ class daemon_factory():
|
|||||||
@rtype: String
|
@rtype: String
|
||||||
'''
|
'''
|
||||||
storlet_pipe_name = self.storlet_name_to_pipe_name[storlet_name]
|
storlet_pipe_name = self.storlet_name_to_pipe_name[storlet_name]
|
||||||
self.logger.debug('Send PING command to {0} via {1}'.\
|
self.logger.debug('Send PING command to {0} via {1}'.
|
||||||
format(storlet_name,storlet_pipe_name))
|
format(storlet_name, storlet_pipe_name))
|
||||||
read_fd, write_fd = os.pipe()
|
read_fd, write_fd = os.pipe()
|
||||||
dtg = SBusDatagram.create_service_datagram(SBUS_CMD_PING, write_fd)
|
dtg = SBusDatagram.create_service_datagram(SBUS_CMD_PING, write_fd)
|
||||||
b_status = False
|
b_status = False
|
||||||
@ -243,6 +250,7 @@ class daemon_factory():
|
|||||||
return b_status, error_text
|
return b_status, error_text
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def process_start_daemon(self,
|
def process_start_daemon(self,
|
||||||
daemon_language,
|
daemon_language,
|
||||||
storlet_path,
|
storlet_path,
|
||||||
@ -251,8 +259,8 @@ class daemon_factory():
|
|||||||
uds_path,
|
uds_path,
|
||||||
log_level,
|
log_level,
|
||||||
container_id):
|
container_id):
|
||||||
'''
|
'''@summary: process_start_daemon
|
||||||
@summary: process_start_daemon
|
|
||||||
Start storlet daemon process
|
Start storlet daemon process
|
||||||
|
|
||||||
@see: get_jvm_args for the list of parameters
|
@see: get_jvm_args for the list of parameters
|
||||||
@ -294,7 +302,7 @@ class daemon_factory():
|
|||||||
error_text = '{0} is already running'.format(storlet_name)
|
error_text = '{0} is already running'.format(storlet_name)
|
||||||
self.logger.debug(error_text)
|
self.logger.debug(error_text)
|
||||||
else:
|
else:
|
||||||
error_text = '{0} is not running. About to spawn process'.\
|
error_text = '{0} is not running. About to spawn process'. \
|
||||||
format(storlet_name)
|
format(storlet_name)
|
||||||
self.logger.debug(error_text)
|
self.logger.debug(error_text)
|
||||||
b_status, error_text = self.spawn_subprocess(pargs)
|
b_status, error_text = self.spawn_subprocess(pargs)
|
||||||
@ -302,9 +310,10 @@ class daemon_factory():
|
|||||||
return b_status, error_text
|
return b_status, error_text
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def get_process_status_by_name(self, storlet_name):
|
def get_process_status_by_name(self, storlet_name):
|
||||||
'''
|
'''@summary: get_process_status_by_name
|
||||||
@summary: get_process_status_by_name
|
|
||||||
Check if the daemon runs for the specific storlet
|
Check if the daemon runs for the specific storlet
|
||||||
|
|
||||||
@param storlet_name: Storlet name we are checking the daemon for
|
@param storlet_name: Storlet name we are checking the daemon for
|
||||||
@ -327,16 +336,17 @@ class daemon_factory():
|
|||||||
b_status, error_text = self.get_process_status_by_pid(
|
b_status, error_text = self.get_process_status_by_pid(
|
||||||
daemon_pid, storlet_name)
|
daemon_pid, storlet_name)
|
||||||
else:
|
else:
|
||||||
error_text = 'Storlet name {0} not found in map'.\
|
error_text = 'Storlet name {0} not found in map'. \
|
||||||
format(storlet_name)
|
format(storlet_name)
|
||||||
self.logger.debug(error_text)
|
self.logger.debug(error_text)
|
||||||
|
|
||||||
return b_status, error_text
|
return b_status, error_text
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def get_process_status_by_pid(self, daemon_pid, storlet_name):
|
def get_process_status_by_pid(self, daemon_pid, storlet_name):
|
||||||
'''
|
'''@summary: get_process_status_by_pid
|
||||||
@summary: get_process_status_by_pid
|
|
||||||
Check if a process with specific ID runs
|
Check if a process with specific ID runs
|
||||||
|
|
||||||
@param daemon_pid: Storlet daemon process ID
|
@param daemon_pid: Storlet daemon process ID
|
||||||
@ -355,14 +365,14 @@ class daemon_factory():
|
|||||||
obtained_code = 0
|
obtained_code = 0
|
||||||
try:
|
try:
|
||||||
obtained_pid, obtained_code = os.waitpid(daemon_pid, os.WNOHANG)
|
obtained_pid, obtained_code = os.waitpid(daemon_pid, os.WNOHANG)
|
||||||
error_text = 'Storlet {0}, PID = {1}, ErrCode = {2}'.\
|
error_text = 'Storlet {0}, PID = {1}, ErrCode = {2}'. \
|
||||||
format(storlet_name, obtained_pid, obtained_code)
|
format(storlet_name, obtained_pid, obtained_code)
|
||||||
self.logger.debug(error_text)
|
self.logger.debug(error_text)
|
||||||
except OSError, err:
|
except OSError as err:
|
||||||
if err.errno == errno.ESRCH:
|
if err.errno == errno.ESRCH:
|
||||||
error_text = 'No running daemon for {0}'.format(storlet_name)
|
error_text = 'No running daemon for {0}'.format(storlet_name)
|
||||||
elif err.errno == errno.EPERM:
|
elif err.errno == errno.EPERM:
|
||||||
error_text = 'No permission to access daemon for {0}'.\
|
error_text = 'No permission to access daemon for {0}'. \
|
||||||
format(storlet_name)
|
format(storlet_name)
|
||||||
else:
|
else:
|
||||||
error_text = 'Unknown error'
|
error_text = 'Unknown error'
|
||||||
@ -376,9 +386,10 @@ class daemon_factory():
|
|||||||
return b_status, error_text
|
return b_status, error_text
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def process_kill(self, storlet_name):
|
def process_kill(self, storlet_name):
|
||||||
'''
|
'''@summary: process_kill
|
||||||
@summary: process_kill
|
|
||||||
Kill the storlet daemon immediately
|
Kill the storlet daemon immediately
|
||||||
(kill -9 $DMN_PID)
|
(kill -9 $DMN_PID)
|
||||||
|
|
||||||
@ -399,10 +410,10 @@ class daemon_factory():
|
|||||||
try:
|
try:
|
||||||
os.kill(dmn_pid, signal.SIGKILL)
|
os.kill(dmn_pid, signal.SIGKILL)
|
||||||
obtained_pid, obtained_code = os.waitpid(dmn_pid, os.WNOHANG)
|
obtained_pid, obtained_code = os.waitpid(dmn_pid, os.WNOHANG)
|
||||||
error_text = 'Storlet {0}, PID = {1}, ErrCode = {2}'.\
|
error_text = 'Storlet {0}, PID = {1}, ErrCode = {2}'. \
|
||||||
format(storlet_name, obtained_pid, obtained_code)
|
format(storlet_name, obtained_pid, obtained_code)
|
||||||
self.logger.debug(error_text)
|
self.logger.debug(error_text)
|
||||||
except:
|
except Exception:
|
||||||
self.logger.debug('Crash while killing storlet')
|
self.logger.debug('Crash while killing storlet')
|
||||||
self.storlet_name_to_pid.pop(storlet_name)
|
self.storlet_name_to_pid.pop(storlet_name)
|
||||||
else:
|
else:
|
||||||
@ -412,10 +423,11 @@ class daemon_factory():
|
|||||||
return b_success, error_text
|
return b_success, error_text
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def process_kill_all(self):
|
def process_kill_all(self):
|
||||||
'''
|
'''@summary: process_kill_all Iterate through storlet daemons.
|
||||||
@summary: process_kill_all
|
|
||||||
Iterate through storlet daemons. Kill every one.
|
Kill every one.
|
||||||
|
|
||||||
@return: Status (True)
|
@return: Status (True)
|
||||||
@rtype: Boolean
|
@rtype: Boolean
|
||||||
@ -427,9 +439,10 @@ class daemon_factory():
|
|||||||
return True, 'OK'
|
return True, 'OK'
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def shutdown_all_processes(self):
|
def shutdown_all_processes(self):
|
||||||
'''
|
'''@summary: shutdown_all_processes
|
||||||
@summary: shutdown_all_processes
|
|
||||||
send HALT command to every spawned process
|
send HALT command to every spawned process
|
||||||
'''
|
'''
|
||||||
answer = ''
|
answer = ''
|
||||||
@ -441,9 +454,9 @@ class daemon_factory():
|
|||||||
return True, answer
|
return True, answer
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def shutdown_process(self, storlet_name):
|
def shutdown_process(self, storlet_name):
|
||||||
'''
|
'''@summary: send HALT command to storlet daemon
|
||||||
@summary: send HALT command to storlet daemon
|
|
||||||
|
|
||||||
@param storlet_name: Storlet name we are checking the daemon for
|
@param storlet_name: Storlet name we are checking the daemon for
|
||||||
@type storlet_name: String
|
@type storlet_name: String
|
||||||
@ -452,31 +465,31 @@ class daemon_factory():
|
|||||||
@rtype: Boolean
|
@rtype: Boolean
|
||||||
@return: Description text of possible error
|
@return: Description text of possible error
|
||||||
@rtype: String
|
@rtype: String
|
||||||
'''
|
'''
|
||||||
|
|
||||||
b_status = False
|
b_status = False
|
||||||
error_text = ''
|
|
||||||
self.logger.debug('Inside shutdown_process {0}'.format(storlet_name))
|
self.logger.debug('Inside shutdown_process {0}'.format(storlet_name))
|
||||||
storlet_pipe_name = self.storlet_name_to_pipe_name[storlet_name]
|
storlet_pipe_name = self.storlet_name_to_pipe_name[storlet_name]
|
||||||
self.logger.debug('Send HALT command to {0} via {1}'.\
|
self.logger.debug('Send HALT command to {0} via {1}'.
|
||||||
format(storlet_name,storlet_pipe_name))
|
format(storlet_name, storlet_pipe_name))
|
||||||
read_fd, write_fd = os.pipe()
|
read_fd, write_fd = os.pipe()
|
||||||
dtg = SBusDatagram.create_service_datagram(SBUS_CMD_HALT, write_fd)
|
dtg = SBusDatagram.create_service_datagram(SBUS_CMD_HALT, write_fd)
|
||||||
SBus.send(storlet_pipe_name, dtg)
|
SBus.send(storlet_pipe_name, dtg)
|
||||||
os.close(read_fd)
|
os.close(read_fd)
|
||||||
os.close(write_fd)
|
os.close(write_fd)
|
||||||
dmn_pid = self.storlet_name_to_pid.get(storlet_name, -1)
|
dmn_pid = self.storlet_name_to_pid.get(storlet_name, -1)
|
||||||
self.logger.debug('Storlet Daemon PID is {0}'.\
|
self.logger.debug('Storlet Daemon PID is {0}'.format(dmn_pid))
|
||||||
format(dmn_pid))
|
|
||||||
if -1 != dmn_pid:
|
if -1 != dmn_pid:
|
||||||
os.waitpid(dmn_pid,0)
|
os.waitpid(dmn_pid, 0)
|
||||||
self.storlet_name_to_pid.pop(storlet_name)
|
self.storlet_name_to_pid.pop(storlet_name)
|
||||||
b_status = True
|
b_status = True
|
||||||
return b_status
|
return b_status
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def dispatch_command(self, dtg, container_id):
|
def dispatch_command(self, dtg, container_id):
|
||||||
'''
|
'''@summary: dispatch_command
|
||||||
@summary: dispatch_command
|
|
||||||
Parse datagram. React on the request.
|
Parse datagram. React on the request.
|
||||||
|
|
||||||
@param dtg: Datagram to process
|
@param dtg: Datagram to process
|
||||||
@ -490,7 +503,8 @@ class daemon_factory():
|
|||||||
@rtype: String
|
@rtype: String
|
||||||
@return: Flag - whether we need to continue operating
|
@return: Flag - whether we need to continue operating
|
||||||
@rtype: Boolean
|
@rtype: Boolean
|
||||||
'''
|
'''
|
||||||
|
|
||||||
b_status = False
|
b_status = False
|
||||||
error_text = ''
|
error_text = ''
|
||||||
b_iterate = True
|
b_iterate = True
|
||||||
@ -498,17 +512,17 @@ class daemon_factory():
|
|||||||
try:
|
try:
|
||||||
command = dtg.get_command()
|
command = dtg.get_command()
|
||||||
except Exception:
|
except Exception:
|
||||||
error_text = "Received message does not have command"\
|
error_text = "Received message does not have command" \
|
||||||
" identifier. continuing."
|
" identifier. continuing."
|
||||||
b_status = False
|
b_status = False
|
||||||
self.logger.error( error_text )
|
self.logger.error(error_text)
|
||||||
else:
|
else:
|
||||||
self.logger.debug("Received command {0}".format(command))
|
self.logger.debug("Received command {0}".format(command))
|
||||||
|
|
||||||
prms = dtg.get_exec_params()
|
prms = dtg.get_exec_params()
|
||||||
if command == SBUS_CMD_START_DAEMON:
|
if command == SBUS_CMD_START_DAEMON:
|
||||||
self.logger.debug( 'Do SBUS_CMD_START_DAEMON' )
|
self.logger.debug('Do SBUS_CMD_START_DAEMON')
|
||||||
self.logger.debug( 'prms = %s'%str(prms) )
|
self.logger.debug('prms = %s' % str(prms))
|
||||||
b_status, error_text = \
|
b_status, error_text = \
|
||||||
self.process_start_daemon(prms['daemon_language'],
|
self.process_start_daemon(prms['daemon_language'],
|
||||||
prms['storlet_path'],
|
prms['storlet_path'],
|
||||||
@ -518,43 +532,45 @@ class daemon_factory():
|
|||||||
prms['log_level'],
|
prms['log_level'],
|
||||||
container_id)
|
container_id)
|
||||||
elif command == SBUS_CMD_STOP_DAEMON:
|
elif command == SBUS_CMD_STOP_DAEMON:
|
||||||
self.logger.debug( 'Do SBUS_CMD_STOP_DAEMON' )
|
self.logger.debug('Do SBUS_CMD_STOP_DAEMON')
|
||||||
b_status, error_text = self.process_kill(\
|
b_status, error_text = \
|
||||||
prms['storlet_name'])
|
self.process_kill(prms['storlet_name'])
|
||||||
elif command == SBUS_CMD_DAEMON_STATUS:
|
elif command == SBUS_CMD_DAEMON_STATUS:
|
||||||
self.logger.debug( 'Do SBUS_CMD_DAEMON_STATUS' )
|
self.logger.debug('Do SBUS_CMD_DAEMON_STATUS')
|
||||||
b_status, error_text = self.get_process_status_by_name(\
|
b_status, error_text = \
|
||||||
prms['storlet_name'])
|
self.get_process_status_by_name(prms['storlet_name'])
|
||||||
elif command == SBUS_CMD_STOP_DAEMONS:
|
elif command == SBUS_CMD_STOP_DAEMONS:
|
||||||
self.logger.debug( 'Do SBUS_CMD_STOP_DAEMONS' )
|
self.logger.debug('Do SBUS_CMD_STOP_DAEMONS')
|
||||||
b_status, error_text = self.process_kill_all()
|
b_status, error_text = self.process_kill_all()
|
||||||
b_iterate = False
|
b_iterate = False
|
||||||
elif command == SBUS_CMD_HALT:
|
elif command == SBUS_CMD_HALT:
|
||||||
self.logger.debug( 'Do SBUS_CMD_HALT' )
|
self.logger.debug('Do SBUS_CMD_HALT')
|
||||||
b_status, error_text = self.shutdown_all_processes()
|
b_status, error_text = self.shutdown_all_processes()
|
||||||
b_iterate = False
|
b_iterate = False
|
||||||
elif command == SBUS_CMD_PING:
|
elif command == SBUS_CMD_PING:
|
||||||
self.logger.debug( 'Do SBUS_CMD_PING' )
|
self.logger.debug('Do SBUS_CMD_PING')
|
||||||
b_status = True
|
b_status = True
|
||||||
error_text = 'OK'
|
error_text = 'OK'
|
||||||
else:
|
else:
|
||||||
b_status = False
|
b_status = False
|
||||||
error_text = "got unknown command %d" % command
|
error_text = "got unknown command %d" % command
|
||||||
self.logger.error( error_text )
|
self.logger.error(error_text)
|
||||||
|
|
||||||
self.logger.debug( 'Done' )
|
self.logger.debug('Done')
|
||||||
return b_status, error_text, b_iterate
|
return b_status, error_text, b_iterate
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def main_loop(self, container_id):
|
def main_loop(self, container_id):
|
||||||
'''
|
'''@summary: main_loop
|
||||||
@summary: main_loop
|
|
||||||
The 'internal' loop. Listen to SBus, receive datagram,
|
The 'internal' loop. Listen to SBus, receive datagram,
|
||||||
dispatch command, report back.
|
dispatch command, report back.
|
||||||
'''
|
'''
|
||||||
|
|
||||||
# Create SBus. Listen and process requests
|
# Create SBus. Listen and process requests
|
||||||
sbus = SBus()
|
sbus = SBus()
|
||||||
fd = sbus.create( self.pipe_path )
|
fd = sbus.create(self.pipe_path)
|
||||||
if fd < 0:
|
if fd < 0:
|
||||||
self.logger.error("Failed to create SBus. exiting.")
|
self.logger.error("Failed to create SBus. exiting.")
|
||||||
return
|
return
|
||||||
@ -576,26 +592,28 @@ class daemon_factory():
|
|||||||
return
|
return
|
||||||
|
|
||||||
try:
|
try:
|
||||||
outfd = dtg.get_first_file_of_type( SBUS_FD_OUTPUT_OBJECT )
|
outfd = dtg.get_first_file_of_type(SBUS_FD_OUTPUT_OBJECT)
|
||||||
except Exception:
|
except Exception:
|
||||||
self.logger.error("Received message does not have outfd."\
|
self.logger.error("Received message does not have outfd."
|
||||||
" continuing.")
|
" continuing.")
|
||||||
continue
|
continue
|
||||||
else:
|
else:
|
||||||
self.logger.debug("Received outfd %d" % outfd.fileno())
|
self.logger.debug("Received outfd %d" % outfd.fileno())
|
||||||
|
|
||||||
b_status, error_text, b_iterate = self.dispatch_command(dtg, container_id)
|
b_status, error_text, b_iterate = \
|
||||||
|
self.dispatch_command(dtg, container_id)
|
||||||
|
|
||||||
self.log_and_report(outfd, b_status, error_text)
|
self.log_and_report(outfd, b_status, error_text)
|
||||||
outfd.close()
|
outfd.close()
|
||||||
|
|
||||||
# We left the main loop for some reason. Terminating.
|
# We left the main loop for some reason. Terminating.
|
||||||
self.logger.debug( 'Leaving main loop' )
|
self.logger.debug('Leaving main loop')
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def log_and_report(self, outfd, b_status, error_text):
|
def log_and_report(self, outfd, b_status, error_text):
|
||||||
'''
|
'''@summary: log_and_report
|
||||||
@summary: log_and_report
|
|
||||||
Send the result description message
|
Send the result description message
|
||||||
back to swift middlewear
|
back to swift middlewear
|
||||||
|
|
||||||
@ -608,22 +626,23 @@ class daemon_factory():
|
|||||||
|
|
||||||
@rtype: void
|
@rtype: void
|
||||||
'''
|
'''
|
||||||
num = -1;
|
|
||||||
answer = str(b_status) + ': ' + error_text
|
answer = str(b_status) + ': ' + error_text
|
||||||
self.logger.debug(' Just processed command')
|
self.logger.debug(' Just processed command')
|
||||||
self.logger.debug(' Going to answer: %s'%answer)
|
self.logger.debug(' Going to answer: %s' % answer)
|
||||||
try:
|
try:
|
||||||
num = outfd.write( answer )
|
outfd.write(answer)
|
||||||
self.logger.debug(" ... and still alive")
|
self.logger.debug(" ... and still alive")
|
||||||
except:
|
except Exception:
|
||||||
self.logger.debug('Problem while writing response %s'%answer)
|
self.logger.debug('Problem while writing response %s' % answer)
|
||||||
|
|
||||||
'''======================= END OF daemon_factory CLASS ===================='''
|
'''======================= END OF daemon_factory CLASS ===================='''
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def start_logger(logger_name, log_level, container_id):
|
def start_logger(logger_name, log_level, container_id):
|
||||||
'''
|
'''@summary: start_logger
|
||||||
@summary: start_logger
|
|
||||||
Initialize logging of this process.
|
Initialize logging of this process.
|
||||||
Set the logger format.
|
Set the logger format.
|
||||||
|
|
||||||
@ -648,7 +667,6 @@ def start_logger(logger_name, log_level, container_id):
|
|||||||
else:
|
else:
|
||||||
level = logging.ERROR
|
level = logging.ERROR
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger("CONT #" + container_id + ": " + logger_name)
|
logger = logging.getLogger("CONT #" + container_id + ": " + logger_name)
|
||||||
|
|
||||||
if log_level == 'OFF':
|
if log_level == 'OFF':
|
||||||
@ -656,18 +674,18 @@ def start_logger(logger_name, log_level, container_id):
|
|||||||
else:
|
else:
|
||||||
logger.setLevel(level)
|
logger.setLevel(level)
|
||||||
|
|
||||||
for i in range(0,4):
|
for i in range(0, 4):
|
||||||
try:
|
try:
|
||||||
sysLogh = SysLogHandler('/dev/log')
|
sysLogh = SysLogHandler('/dev/log')
|
||||||
break
|
break
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
if i<3:
|
if i < 3:
|
||||||
time.sleep(1)
|
time.sleep(1)
|
||||||
else:
|
else:
|
||||||
raise e
|
raise e
|
||||||
|
|
||||||
str_format = '%(name)-12s: %(levelname)-8s %(funcName)s'+\
|
str_format = '%(name)-12s: %(levelname)-8s %(funcName)s' + \
|
||||||
' %(lineno)s [%(process)d, %(threadName)s]'+\
|
' %(lineno)s [%(process)d, %(threadName)s]' + \
|
||||||
' %(message)s'
|
' %(message)s'
|
||||||
formatter = logging.Formatter(str_format)
|
formatter = logging.Formatter(str_format)
|
||||||
sysLogh.setFormatter(formatter)
|
sysLogh.setFormatter(formatter)
|
||||||
@ -676,25 +694,30 @@ def start_logger(logger_name, log_level, container_id):
|
|||||||
return logger
|
return logger
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def usage():
|
def usage():
|
||||||
'''
|
'''@summary: usage
|
||||||
@summary: usage
|
|
||||||
Print the expected command line arguments.
|
Print the expected command line arguments.
|
||||||
|
|
||||||
@rtype: void
|
@rtype: void
|
||||||
'''
|
'''
|
||||||
print "daemon_factory <path> <log level> <container_id>"
|
print("daemon_factory <path> <log level> <container_id>")
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def main(argv):
|
def main(argv):
|
||||||
'''
|
'''@summary: main
|
||||||
@summary: main
|
|
||||||
The entry point.
|
The entry point.
|
||||||
- Initialize logger,
|
- Initialize logger,
|
||||||
- impersonate to swift user,
|
- impersonate to swift user,
|
||||||
- create an instance of daemon_factory,
|
- create an instance of daemon_factory,
|
||||||
- start the main loop.
|
- start the main loop.
|
||||||
'''
|
'''
|
||||||
|
|
||||||
if (len(argv) != 3):
|
if (len(argv) != 3):
|
||||||
usage()
|
usage()
|
||||||
return
|
return
|
||||||
@ -708,9 +731,8 @@ def main(argv):
|
|||||||
|
|
||||||
# Impersonate the swift user
|
# Impersonate the swift user
|
||||||
pw = pwd.getpwnam('swift')
|
pw = pwd.getpwnam('swift')
|
||||||
os.setresgid(pw.pw_gid,pw.pw_gid,pw.pw_gid)
|
os.setresgid(pw.pw_gid, pw.pw_gid, pw.pw_gid)
|
||||||
os.setresuid(pw.pw_uid,pw.pw_uid,pw.pw_uid)
|
os.setresuid(pw.pw_uid, pw.pw_uid, pw.pw_uid)
|
||||||
|
|
||||||
|
|
||||||
factory = daemon_factory(pipe_path, logger)
|
factory = daemon_factory(pipe_path, logger)
|
||||||
factory.main_loop(container_id)
|
factory.main_loop(container_id)
|
||||||
|
@ -14,8 +14,9 @@ See the License for the specific language governing permissions and
|
|||||||
Limitations under the License.
|
Limitations under the License.
|
||||||
-------------------------------------------------------------------------'''
|
-------------------------------------------------------------------------'''
|
||||||
|
|
||||||
from setuptools import setup, Extension
|
from setuptools import setup
|
||||||
setup(name = 'storlet_daemon_factory',
|
|
||||||
version = '1.0',
|
setup(name='storlet_daemon_factory',
|
||||||
package_dir={'storlet_daemon_factory':''},
|
version='1.0',
|
||||||
|
package_dir={'storlet_daemon_factory': ''},
|
||||||
packages=['storlet_daemon_factory'])
|
packages=['storlet_daemon_factory'])
|
||||||
|
@ -13,10 +13,11 @@ See the License for the specific language governing permissions and
|
|||||||
Limitations under the License.
|
Limitations under the License.
|
||||||
-------------------------------------------------------------------------'''
|
-------------------------------------------------------------------------'''
|
||||||
from setuptools import setup
|
from setuptools import setup
|
||||||
paste_factory = ['storlet_handler = storlet_middleware.storlet_handler:filter_factory']
|
paste_factory = ['storlet_handler = '
|
||||||
|
'storlet_middleware.storlet_handler:filter_factory']
|
||||||
|
|
||||||
setup(name='storlets',
|
setup(name='storlets',
|
||||||
version='1.0',
|
version='1.0',
|
||||||
packages=['storlet_middleware','storlet_gateway'],
|
packages=['storlet_middleware', 'storlet_gateway'],
|
||||||
entry_points={'paste.filter_factory': paste_factory}
|
entry_points={'paste.filter_factory': paste_factory}
|
||||||
)
|
)
|
||||||
|
@ -1,3 +0,0 @@
|
|||||||
|
|
||||||
|
|
||||||
|
|
@ -20,18 +20,20 @@ Created on Mar 24, 2015
|
|||||||
'''
|
'''
|
||||||
|
|
||||||
import os
|
import os
|
||||||
import sys
|
|
||||||
import shutil
|
|
||||||
import select
|
import select
|
||||||
from eventlet import Timeout
|
import shutil
|
||||||
|
|
||||||
|
from eventlet import Timeout
|
||||||
|
from storlet_middleware.storlet_common import StorletGatewayBase
|
||||||
|
from storlet_runtime import RunTimePaths
|
||||||
|
from storlet_runtime import RunTimeSandbox
|
||||||
|
from storlet_runtime import StorletInvocationGETProtocol
|
||||||
|
from storlet_runtime import StorletInvocationPUTProtocol
|
||||||
|
from storlet_runtime import StorletInvocationSLOProtocol
|
||||||
from swift.common.internal_client import InternalClient as ic
|
from swift.common.internal_client import InternalClient as ic
|
||||||
from swift.common.swob import Request
|
from swift.common.swob import Request
|
||||||
from storlet_runtime import RunTimeSandbox, RunTimePaths
|
|
||||||
from storlet_runtime import StorletInvocationGETProtocol,\
|
|
||||||
StorletInvocationPUTProtocol, StorletInvocationSLOProtocol
|
|
||||||
from swift.common.utils import config_true_value
|
from swift.common.utils import config_true_value
|
||||||
from storlet_middleware.storlet_common import StorletGatewayBase
|
|
||||||
|
|
||||||
'''---------------------------------------------------------------------------
|
'''---------------------------------------------------------------------------
|
||||||
The Storlet Gateway API
|
The Storlet Gateway API
|
||||||
@ -52,24 +54,25 @@ The API is made of:
|
|||||||
---------------------------------------------------------------------------'''
|
---------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
class DockerStorletRequest():
|
class DockerStorletRequest(object):
|
||||||
'''
|
'''The StorletRequest class represents a request to be processed by the
|
||||||
The StorletRequest class represents a request to be processed by the
|
|
||||||
storlet the request is derived from the Swift request and
|
storlet the request is derived from the Swift request and
|
||||||
essentially consists of:
|
essentially consists of:
|
||||||
1. A data stream to be processed
|
1. A data stream to be processed
|
||||||
2. Metadata identifying the stream
|
2. Metadata identifying the stream
|
||||||
'''
|
'''
|
||||||
|
|
||||||
def user_metadata(self, headers):
|
def user_metadata(self, headers):
|
||||||
metadata = {}
|
metadata = {}
|
||||||
for key in headers:
|
for key in headers:
|
||||||
if (key.startswith('X-Storlet') or
|
if (key.startswith('X-Storlet') or
|
||||||
key.startswith('X-Object-Meta-Storlet')):
|
key.startswith('X-Object-Meta-Storlet')):
|
||||||
pass
|
pass
|
||||||
elif (key.startswith('X-Object-Meta-') or
|
elif (key.startswith('X-Object-Meta-') or
|
||||||
key.startswith('X-Object-Meta-'.lower())):
|
key.startswith('X-Object-Meta-'.lower())):
|
||||||
short_key = key[len('X-Object-Meta-'):]
|
short_key = key[len('X-Object-Meta-'):]
|
||||||
metadata[short_key] = headers[key]
|
metadata[short_key] = headers[key]
|
||||||
return metadata
|
return metadata
|
||||||
|
|
||||||
def _getInitialRequest(self):
|
def _getInitialRequest(self):
|
||||||
@ -149,9 +152,9 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
|
|
||||||
return chunk
|
return chunk
|
||||||
|
|
||||||
def next(self, size = 1024):
|
def next(self, size=1024):
|
||||||
chunk = None
|
chunk = None
|
||||||
r, w, e = select.select([ self.obj_data ], [], [ ], self.timeout)
|
r, w, e = select.select([self.obj_data], [], [], self.timeout)
|
||||||
if len(r) == 0:
|
if len(r) == 0:
|
||||||
self.close()
|
self.close()
|
||||||
if self.obj_data in r:
|
if self.obj_data in r:
|
||||||
@ -167,11 +170,12 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
|
|
||||||
def readline(self, size=-1):
|
def readline(self, size=-1):
|
||||||
return ''
|
return ''
|
||||||
|
|
||||||
def readlines(self, sizehint=-1):
|
def readlines(self, sizehint=-1):
|
||||||
pass;
|
pass
|
||||||
|
|
||||||
def close(self):
|
def close(self):
|
||||||
if self.closed == True:
|
if self.closed is True:
|
||||||
return
|
return
|
||||||
self.closed = True
|
self.closed = True
|
||||||
os.close(self.obj_data)
|
os.close(self.obj_data)
|
||||||
@ -179,7 +183,6 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
def __del__(self):
|
def __del__(self):
|
||||||
self.close()
|
self.close()
|
||||||
|
|
||||||
|
|
||||||
def validateStorletUpload(self, req):
|
def validateStorletUpload(self, req):
|
||||||
|
|
||||||
if (self.container == self.sconf['storlet_container']):
|
if (self.container == self.sconf['storlet_container']):
|
||||||
@ -223,9 +226,9 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
self._clean_storlet_stuff_from_request(req.headers)
|
self._clean_storlet_stuff_from_request(req.headers)
|
||||||
req.headers.pop('X-Run-Storlet')
|
req.headers.pop('X-Run-Storlet')
|
||||||
|
|
||||||
slog_path = self.\
|
slog_path = self. \
|
||||||
paths.slog_path(self.idata['storlet_main_class'])
|
paths.slog_path(self.idata['storlet_main_class'])
|
||||||
storlet_pipe_path = self.\
|
storlet_pipe_path = self. \
|
||||||
paths.host_storlet_pipe(self.idata['storlet_main_class'])
|
paths.host_storlet_pipe(self.idata['storlet_main_class'])
|
||||||
|
|
||||||
sprotocol = StorletInvocationPUTProtocol(sreq,
|
sprotocol = StorletInvocationPUTProtocol(sreq,
|
||||||
@ -237,7 +240,9 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
self._set_metadata_in_headers(req.headers, out_md)
|
self._set_metadata_in_headers(req.headers, out_md)
|
||||||
self._upload_storlet_logs(slog_path)
|
self._upload_storlet_logs(slog_path)
|
||||||
|
|
||||||
return out_md, StorletGatewayDocker.IterLike(self.data_read_fd, self.storlet_timeout, sprotocol._cancel)
|
return out_md, StorletGatewayDocker.IterLike(self.data_read_fd,
|
||||||
|
self.storlet_timeout,
|
||||||
|
sprotocol._cancel)
|
||||||
|
|
||||||
def gatewayProxyGETFlow(self, req, container, obj, orig_resp):
|
def gatewayProxyGETFlow(self, req, container, obj, orig_resp):
|
||||||
# Flow for running the GET computation on the proxy
|
# Flow for running the GET computation on the proxy
|
||||||
@ -250,9 +255,9 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
docker_updated)
|
docker_updated)
|
||||||
self._add_system_params(req.params)
|
self._add_system_params(req.params)
|
||||||
|
|
||||||
slog_path = self.\
|
slog_path = self. \
|
||||||
paths.slog_path(self.idata['storlet_main_class'])
|
paths.slog_path(self.idata['storlet_main_class'])
|
||||||
storlet_pipe_path = self.\
|
storlet_pipe_path = self. \
|
||||||
paths.host_storlet_pipe(self.idata['storlet_main_class'])
|
paths.host_storlet_pipe(self.idata['storlet_main_class'])
|
||||||
|
|
||||||
sprotocol = StorletInvocationSLOProtocol(sreq,
|
sprotocol = StorletInvocationSLOProtocol(sreq,
|
||||||
@ -264,7 +269,9 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
self._set_metadata_in_headers(orig_resp.headers, out_md)
|
self._set_metadata_in_headers(orig_resp.headers, out_md)
|
||||||
self._upload_storlet_logs(slog_path)
|
self._upload_storlet_logs(slog_path)
|
||||||
|
|
||||||
return out_md, StorletGatewayDocker.IterLike(self.data_read_fd, self.storlet_timeout, sprotocol._cancel)
|
return out_md, StorletGatewayDocker.IterLike(self.data_read_fd,
|
||||||
|
self.storlet_timeout,
|
||||||
|
sprotocol._cancel)
|
||||||
|
|
||||||
def gatewayObjectGetFlow(self, req, container, obj, orig_resp):
|
def gatewayObjectGetFlow(self, req, container, obj, orig_resp):
|
||||||
sreq = StorletGETRequest(self.account, orig_resp, req.params)
|
sreq = StorletGETRequest(self.account, orig_resp, req.params)
|
||||||
@ -276,9 +283,9 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
docker_updated)
|
docker_updated)
|
||||||
self._add_system_params(req.params)
|
self._add_system_params(req.params)
|
||||||
|
|
||||||
slog_path = self.\
|
slog_path = self. \
|
||||||
paths.slog_path(self.idata['storlet_main_class'])
|
paths.slog_path(self.idata['storlet_main_class'])
|
||||||
storlet_pipe_path = self.paths.\
|
storlet_pipe_path = self.paths. \
|
||||||
host_storlet_pipe(self.idata['storlet_main_class'])
|
host_storlet_pipe(self.idata['storlet_main_class'])
|
||||||
|
|
||||||
sprotocol = StorletInvocationGETProtocol(sreq, storlet_pipe_path,
|
sprotocol = StorletInvocationGETProtocol(sreq, storlet_pipe_path,
|
||||||
@ -290,7 +297,9 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
self._set_metadata_in_headers(orig_resp.headers, out_md)
|
self._set_metadata_in_headers(orig_resp.headers, out_md)
|
||||||
self._upload_storlet_logs(slog_path)
|
self._upload_storlet_logs(slog_path)
|
||||||
|
|
||||||
return out_md, StorletGatewayDocker.IterLike(self.data_read_fd, self.storlet_timeout, sprotocol._cancel)
|
return out_md, StorletGatewayDocker.IterLike(self.data_read_fd,
|
||||||
|
self.storlet_timeout,
|
||||||
|
sprotocol._cancel)
|
||||||
|
|
||||||
def verify_access(self, env, version, account, container, object):
|
def verify_access(self, env, version, account, container, object):
|
||||||
self.logger.info('Verify access to {0}/{1}/{2}'.format(account,
|
self.logger.info('Verify access to {0}/{1}/{2}'.format(account,
|
||||||
@ -343,34 +352,34 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
req.headers['X-Storlet-' + key] = val
|
req.headers['X-Storlet-' + key] = val
|
||||||
|
|
||||||
def _add_system_params(self, params):
|
def _add_system_params(self, params):
|
||||||
'''
|
'''Adds Storlet engine specific parameters to the invocation
|
||||||
Adds Storlet engine specific parameters to the invocation
|
|
||||||
currently, this consists only of the execution path of the
|
currently, this consists only of the execution path of the
|
||||||
Storlet within the Docker container.
|
Storlet within the Docker container.
|
||||||
'''
|
'''
|
||||||
params['storlet_execution_path'] = self.\
|
params['storlet_execution_path'] = self. \
|
||||||
paths.sbox_storlet_exec(self.idata['storlet_main_class'])
|
paths.sbox_storlet_exec(self.idata['storlet_main_class'])
|
||||||
|
|
||||||
def _clean_storlet_stuff_from_request(self, headers):
|
def _clean_storlet_stuff_from_request(self, headers):
|
||||||
for key in headers:
|
for key in headers:
|
||||||
if (key.startswith('X-Storlet') or
|
if (key.startswith('X-Storlet') or
|
||||||
key.startswith('X-Object-Meta-Storlet')):
|
key.startswith('X-Object-Meta-Storlet')):
|
||||||
del headers[key]
|
del headers[key]
|
||||||
return headers
|
return headers
|
||||||
|
|
||||||
def _get_storlet_invocation_data(self, req):
|
def _get_storlet_invocation_data(self, req):
|
||||||
data = dict()
|
data = dict()
|
||||||
data['storlet_name'] = req.headers.get('X-Run-Storlet')
|
data['storlet_name'] = req.headers.get('X-Run-Storlet')
|
||||||
data['generate_log'] = req.headers.get('X-Storlet-Generate-Log', False)
|
data['generate_log'] = req.headers.get('X-Storlet-Generate-Log', False)
|
||||||
data['storlet_original_timestamp'] = req.headers.\
|
data['storlet_original_timestamp'] = req.headers. \
|
||||||
get('X-Storlet-X-Timestamp')
|
get('X-Storlet-X-Timestamp')
|
||||||
data['storlet_original_size'] = req.headers.\
|
data['storlet_original_size'] = req.headers. \
|
||||||
get('X-Storlet-Content-Length')
|
get('X-Storlet-Content-Length')
|
||||||
data['storlet_md'] = {'storlet_original_timestamp':
|
data['storlet_md'] = {'storlet_original_timestamp':
|
||||||
data['storlet_original_timestamp'],
|
data['storlet_original_timestamp'],
|
||||||
'storlet_original_size':
|
'storlet_original_size':
|
||||||
data['storlet_original_size']}
|
data['storlet_original_size']}
|
||||||
data['storlet_main_class'] = req.headers.\
|
data['storlet_main_class'] = req.headers. \
|
||||||
get('X-Object-Meta-Storlet-Main')
|
get('X-Object-Meta-Storlet-Main')
|
||||||
|
|
||||||
scope = self.account
|
scope = self.account
|
||||||
@ -378,7 +387,7 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
if data['scope'].rfind(':') > 0:
|
if data['scope'].rfind(':') > 0:
|
||||||
data['scope'] = data['scope'][:data['scope'].rfind(':')]
|
data['scope'] = data['scope'][:data['scope'].rfind(':')]
|
||||||
|
|
||||||
data['storlet_dependency'] = req.headers.\
|
data['storlet_dependency'] = req.headers. \
|
||||||
get('X-Object-Meta-Storlet-Dependency')
|
get('X-Object-Meta-Storlet-Dependency')
|
||||||
data['request_params'] = req.params
|
data['request_params'] = req.params
|
||||||
return data
|
return data
|
||||||
@ -395,7 +404,7 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
try:
|
try:
|
||||||
headers = dict()
|
headers = dict()
|
||||||
headers['CONTENT_TYPE'] = 'text/html'
|
headers['CONTENT_TYPE'] = 'text/html'
|
||||||
log_obj_name = '%s.log' %\
|
log_obj_name = '%s.log' % \
|
||||||
self.idata['storlet_name'][:self.idata['storlet_name'].
|
self.idata['storlet_name'][:self.idata['storlet_name'].
|
||||||
find('-')]
|
find('-')]
|
||||||
client.upload_object(logfile, self.account,
|
client.upload_object(logfile, self.account,
|
||||||
@ -405,8 +414,8 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
raise e
|
raise e
|
||||||
|
|
||||||
def bring_from_cache(self, obj_name, is_storlet):
|
def bring_from_cache(self, obj_name, is_storlet):
|
||||||
'''
|
'''Auxiliary function that:
|
||||||
Auxiliary function that:
|
|
||||||
(1) Brings from Swift obj_name, whether this is a
|
(1) Brings from Swift obj_name, whether this is a
|
||||||
storlet or a storlet dependency.
|
storlet or a storlet dependency.
|
||||||
(2) Copies from local cache into the Docker conrainer
|
(2) Copies from local cache into the Docker conrainer
|
||||||
@ -425,7 +434,7 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
swift_source_container = self.paths.storlet_container
|
swift_source_container = self.paths.storlet_container
|
||||||
|
|
||||||
if not os.path.exists(cache_dir):
|
if not os.path.exists(cache_dir):
|
||||||
os.makedirs(cache_dir, 0755)
|
os.makedirs(cache_dir, 0o755)
|
||||||
|
|
||||||
# cache_target_path is the actual object we need to deal with
|
# cache_target_path is the actual object we need to deal with
|
||||||
# e.g. a concrete storlet or dependency we need to bring/update
|
# e.g. a concrete storlet or dependency we need to bring/update
|
||||||
@ -467,7 +476,7 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
fn.close()
|
fn.close()
|
||||||
|
|
||||||
if not is_storlet:
|
if not is_storlet:
|
||||||
expected_perm = resp.headers.\
|
expected_perm = resp.headers. \
|
||||||
get('X-Object-Meta-Storlet-Dependency-Permissions', '')
|
get('X-Object-Meta-Storlet-Dependency-Permissions', '')
|
||||||
if expected_perm != '':
|
if expected_perm != '':
|
||||||
os.chmod(cache_target_path, int(expected_perm, 8))
|
os.chmod(cache_target_path, int(expected_perm, 8))
|
||||||
@ -479,12 +488,12 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
# 1. The Docker container does not hold a copy of the object
|
# 1. The Docker container does not hold a copy of the object
|
||||||
# 2. The Docker container holds an older version of the object
|
# 2. The Docker container holds an older version of the object
|
||||||
update_docker = False
|
update_docker = False
|
||||||
docker_storlet_path = self.paths.\
|
docker_storlet_path = self.paths. \
|
||||||
host_storlet(self.idata['storlet_main_class'])
|
host_storlet(self.idata['storlet_main_class'])
|
||||||
docker_target_path = os.path.join(docker_storlet_path, obj_name)
|
docker_target_path = os.path.join(docker_storlet_path, obj_name)
|
||||||
|
|
||||||
if not os.path.exists(docker_storlet_path):
|
if not os.path.exists(docker_storlet_path):
|
||||||
os.makedirs(docker_storlet_path, 0755)
|
os.makedirs(docker_storlet_path, 0o755)
|
||||||
update_docker = True
|
update_docker = True
|
||||||
elif not os.path.isfile(docker_target_path):
|
elif not os.path.isfile(docker_target_path):
|
||||||
update_docker = True
|
update_docker = True
|
||||||
@ -493,7 +502,7 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
fstat_docker_object = os.stat(docker_target_path)
|
fstat_docker_object = os.stat(docker_target_path)
|
||||||
b_size_changed = fstat_cached_object.st_size \
|
b_size_changed = fstat_cached_object.st_size \
|
||||||
!= fstat_docker_object.st_size
|
!= fstat_docker_object.st_size
|
||||||
b_time_changed = float(fstat_cached_object.st_mtime) <\
|
b_time_changed = float(fstat_cached_object.st_mtime) < \
|
||||||
float(fstat_docker_object.st_mtime)
|
float(fstat_docker_object.st_mtime)
|
||||||
if (b_size_changed or b_time_changed):
|
if (b_size_changed or b_time_changed):
|
||||||
update_docker = True
|
update_docker = True
|
||||||
@ -506,8 +515,8 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
return update_docker
|
return update_docker
|
||||||
|
|
||||||
def update_docker_container_from_cache(self):
|
def update_docker_container_from_cache(self):
|
||||||
'''
|
'''Iterates over the storlet name and its dependencies appearing
|
||||||
Iterates over the storlet name and its dependencies appearing
|
|
||||||
in the invocation data and make sure they are brought to the
|
in the invocation data and make sure they are brought to the
|
||||||
local cache, and from there to the Docker container.
|
local cache, and from there to the Docker container.
|
||||||
Uses the bring_from_cache auxiliary function.
|
Uses the bring_from_cache auxiliary function.
|
||||||
@ -516,7 +525,7 @@ class StorletGatewayDocker(StorletGatewayBase):
|
|||||||
# where at the host side, reside the storlet containers
|
# where at the host side, reside the storlet containers
|
||||||
storlet_path = self.paths.host_storlet_prefix()
|
storlet_path = self.paths.host_storlet_prefix()
|
||||||
if not os.path.exists(storlet_path):
|
if not os.path.exists(storlet_path):
|
||||||
os.makedirs(storlet_path, 0755)
|
os.makedirs(storlet_path, 0o755)
|
||||||
|
|
||||||
# Iterate over storlet and dependencies, and make sure
|
# Iterate over storlet and dependencies, and make sure
|
||||||
# they are updated within the Docker container.
|
# they are updated within the Docker container.
|
||||||
|
@ -20,45 +20,51 @@ Created on Feb 10, 2015
|
|||||||
'''
|
'''
|
||||||
|
|
||||||
import os
|
import os
|
||||||
import time
|
|
||||||
import stat
|
|
||||||
import select
|
import select
|
||||||
import commands
|
import stat
|
||||||
|
import subprocess
|
||||||
|
import time
|
||||||
|
|
||||||
import eventlet
|
import eventlet
|
||||||
from eventlet.timeout import Timeout
|
from eventlet.timeout import Timeout
|
||||||
import json
|
import json
|
||||||
import shutil
|
|
||||||
import sys
|
|
||||||
|
|
||||||
from swift.common.constraints import MAX_META_OVERALL_SIZE
|
from SBusPythonFacade.SBus import SBus
|
||||||
from swift.common.swob import HTTPBadRequest, Request,\
|
from SBusPythonFacade.SBusDatagram import SBusDatagram
|
||||||
HTTPInternalServerError
|
from SBusPythonFacade.SBusFileDescription import SBUS_FD_INPUT_OBJECT
|
||||||
|
from SBusPythonFacade.SBusFileDescription import SBUS_FD_LOGGER
|
||||||
from SBusPythonFacade.SBus import *
|
from SBusPythonFacade.SBusFileDescription import SBUS_FD_OUTPUT_OBJECT
|
||||||
from SBusPythonFacade.SBusDatagram import *
|
from SBusPythonFacade.SBusFileDescription import SBUS_FD_OUTPUT_OBJECT_METADATA
|
||||||
from SBusPythonFacade.SBusStorletCommand import *
|
from SBusPythonFacade.SBusFileDescription import SBUS_FD_OUTPUT_TASK_ID
|
||||||
from SBusPythonFacade.SBusFileDescription import *
|
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_CANCEL
|
||||||
|
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_DAEMON_STATUS
|
||||||
|
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_EXECUTE
|
||||||
|
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_PING
|
||||||
|
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_START_DAEMON
|
||||||
|
from SBusPythonFacade.SBusStorletCommand import SBUS_CMD_STOP_DAEMON
|
||||||
from storlet_middleware.storlet_common import StorletLogger
|
from storlet_middleware.storlet_common import StorletLogger
|
||||||
|
from swift.common.constraints import MAX_META_OVERALL_SIZE
|
||||||
|
|
||||||
eventlet.monkey_patch()
|
eventlet.monkey_patch()
|
||||||
|
|
||||||
|
|
||||||
'''---------------------------------------------------------------------------
|
'''---------------------------------------------------------------------------
|
||||||
Sandbox API
|
Sandbox API
|
||||||
'''
|
'''
|
||||||
|
|
||||||
class RunTimePaths():
|
|
||||||
'''
|
class RunTimePaths(object):
|
||||||
The Storlet Engine need to be access stuff located in many paths:
|
'''The Storlet Engine need to be access stuff located in many paths:
|
||||||
1. The various communication channels represented as pipes in the filesystem
|
|
||||||
|
1. The various communication channels represented as pipes in the
|
||||||
|
filesystem
|
||||||
2. Directories where to place Storlets
|
2. Directories where to place Storlets
|
||||||
3. Directories where to place logs
|
3. Directories where to place logs
|
||||||
|
|
||||||
Communication channels
|
Communication channels
|
||||||
----------------------
|
----------------------
|
||||||
The RunTimeSandbox communicates with the Sandbox via two types of pipes
|
The RunTimeSandbox communicates with the Sandbox via two types of pipes
|
||||||
1. factory pipe - defined per account, used for communication with the sandbox
|
1. factory pipe - defined per account, used for communication with the
|
||||||
|
sandbox
|
||||||
for e.g. start/stop a storlet daemon
|
for e.g. start/stop a storlet daemon
|
||||||
2. Storlet pipe - defined per account and Storlet, used for communication
|
2. Storlet pipe - defined per account and Storlet, used for communication
|
||||||
with a storlet daemon, e.g. to call the invoke API
|
with a storlet daemon, e.g. to call the invoke API
|
||||||
@ -87,7 +93,8 @@ class RunTimePaths():
|
|||||||
|
|
||||||
Storlets Locations
|
Storlets Locations
|
||||||
------------------
|
------------------
|
||||||
The Storlet binaries are accessible from the sandbox using a mounted directory.
|
The Storlet binaries are accessible from the sandbox using a mounted
|
||||||
|
directory.
|
||||||
This directory is called the storlet directories.
|
This directory is called the storlet directories.
|
||||||
On the host side it is of the form <storlet_dir>/<account>/<storlet_name>
|
On the host side it is of the form <storlet_dir>/<account>/<storlet_name>
|
||||||
On the sandbox side it is of the form /home/swift/<storlet_name>
|
On the sandbox side it is of the form /home/swift/<storlet_name>
|
||||||
@ -99,6 +106,7 @@ class RunTimePaths():
|
|||||||
Logs are located in paths of the form:
|
Logs are located in paths of the form:
|
||||||
<log_dir>/<account>/<storlet_name>.log
|
<log_dir>/<account>/<storlet_name>.log
|
||||||
'''
|
'''
|
||||||
|
|
||||||
def __init__(self, account, conf):
|
def __init__(self, account, conf):
|
||||||
self.account = account
|
self.account = account
|
||||||
self.scope = account[5:18]
|
self.scope = account[5:18]
|
||||||
@ -107,14 +115,13 @@ class RunTimePaths():
|
|||||||
self.factory_pipe_suffix = 'factory_pipe'
|
self.factory_pipe_suffix = 'factory_pipe'
|
||||||
self.sandbox_pipe_prefix = '/mnt/channels'
|
self.sandbox_pipe_prefix = '/mnt/channels'
|
||||||
self.storlet_pipe_suffix = '_storlet_pipe'
|
self.storlet_pipe_suffix = '_storlet_pipe'
|
||||||
self.sandbox_storlet_dir_prefix = '/home/swift'
|
self.sandbox_storlet_dir_prefix = '/home/swift'
|
||||||
self.host_storlet_root = conf['storlets_dir']
|
self.host_storlet_root = conf['storlets_dir']
|
||||||
self.host_log_path_root = conf['log_dir']
|
self.host_log_path_root = conf['log_dir']
|
||||||
self.host_cache_root = conf['cache_dir']
|
self.host_cache_root = conf['cache_dir']
|
||||||
self.storlet_container = conf['storlet_container']
|
self.storlet_container = conf['storlet_container']
|
||||||
self.storlet_dependency = conf['storlet_dependency']
|
self.storlet_dependency = conf['storlet_dependency']
|
||||||
|
|
||||||
|
|
||||||
def host_pipe_prefix(self):
|
def host_pipe_prefix(self):
|
||||||
return os.path.join(self.host_pipe_root, self.scope)
|
return os.path.join(self.host_pipe_root, self.scope)
|
||||||
|
|
||||||
@ -153,20 +160,25 @@ class RunTimePaths():
|
|||||||
return log_dir
|
return log_dir
|
||||||
|
|
||||||
def get_host_storlet_cache_dir(self):
|
def get_host_storlet_cache_dir(self):
|
||||||
return os.path.join(self.host_cache_root, self.scope,self.storlet_container)
|
return os.path.join(self.host_cache_root, self.scope,
|
||||||
|
self.storlet_container)
|
||||||
|
|
||||||
def get_host_dependency_cache_dir(self):
|
def get_host_dependency_cache_dir(self):
|
||||||
return os.path.join(self.host_cache_root, self.scope,self.storlet_dependency)
|
return os.path.join(self.host_cache_root, self.scope,
|
||||||
|
self.storlet_dependency)
|
||||||
|
|
||||||
'''---------------------------------------------------------------------------
|
'''---------------------------------------------------------------------------
|
||||||
Docker Stateful Container API
|
Docker Stateful Container API
|
||||||
The RunTimeSandbox serve as an API between the Docker Gateway and
|
The RunTimeSandbox serve as an API between the Docker Gateway and
|
||||||
a re-usable per account sandbox
|
a re-usable per account sandbox
|
||||||
---------------------------------------------------------------------------'''
|
---------------------------------------------------------------------------'''
|
||||||
class RunTimeSandbox():
|
|
||||||
'''
|
|
||||||
The RunTimeSandbox represents a re-usable per account sandbox. The sandbox
|
class RunTimeSandbox(object):
|
||||||
is re-usable in the sense that it can run several storlet daemons.
|
'''The RunTimeSandbox represents a re-usable per account sandbox.
|
||||||
|
|
||||||
|
The sandbox is re-usable in the sense that it can run several storlet
|
||||||
|
daemons.
|
||||||
|
|
||||||
The following methods are supported:
|
The following methods are supported:
|
||||||
ping - pings the sandbox for liveness
|
ping - pings the sandbox for liveness
|
||||||
@ -182,22 +194,24 @@ class RunTimeSandbox():
|
|||||||
self.account = account
|
self.account = account
|
||||||
|
|
||||||
self.sandbox_ping_interval = 0.5
|
self.sandbox_ping_interval = 0.5
|
||||||
self.sandbox_wait_timeout = int(conf['restart_linux_container_timeout'])
|
self.sandbox_wait_timeout = \
|
||||||
|
int(conf['restart_linux_container_timeout'])
|
||||||
|
|
||||||
self.docker_repo = conf['docker_repo']
|
self.docker_repo = conf['docker_repo']
|
||||||
self.docker_image_name_prefix = 'tenant'
|
self.docker_image_name_prefix = 'tenant'
|
||||||
|
|
||||||
# TODO: should come from upper layer Storlet metadata
|
# TODO(should come from upper layer Storlet metadata)
|
||||||
self.storlet_language = 'java'
|
self.storlet_language = 'java'
|
||||||
|
|
||||||
# TODO: add line in conf
|
# TODO(add line in conf)
|
||||||
self.storlet_daemon_thread_pool_size = int(conf.get('storlet_daemon_thread_pool_size',5))
|
self.storlet_daemon_thread_pool_size = \
|
||||||
self.storlet_daemon_debug_level = conf.get('storlet_daemon_debug_level','TRACE')
|
int(conf.get('storlet_daemon_thread_pool_size', 5))
|
||||||
|
self.storlet_daemon_debug_level = \
|
||||||
|
conf.get('storlet_daemon_debug_level', 'TRACE')
|
||||||
|
|
||||||
# TODO: change logger's route if possible
|
# TODO(change logger's route if possible)
|
||||||
self.logger = logger
|
self.logger = logger
|
||||||
|
|
||||||
|
|
||||||
def _parse_sandbox_factory_answer(self, str_answer):
|
def _parse_sandbox_factory_answer(self, str_answer):
|
||||||
two_tokens = str_answer.split(':', 1)
|
two_tokens = str_answer.split(':', 1)
|
||||||
b_success = False
|
b_success = False
|
||||||
@ -209,17 +223,17 @@ class RunTimeSandbox():
|
|||||||
pipe_path = self.paths.host_factory_pipe()
|
pipe_path = self.paths.host_factory_pipe()
|
||||||
|
|
||||||
read_fd, write_fd = os.pipe()
|
read_fd, write_fd = os.pipe()
|
||||||
dtg = SBusDatagram.create_service_datagram( SBUS_CMD_PING, write_fd )
|
dtg = SBusDatagram.create_service_datagram(SBUS_CMD_PING, write_fd)
|
||||||
rc = SBus.send( pipe_path, dtg )
|
rc = SBus.send(pipe_path, dtg)
|
||||||
if (rc < 0):
|
if (rc < 0):
|
||||||
return -1
|
return -1
|
||||||
|
|
||||||
reply = os.read(read_fd,10)
|
reply = os.read(read_fd, 10)
|
||||||
os.close(read_fd)
|
os.close(read_fd)
|
||||||
os.close(write_fd)
|
os.close(write_fd)
|
||||||
|
|
||||||
res, error_txt = self._parse_sandbox_factory_answer(reply)
|
res, error_txt = self._parse_sandbox_factory_answer(reply)
|
||||||
if res == True:
|
if res is True:
|
||||||
return 1
|
return 1
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
@ -228,7 +242,7 @@ class RunTimeSandbox():
|
|||||||
up = 0
|
up = 0
|
||||||
to = Timeout(self.sandbox_wait_timeout)
|
to = Timeout(self.sandbox_wait_timeout)
|
||||||
try:
|
try:
|
||||||
while do_wait == True:
|
while do_wait is True:
|
||||||
rc = self.ping()
|
rc = self.ping()
|
||||||
if (rc != 1):
|
if (rc != 1):
|
||||||
time.sleep(self.sandbox_ping_interval)
|
time.sleep(self.sandbox_ping_interval)
|
||||||
@ -237,7 +251,7 @@ class RunTimeSandbox():
|
|||||||
to.cancel()
|
to.cancel()
|
||||||
do_wait = False
|
do_wait = False
|
||||||
up = 1
|
up = 1
|
||||||
except Timeout as t:
|
except Timeout:
|
||||||
self.logger.info("wait for sandbox %s timedout" % self.account)
|
self.logger.info("wait for sandbox %s timedout" % self.account)
|
||||||
do_wait = False
|
do_wait = False
|
||||||
finally:
|
finally:
|
||||||
@ -246,8 +260,7 @@ class RunTimeSandbox():
|
|||||||
return up
|
return up
|
||||||
|
|
||||||
def restart(self):
|
def restart(self):
|
||||||
'''
|
'''Restarts the account's sandbox
|
||||||
Restarts the account's sandbox
|
|
||||||
|
|
||||||
Returned value:
|
Returned value:
|
||||||
True - If the sandbox was started successfully
|
True - If the sandbox was started successfully
|
||||||
@ -264,104 +277,110 @@ class RunTimeSandbox():
|
|||||||
|
|
||||||
docker_container_name = '%s_%s' % (self.docker_image_name_prefix,
|
docker_container_name = '%s_%s' % (self.docker_image_name_prefix,
|
||||||
account_id)
|
account_id)
|
||||||
docker_image_name = '%s/%s' % (self.docker_repo,account_id)
|
docker_image_name = '%s/%s' % (self.docker_repo, account_id)
|
||||||
pipe_mount = '%s:%s' % (self.paths.host_pipe_prefix(),
|
pipe_mount = '%s:%s' % (self.paths.host_pipe_prefix(),
|
||||||
self.paths.sandbox_pipe_prefix)
|
self.paths.sandbox_pipe_prefix)
|
||||||
|
|
||||||
storlet_mount = '%s:%s' % (self.paths.host_storlet_prefix(),
|
storlet_mount = '%s:%s' % (self.paths.host_storlet_prefix(),
|
||||||
self.paths.sandbox_storlet_dir_prefix)
|
self.paths.sandbox_storlet_dir_prefix)
|
||||||
|
|
||||||
cmd = '%s/restart_docker_container %s %s %s %s' % (
|
cmd = [self.paths.host_restart_script_dir +
|
||||||
self.paths.host_restart_script_dir,
|
'/restart_docker_container',
|
||||||
docker_container_name,
|
docker_container_name, docker_image_name, pipe_mount,
|
||||||
docker_image_name,
|
storlet_mount]
|
||||||
pipe_mount,
|
|
||||||
storlet_mount)
|
|
||||||
|
|
||||||
res = commands.getoutput(cmd)
|
subprocess.call(cmd)
|
||||||
return self.wait()
|
return self.wait()
|
||||||
|
|
||||||
def start_storlet_daemon(self, spath, storlet_id):
|
def start_storlet_daemon(self, spath, storlet_id):
|
||||||
prms = {}
|
prms = {}
|
||||||
prms['daemon_language'] = 'java'
|
prms['daemon_language'] = 'java'
|
||||||
prms['storlet_path'] = spath
|
prms['storlet_path'] = spath
|
||||||
prms['storlet_name'] = storlet_id
|
prms['storlet_name'] = storlet_id
|
||||||
prms['uds_path'] = self.paths.sbox_storlet_pipe(storlet_id)
|
prms['uds_path'] = self.paths.sbox_storlet_pipe(storlet_id)
|
||||||
prms['log_level'] = self.storlet_daemon_debug_level
|
prms['log_level'] = self.storlet_daemon_debug_level
|
||||||
prms['pool_size'] = self.storlet_daemon_thread_pool_size
|
prms['pool_size'] = self.storlet_daemon_thread_pool_size
|
||||||
|
|
||||||
read_fd, write_fd = os.pipe()
|
read_fd, write_fd = os.pipe()
|
||||||
dtg = SBusDatagram.create_service_datagram( SBUS_CMD_START_DAEMON,
|
dtg = SBusDatagram.create_service_datagram(SBUS_CMD_START_DAEMON,
|
||||||
write_fd )
|
write_fd)
|
||||||
dtg.set_exec_params( prms )
|
dtg.set_exec_params(prms)
|
||||||
|
|
||||||
pipe_path = self.paths.host_factory_pipe()
|
pipe_path = self.paths.host_factory_pipe()
|
||||||
rc = SBus.send( pipe_path, dtg )
|
rc = SBus.send(pipe_path, dtg)
|
||||||
if (rc < 0):
|
if (rc < 0):
|
||||||
return -1
|
return -1
|
||||||
reply = os.read(read_fd,10)
|
reply = os.read(read_fd, 10)
|
||||||
os.close(read_fd)
|
os.close(read_fd)
|
||||||
os.close(write_fd)
|
os.close(write_fd)
|
||||||
|
|
||||||
res, error_txt = self._parse_sandbox_factory_answer(reply)
|
res, error_txt = self._parse_sandbox_factory_answer(reply)
|
||||||
if res == True:
|
if res is True:
|
||||||
return 1
|
return 1
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
def stop_storlet_daemon(self, storlet_id):
|
def stop_storlet_daemon(self, storlet_id):
|
||||||
read_fd, write_fd = os.pipe()
|
read_fd, write_fd = os.pipe()
|
||||||
dtg = SBusDatagram.create_service_datagram( SBUS_CMD_STOP_DAEMON,
|
dtg = SBusDatagram.create_service_datagram(SBUS_CMD_STOP_DAEMON,
|
||||||
write_fd )
|
write_fd)
|
||||||
dtg.add_exec_param('storlet_name', storlet_id)
|
dtg.add_exec_param('storlet_name', storlet_id)
|
||||||
pipe_path = self.paths.host_factory_pipe()
|
pipe_path = self.paths.host_factory_pipe()
|
||||||
rc = SBus.send( pipe_path, dtg )
|
rc = SBus.send(pipe_path, dtg)
|
||||||
if (rc < 0):
|
if (rc < 0):
|
||||||
self.logger.info("Failed to send status command to %s %s" % (self.account, storlet_id))
|
self.logger.info("Failed to send status command to %s %s" %
|
||||||
|
(self.account, storlet_id))
|
||||||
return -1
|
return -1
|
||||||
|
|
||||||
reply = os.read(read_fd,10)
|
reply = os.read(read_fd, 10)
|
||||||
os.close(read_fd)
|
os.close(read_fd)
|
||||||
os.close(write_fd)
|
os.close(write_fd)
|
||||||
|
|
||||||
res, error_txt = self._parse_sandbox_factory_answer(reply)
|
res, error_txt = self._parse_sandbox_factory_answer(reply)
|
||||||
if res == True:
|
if res is True:
|
||||||
return 1
|
return 1
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
def get_storlet_daemon_status(self, storlet_id):
|
def get_storlet_daemon_status(self, storlet_id):
|
||||||
read_fd, write_fd = os.pipe()
|
read_fd, write_fd = os.pipe()
|
||||||
dtg = SBusDatagram.create_service_datagram( SBUS_CMD_DAEMON_STATUS,
|
dtg = SBusDatagram.create_service_datagram(SBUS_CMD_DAEMON_STATUS,
|
||||||
write_fd )
|
write_fd)
|
||||||
dtg.add_exec_param( 'storlet_name', storlet_id)
|
dtg.add_exec_param('storlet_name', storlet_id)
|
||||||
pipe_path = self.paths.host_factory_pipe()
|
pipe_path = self.paths.host_factory_pipe()
|
||||||
rc = SBus.send(pipe_path, dtg)
|
rc = SBus.send(pipe_path, dtg)
|
||||||
if (rc < 0):
|
if (rc < 0):
|
||||||
self.logger.info("Failed to send status command to %s %s" % (self.account, storlet_id))
|
self.logger.info("Failed to send status command to %s %s" %
|
||||||
|
(self.account, storlet_id))
|
||||||
return -1
|
return -1
|
||||||
reply = os.read(read_fd,10)
|
reply = os.read(read_fd, 10)
|
||||||
os.close(read_fd)
|
os.close(read_fd)
|
||||||
os.close(write_fd)
|
os.close(write_fd)
|
||||||
|
|
||||||
res, error_txt = self._parse_sandbox_factory_answer(reply)
|
res, error_txt = self._parse_sandbox_factory_answer(reply)
|
||||||
if res == True:
|
if res is True:
|
||||||
return 1
|
return 1
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
def activate_storlet_daemon(self, invocation_data, cache_updated = True):
|
def activate_storlet_daemon(self, invocation_data, cache_updated=True):
|
||||||
storlet_daemon_status = self.get_storlet_daemon_status(invocation_data['storlet_main_class'])
|
storlet_daemon_status = \
|
||||||
|
self.get_storlet_daemon_status(invocation_data[
|
||||||
|
'storlet_main_class'])
|
||||||
if (storlet_daemon_status == -1):
|
if (storlet_daemon_status == -1):
|
||||||
# We failed to send a command to the factory.
|
# We failed to send a command to the factory.
|
||||||
# Best we can do is execute the container.
|
# Best we can do is execute the container.
|
||||||
self.logger.debug('Failed to check Storlet daemon status, restart Docker container')
|
self.logger.debug('Failed to check Storlet daemon status, '
|
||||||
|
'restart Docker container')
|
||||||
res = self.restart()
|
res = self.restart()
|
||||||
if (res != 1):
|
if (res != 1):
|
||||||
raise Exception('Docker container is not responsive')
|
raise Exception('Docker container is not responsive')
|
||||||
storlet_daemon_status = 0
|
storlet_daemon_status = 0
|
||||||
|
|
||||||
if (cache_updated == True and storlet_daemon_status == 1):
|
if (cache_updated is True and storlet_daemon_status == 1):
|
||||||
# The cache was updated while the daemon is running we need to stop it.
|
# The cache was updated while the daemon is running we need to
|
||||||
self.logger.debug('The cache was updated, and the storlet daemon is running. Stopping daemon')
|
# stop it.
|
||||||
res = self.stop_storlet_daemon( invocation_data['storlet_main_class'] )
|
self.logger.debug('The cache was updated, and the storlet daemon '
|
||||||
|
'is running. Stopping daemon')
|
||||||
|
res = \
|
||||||
|
self.stop_storlet_daemon(invocation_data['storlet_main_class'])
|
||||||
if res != 1:
|
if res != 1:
|
||||||
res = self.restart()
|
res = self.restart()
|
||||||
if (res != 1):
|
if (res != 1):
|
||||||
@ -372,37 +391,43 @@ class RunTimeSandbox():
|
|||||||
|
|
||||||
if (storlet_daemon_status == 0):
|
if (storlet_daemon_status == 0):
|
||||||
self.logger.debug('Going to start storlet daemon!')
|
self.logger.debug('Going to start storlet daemon!')
|
||||||
class_path = '/home/swift/%s/%s' % (invocation_data['storlet_main_class'],
|
class_path = \
|
||||||
invocation_data['storlet_name'])
|
'/home/swift/%s/%s' % (invocation_data['storlet_main_class'],
|
||||||
|
invocation_data['storlet_name'])
|
||||||
for dep in invocation_data['storlet_dependency'].split(','):
|
for dep in invocation_data['storlet_dependency'].split(','):
|
||||||
class_path = '%s:/home/swift/%s/%s' %\
|
class_path = '%s:/home/swift/%s/%s' % \
|
||||||
(class_path,
|
(class_path,
|
||||||
invocation_data['storlet_main_class'],
|
invocation_data['storlet_main_class'],
|
||||||
dep)
|
dep)
|
||||||
|
|
||||||
daemon_status = self.start_storlet_daemon(
|
daemon_status = \
|
||||||
class_path,
|
self.start_storlet_daemon(class_path,
|
||||||
invocation_data['storlet_main_class'])
|
invocation_data[
|
||||||
|
'storlet_main_class'])
|
||||||
|
|
||||||
if daemon_status != 1:
|
if daemon_status != 1:
|
||||||
self.logger.error('Daemon start Failed, returned code is %d' % daemon_status)
|
self.logger.error('Daemon start Failed, returned code is %d' %
|
||||||
|
daemon_status)
|
||||||
raise Exception('Daemon start failed')
|
raise Exception('Daemon start failed')
|
||||||
else:
|
else:
|
||||||
self.logger.debug('Daemon started')
|
self.logger.debug('Daemon started')
|
||||||
|
|
||||||
'''---------------------------------------------------------------------------
|
'''---------------------------------------------------------------------------
|
||||||
Storlet Daemon API
|
Storlet Daemon API
|
||||||
The StorletInvocationGETProtocol, StorletInvocationPUTProtocol, StorletInvocationSLOProtocol
|
The StorletInvocationGETProtocol, StorletInvocationPUTProtocol,
|
||||||
|
StorletInvocationSLOProtocol
|
||||||
server as an API between the Docker Gateway and the Storlet Daemon which
|
server as an API between the Docker Gateway and the Storlet Daemon which
|
||||||
runs inside the Docker container. These classes implement the Storlet execution
|
runs inside the Docker container. These classes implement the Storlet execution
|
||||||
protocol
|
protocol
|
||||||
---------------------------------------------------------------------------'''
|
---------------------------------------------------------------------------'''
|
||||||
class StorletInvocationProtocol():
|
|
||||||
|
|
||||||
|
class StorletInvocationProtocol(object):
|
||||||
|
|
||||||
def _add_input_stream(self, appendFd):
|
def _add_input_stream(self, appendFd):
|
||||||
#self.fds.append(self.srequest.stream
|
# self.fds.append(self.srequest.stream
|
||||||
self.fds.append(appendFd)
|
self.fds.append(appendFd)
|
||||||
# TODO: Break request metadata and systemmetadata
|
# TODO(Break request metadata and systemmetadata)
|
||||||
md = dict()
|
md = dict()
|
||||||
md['type'] = SBUS_FD_INPUT_OBJECT
|
md['type'] = SBUS_FD_INPUT_OBJECT
|
||||||
if self.srequest.user_metadata is not None:
|
if self.srequest.user_metadata is not None:
|
||||||
@ -455,24 +480,23 @@ class StorletInvocationProtocol():
|
|||||||
|
|
||||||
def _cancel(self):
|
def _cancel(self):
|
||||||
read_fd, write_fd = os.pipe()
|
read_fd, write_fd = os.pipe()
|
||||||
dtg = SBusDatagram.create_service_datagram( SBUS_CMD_CANCEL, write_fd )
|
dtg = SBusDatagram.create_service_datagram(SBUS_CMD_CANCEL, write_fd)
|
||||||
dtg.set_task_id(self.task_id)
|
dtg.set_task_id(self.task_id)
|
||||||
rc = SBus.send( self.storlet_pipe_path, dtg )
|
rc = SBus.send(self.storlet_pipe_path, dtg)
|
||||||
if (rc < 0):
|
if (rc < 0):
|
||||||
return -1
|
return -1
|
||||||
|
|
||||||
reply = os.read(read_fd,10)
|
os.read(read_fd, 10)
|
||||||
os.close(read_fd)
|
os.close(read_fd)
|
||||||
os.close(write_fd)
|
os.close(write_fd)
|
||||||
|
|
||||||
|
|
||||||
def _invoke(self):
|
def _invoke(self):
|
||||||
dtg = SBusDatagram()
|
dtg = SBusDatagram()
|
||||||
dtg.set_files( self.fds )
|
dtg.set_files(self.fds)
|
||||||
dtg.set_metadata( self.fdmd )
|
dtg.set_metadata(self.fdmd)
|
||||||
dtg.set_exec_params( self.srequest.params )
|
dtg.set_exec_params(self.srequest.params)
|
||||||
dtg.set_command(SBUS_CMD_EXECUTE)
|
dtg.set_command(SBUS_CMD_EXECUTE)
|
||||||
rc = SBus.send( self.storlet_pipe_path, dtg )
|
rc = SBus.send(self.storlet_pipe_path, dtg)
|
||||||
|
|
||||||
if (rc < 0):
|
if (rc < 0):
|
||||||
raise Exception("Failed to send execute command")
|
raise Exception("Failed to send execute command")
|
||||||
@ -480,7 +504,8 @@ class StorletInvocationProtocol():
|
|||||||
self._wait_for_read_with_timeout(self.execution_str_read_fd)
|
self._wait_for_read_with_timeout(self.execution_str_read_fd)
|
||||||
self.task_id = os.read(self.execution_str_read_fd, 10)
|
self.task_id = os.read(self.execution_str_read_fd, 10)
|
||||||
|
|
||||||
def __init__(self, srequest, storlet_pipe_path, storlet_logger_path, timeout):
|
def __init__(self, srequest, storlet_pipe_path, storlet_logger_path,
|
||||||
|
timeout):
|
||||||
self.srequest = srequest
|
self.srequest = srequest
|
||||||
self.storlet_pipe_path = storlet_pipe_path
|
self.storlet_pipe_path = storlet_pipe_path
|
||||||
self.storlet_logger_path = storlet_logger_path
|
self.storlet_logger_path = storlet_logger_path
|
||||||
@ -504,7 +529,7 @@ class StorletInvocationProtocol():
|
|||||||
os.makedirs(storlet_logger_path)
|
os.makedirs(storlet_logger_path)
|
||||||
|
|
||||||
def _wait_for_read_with_timeout(self, fd):
|
def _wait_for_read_with_timeout(self, fd):
|
||||||
r, w, e = select.select([ fd ], [], [ ], self.timeout)
|
r, w, e = select.select([fd], [], [], self.timeout)
|
||||||
if len(r) == 0:
|
if len(r) == 0:
|
||||||
if self.task_id:
|
if self.task_id:
|
||||||
self._cancel()
|
self._cancel()
|
||||||
@ -519,16 +544,20 @@ class StorletInvocationProtocol():
|
|||||||
md = json.loads(flat_json)
|
md = json.loads(flat_json)
|
||||||
return md
|
return md
|
||||||
|
|
||||||
|
|
||||||
class StorletInvocationGETProtocol(StorletInvocationProtocol):
|
class StorletInvocationGETProtocol(StorletInvocationProtocol):
|
||||||
|
|
||||||
def _add_input_stream(self):
|
def _add_input_stream(self):
|
||||||
StorletInvocationProtocol._add_input_stream(self, self.srequest.stream)
|
StorletInvocationProtocol._add_input_stream(self, self.srequest.stream)
|
||||||
|
|
||||||
def __init__(self, srequest, storlet_pipe_path, storlet_logger_path, timeout):
|
def __init__(self, srequest, storlet_pipe_path, storlet_logger_path,
|
||||||
StorletInvocationProtocol.__init__(self, srequest, storlet_pipe_path, storlet_logger_path, timeout)
|
timeout):
|
||||||
|
StorletInvocationProtocol.__init__(self, srequest, storlet_pipe_path,
|
||||||
|
storlet_logger_path, timeout)
|
||||||
|
|
||||||
def communicate(self):
|
def communicate(self):
|
||||||
self.storlet_logger = StorletLogger(self.storlet_logger_path, 'storlet_invoke')
|
self.storlet_logger = StorletLogger(self.storlet_logger_path,
|
||||||
|
'storlet_invoke')
|
||||||
self.storlet_logger.open()
|
self.storlet_logger.open()
|
||||||
|
|
||||||
self._prepare_invocation_descriptors()
|
self._prepare_invocation_descriptors()
|
||||||
@ -547,20 +576,24 @@ class StorletInvocationGETProtocol(StorletInvocationProtocol):
|
|||||||
|
|
||||||
return out_md, self.data_read_fd
|
return out_md, self.data_read_fd
|
||||||
|
|
||||||
|
|
||||||
class StorletInvocationProxyProtocol(StorletInvocationProtocol):
|
class StorletInvocationProxyProtocol(StorletInvocationProtocol):
|
||||||
|
|
||||||
def __init__(self, srequest, storlet_pipe_path, storlet_logger_path, timeout):
|
def __init__(self, srequest, storlet_pipe_path, storlet_logger_path,
|
||||||
StorletInvocationProtocol.__init__(self, srequest, storlet_pipe_path, storlet_logger_path, timeout)
|
timeout):
|
||||||
|
StorletInvocationProtocol.__init__(self, srequest, storlet_pipe_path,
|
||||||
|
storlet_logger_path, timeout)
|
||||||
self.input_data_read_fd, self.input_data_write_fd = os.pipe()
|
self.input_data_read_fd, self.input_data_write_fd = os.pipe()
|
||||||
# YM this pipe permits to take data from srequest.stream to input_data_write_fd
|
# YM this pipe permits to take data from srequest.stream to
|
||||||
|
# input_data_write_fd
|
||||||
# YM the write side stays with us, the read side is sent to storlet
|
# YM the write side stays with us, the read side is sent to storlet
|
||||||
|
|
||||||
|
|
||||||
def _add_input_stream(self):
|
def _add_input_stream(self):
|
||||||
StorletInvocationProtocol._add_input_stream(self, self.input_data_read_fd)
|
StorletInvocationProtocol._add_input_stream(self,
|
||||||
|
self.input_data_read_fd)
|
||||||
|
|
||||||
def _wait_for_write_with_timeout(self,fd):
|
def _wait_for_write_with_timeout(self, fd):
|
||||||
r, w, e = select.select([ ], [ fd ], [ ], self.timeout)
|
r, w, e = select.select([], [fd], [], self.timeout)
|
||||||
if len(w) == 0:
|
if len(w) == 0:
|
||||||
raise Timeout('Timeout while waiting for storlet to read')
|
raise Timeout('Timeout while waiting for storlet to read')
|
||||||
if fd in w:
|
if fd in w:
|
||||||
@ -580,7 +613,8 @@ class StorletInvocationProxyProtocol(StorletInvocationProtocol):
|
|||||||
timeout.cancel()
|
timeout.cancel()
|
||||||
|
|
||||||
def communicate(self):
|
def communicate(self):
|
||||||
self.storlet_logger = StorletLogger(self.storlet_logger_path, 'storlet_invoke')
|
self.storlet_logger = StorletLogger(self.storlet_logger_path,
|
||||||
|
'storlet_invoke')
|
||||||
self.storlet_logger.open()
|
self.storlet_logger.open()
|
||||||
|
|
||||||
self._prepare_invocation_descriptors()
|
self._prepare_invocation_descriptors()
|
||||||
@ -607,10 +641,14 @@ class StorletInvocationProxyProtocol(StorletInvocationProtocol):
|
|||||||
|
|
||||||
return out_md, self.data_read_fd
|
return out_md, self.data_read_fd
|
||||||
|
|
||||||
|
|
||||||
class StorletInvocationPUTProtocol(StorletInvocationProxyProtocol):
|
class StorletInvocationPUTProtocol(StorletInvocationProxyProtocol):
|
||||||
|
|
||||||
def __init__(self, srequest, storlet_pipe_path, storlet_logger_path, timeout):
|
def __init__(self, srequest, storlet_pipe_path, storlet_logger_path,
|
||||||
StorletInvocationProxyProtocol.__init__(self, srequest, storlet_pipe_path, storlet_logger_path, timeout)
|
timeout):
|
||||||
|
StorletInvocationProxyProtocol.__init__(self, srequest,
|
||||||
|
storlet_pipe_path,
|
||||||
|
storlet_logger_path, timeout)
|
||||||
|
|
||||||
def _write_input_data(self):
|
def _write_input_data(self):
|
||||||
writer = os.fdopen(self.input_data_write_fd, 'w')
|
writer = os.fdopen(self.input_data_write_fd, 'w')
|
||||||
@ -622,8 +660,11 @@ class StorletInvocationPUTProtocol(StorletInvocationProxyProtocol):
|
|||||||
|
|
||||||
class StorletInvocationSLOProtocol(StorletInvocationProxyProtocol):
|
class StorletInvocationSLOProtocol(StorletInvocationProxyProtocol):
|
||||||
|
|
||||||
def __init__(self, srequest, storlet_pipe_path, storlet_logger_path, timeout):
|
def __init__(self, srequest, storlet_pipe_path, storlet_logger_path,
|
||||||
StorletInvocationProxyProtocol.__init__(self, srequest, storlet_pipe_path, storlet_logger_path, timeout)
|
timeout):
|
||||||
|
StorletInvocationProxyProtocol.__init__(self, srequest,
|
||||||
|
storlet_pipe_path,
|
||||||
|
storlet_logger_path, timeout)
|
||||||
|
|
||||||
def _write_input_data(self):
|
def _write_input_data(self):
|
||||||
writer = os.fdopen(self.input_data_write_fd, 'w')
|
writer = os.fdopen(self.input_data_write_fd, 'w')
|
||||||
@ -633,4 +674,3 @@ class StorletInvocationSLOProtocol(StorletInvocationProxyProtocol):
|
|||||||
self._write_with_timeout(writer, chunk)
|
self._write_with_timeout(writer, chunk)
|
||||||
# print >> sys.stderr, 'next SLO chunk...%d'% len(chunk)
|
# print >> sys.stderr, 'next SLO chunk...%d'% len(chunk)
|
||||||
writer.close()
|
writer.close()
|
||||||
|
|
||||||
|
@ -1,4 +1,20 @@
|
|||||||
class StorletStubBase():
|
'''-------------------------------------------------------------------------
|
||||||
|
Copyright IBM Corp. 2015, 2015 All Rights Reserved
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
Limitations under the License.
|
||||||
|
-------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
|
class StorletStubBase(object):
|
||||||
|
|
||||||
def __init__(self, storlet_conf, logger, app, version, account,
|
def __init__(self, storlet_conf, logger, app, version, account,
|
||||||
container, obj):
|
container, obj):
|
||||||
|
@ -1,40 +1,38 @@
|
|||||||
#-----------------------------------------------------------------------------------------------
|
'''-------------------------------------------------------------------------
|
||||||
# Copyright IBM Corp. 2015, 2015 All Rights Reserved
|
Copyright IBM Corp. 2015, 2015 All Rights Reserved
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
# you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
# You may obtain a copy of the License at
|
You may obtain a copy of the License at
|
||||||
#
|
|
||||||
# http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
#
|
|
||||||
# Unless required by applicable law or agreed to in writing, software
|
|
||||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
# See the License for the specific language governing permissions and
|
|
||||||
# Limitations under the License.
|
|
||||||
#-----------------------------------------------------------------------------------------------
|
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
Limitations under the License.
|
||||||
|
-------------------------------------------------------------------------'''
|
||||||
|
|
||||||
'''
|
'''
|
||||||
Created on Feb 18, 2014
|
Created on Feb 18, 2014
|
||||||
|
|
||||||
@author: gilv
|
@author: gilv
|
||||||
'''
|
'''
|
||||||
from eventlet.timeout import Timeout
|
|
||||||
import traceback
|
|
||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
|
import traceback
|
||||||
|
|
||||||
import select
|
|
||||||
|
|
||||||
class StorletTimeout(Exception):
|
class StorletTimeout(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
class StorletLogger(object):
|
class StorletLogger(object):
|
||||||
def __init__(self, path, name):
|
def __init__(self, path, name):
|
||||||
self.full_path = os.path.join(path,'%s.log' % name)
|
self.full_path = os.path.join(path, '%s.log' % name)
|
||||||
|
|
||||||
def open(self):
|
def open(self):
|
||||||
self.file = open(self.full_path,'a')
|
self.file = open(self.full_path, 'a')
|
||||||
|
|
||||||
def getfd(self):
|
def getfd(self):
|
||||||
return self.file.fileno()
|
return self.file.fileno()
|
||||||
@ -49,19 +47,20 @@ class StorletLogger(object):
|
|||||||
def fobj(self):
|
def fobj(self):
|
||||||
return open(self.full_path, 'r')
|
return open(self.full_path, 'r')
|
||||||
|
|
||||||
|
|
||||||
class StorletException(object):
|
class StorletException(object):
|
||||||
|
|
||||||
### Print details about the code line which caused the exception
|
# Print details about the code line which caused the exception
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def handle( logger, exc ):
|
def handle(logger, exc):
|
||||||
logger.info('-'*60)
|
logger.info('-' * 60)
|
||||||
logger.info(exc)
|
logger.info(exc)
|
||||||
### logging.exception()
|
# logging.exception()
|
||||||
traceback.print_exc(file=sys.stdout)
|
traceback.print_exc(file=sys.stdout)
|
||||||
logger.info('-'*60)
|
logger.info('-' * 60)
|
||||||
|
|
||||||
|
|
||||||
class StorletGatewayBase():
|
class StorletGatewayBase(object):
|
||||||
|
|
||||||
def validateStorletUpload(self, request):
|
def validateStorletUpload(self, request):
|
||||||
raise NotImplementedError("Not implemented: validateStorletUpload")
|
raise NotImplementedError("Not implemented: validateStorletUpload")
|
||||||
@ -81,6 +80,7 @@ class StorletGatewayBase():
|
|||||||
def gatewayObjectGetFlow(self, request, container, obj, original_response):
|
def gatewayObjectGetFlow(self, request, container, obj, original_response):
|
||||||
raise NotImplementedError("Not implemented: gatewayObjectGetFlow")
|
raise NotImplementedError("Not implemented: gatewayObjectGetFlow")
|
||||||
|
|
||||||
|
|
||||||
class StorletStubGateway(StorletGatewayBase):
|
class StorletStubGateway(StorletGatewayBase):
|
||||||
|
|
||||||
def __init__(self, sconf, logger, app, version, account, container,
|
def __init__(self, sconf, logger, app, version, account, container,
|
||||||
@ -93,10 +93,10 @@ class StorletStubGateway(StorletGatewayBase):
|
|||||||
self.obj = obj
|
self.obj = obj
|
||||||
self.sconf = sconf
|
self.sconf = sconf
|
||||||
self.dummy_stream = os.pipe()
|
self.dummy_stream = os.pipe()
|
||||||
self.dummy_content = sconf.get('dummy_content','aaaa')
|
self.dummy_content = sconf.get('dummy_content', 'aaaa')
|
||||||
|
|
||||||
def dummy_invocation(self):
|
def dummy_invocation(self):
|
||||||
os.write(self.dummy_stream[1],self.dummy_content)
|
os.write(self.dummy_stream[1], self.dummy_content)
|
||||||
os.close(self.dummy_stream[1])
|
os.close(self.dummy_stream[1])
|
||||||
return self.dummy_stream[0], {}
|
return self.dummy_stream[0], {}
|
||||||
|
|
||||||
|
@ -19,19 +19,23 @@ Created on Feb 18, 2014
|
|||||||
@author: Gil Vernik
|
@author: Gil Vernik
|
||||||
'''
|
'''
|
||||||
|
|
||||||
from storlet_common import StorletTimeout,StorletException
|
|
||||||
|
|
||||||
from swift.common.utils import get_logger, register_swift_info, is_success, config_true_value
|
|
||||||
from swift.common.swob import Request, Response, wsgify, \
|
|
||||||
HTTPBadRequest, HTTPUnauthorized, \
|
|
||||||
HTTPInternalServerError
|
|
||||||
from swift.proxy.controllers.base import get_account_info
|
|
||||||
from swift.common.exceptions import ConnectionTimeout
|
|
||||||
from eventlet import Timeout
|
|
||||||
|
|
||||||
import ConfigParser
|
import ConfigParser
|
||||||
import os
|
from eventlet import Timeout
|
||||||
import sys
|
from storlet_common import StorletException
|
||||||
|
from storlet_common import StorletTimeout
|
||||||
|
from swift.common.exceptions import ConnectionTimeout
|
||||||
|
from swift.common.swob import HTTPBadRequest
|
||||||
|
from swift.common.swob import HTTPInternalServerError
|
||||||
|
from swift.common.swob import HTTPUnauthorized
|
||||||
|
from swift.common.swob import Request
|
||||||
|
from swift.common.swob import Response
|
||||||
|
from swift.common.swob import wsgify
|
||||||
|
from swift.common.utils import config_true_value
|
||||||
|
from swift.common.utils import get_logger
|
||||||
|
from swift.common.utils import is_success
|
||||||
|
from swift.common.utils import register_swift_info
|
||||||
|
from swift.proxy.controllers.base import get_account_info
|
||||||
|
|
||||||
|
|
||||||
class StorletHandlerMiddleware(object):
|
class StorletHandlerMiddleware(object):
|
||||||
|
|
||||||
@ -39,14 +43,14 @@ class StorletHandlerMiddleware(object):
|
|||||||
self.app = app
|
self.app = app
|
||||||
self.logger = get_logger(conf, log_route='storlet_handler')
|
self.logger = get_logger(conf, log_route='storlet_handler')
|
||||||
self.stimeout = int(storlet_conf.get('storlet_timeout'))
|
self.stimeout = int(storlet_conf.get('storlet_timeout'))
|
||||||
self.storlet_containers = [ storlet_conf.get('storlet_container'),
|
self.storlet_containers = [storlet_conf.get('storlet_container'),
|
||||||
storlet_conf.get('storlet_dependency')]
|
storlet_conf.get('storlet_dependency')]
|
||||||
self.execution_server = storlet_conf.get('execution_server')
|
self.execution_server = storlet_conf.get('execution_server')
|
||||||
self.gateway_module = storlet_conf['gateway_module']
|
self.gateway_module = storlet_conf['gateway_module']
|
||||||
self.proxy_only_storlet_execution = storlet_conf['storlet_execute_on_proxy_only']
|
self.proxy_only_storlet_execution = \
|
||||||
|
storlet_conf['storlet_execute_on_proxy_only']
|
||||||
self.gateway_conf = storlet_conf
|
self.gateway_conf = storlet_conf
|
||||||
|
|
||||||
|
|
||||||
@wsgify
|
@wsgify
|
||||||
def __call__(self, req):
|
def __call__(self, req):
|
||||||
try:
|
try:
|
||||||
@ -61,19 +65,16 @@ class StorletHandlerMiddleware(object):
|
|||||||
return req.get_response(self.app)
|
return req.get_response(self.app)
|
||||||
|
|
||||||
self.logger.debug('storlet_handler call in %s: with %s/%s/%s' %
|
self.logger.debug('storlet_handler call in %s: with %s/%s/%s' %
|
||||||
(self.execution_server,
|
(self.execution_server, account, container, obj))
|
||||||
account,
|
|
||||||
container,
|
|
||||||
obj))
|
|
||||||
|
|
||||||
storlet_execution = False
|
storlet_execution = False
|
||||||
if 'X-Run-Storlet' in req.headers:
|
if 'X-Run-Storlet' in req.headers:
|
||||||
storlet_execution = True
|
storlet_execution = True
|
||||||
if (storlet_execution == True and account and container and obj) or \
|
if (storlet_execution is True and account and container and obj) or \
|
||||||
(container in self.storlet_containers and obj):
|
(container in self.storlet_containers and obj):
|
||||||
gateway = self.gateway_module(self.gateway_conf,
|
gateway = self.gateway_module(self.gateway_conf,
|
||||||
self.logger, self.app, version, account,
|
self.logger, self.app, version,
|
||||||
container, obj)
|
account, container, obj)
|
||||||
else:
|
else:
|
||||||
return req.get_response(self.app)
|
return req.get_response(self.app)
|
||||||
|
|
||||||
@ -86,41 +87,38 @@ class StorletHandlerMiddleware(object):
|
|||||||
if not is_success(orig_resp.status_int):
|
if not is_success(orig_resp.status_int):
|
||||||
return orig_resp
|
return orig_resp
|
||||||
|
|
||||||
if self._is_range_request(req) == True or \
|
if self._is_range_request(req) is True or \
|
||||||
self._is_slo_get_request(req, orig_resp, account, \
|
self._is_slo_get_request(req, orig_resp, account,
|
||||||
container, obj) or \
|
container, obj) or \
|
||||||
self.proxy_only_storlet_execution == True:
|
self.proxy_only_storlet_execution is True:
|
||||||
# For SLOs, and proxy only mode
|
# For SLOs, and proxy only mode
|
||||||
# Storlet are executed on the proxy
|
# Storlet are executed on the proxy
|
||||||
# Therefore we return the object part without
|
# Therefore we return the object part without
|
||||||
# Storlet invocation:
|
# Storlet invocation:
|
||||||
self.logger.info(
|
self.logger.info('storlet_handler: invocation '
|
||||||
'storlet_handler: invocation over %s/%s/%s %s' %
|
'over %s/%s/%s %s' %
|
||||||
(account, container, obj,
|
(account, container, obj,
|
||||||
'to be executed on proxy'))
|
'to be executed on proxy'))
|
||||||
return orig_resp
|
return orig_resp
|
||||||
else:
|
else:
|
||||||
# We apply here the Storlet:
|
# We apply here the Storlet:
|
||||||
self.logger.info(
|
self.logger.info('storlet_handler: invocation '
|
||||||
'storlet_handler: invocation over %s/%s/%s %s' %
|
'over %s/%s/%s %s' %
|
||||||
(account, container, obj,
|
(account, container, obj,
|
||||||
'to be executed locally'))
|
'to be executed locally'))
|
||||||
old_env = req.environ.copy()
|
old_env = req.environ.copy()
|
||||||
orig_req = Request.blank(old_env['PATH_INFO'], old_env)
|
orig_req = Request.blank(old_env['PATH_INFO'], old_env)
|
||||||
(out_md, app_iter) = gateway.gatewayObjectGetFlow(req,
|
(out_md, app_iter) = \
|
||||||
container,
|
gateway.gatewayObjectGetFlow(req, container,
|
||||||
obj,
|
obj, orig_resp)
|
||||||
orig_resp)
|
|
||||||
if 'Content-Length' in orig_resp.headers:
|
if 'Content-Length' in orig_resp.headers:
|
||||||
orig_resp.headers.pop('Content-Length')
|
orig_resp.headers.pop('Content-Length')
|
||||||
if 'Transfer-Encoding' in orig_resp.headers:
|
if 'Transfer-Encoding' in orig_resp.headers:
|
||||||
orig_resp.headers.pop('Transfer-Encoding')
|
orig_resp.headers.pop('Transfer-Encoding')
|
||||||
|
|
||||||
return Response(
|
return Response(app_iter, headers=orig_resp.headers,
|
||||||
app_iter,
|
request=orig_req,
|
||||||
headers = orig_resp.headers,
|
conditional_response=True)
|
||||||
request=orig_req,
|
|
||||||
conditional_response=True)
|
|
||||||
|
|
||||||
elif (self.execution_server == 'proxy'):
|
elif (self.execution_server == 'proxy'):
|
||||||
if (storlet_execution or container in self.storlet_containers):
|
if (storlet_execution or container in self.storlet_containers):
|
||||||
@ -152,63 +150,63 @@ class StorletHandlerMiddleware(object):
|
|||||||
gateway.augmentStorletRequest(req)
|
gateway.augmentStorletRequest(req)
|
||||||
original_resp = req.get_response(self.app)
|
original_resp = req.get_response(self.app)
|
||||||
|
|
||||||
if self._is_range_request(req) == True or \
|
if self._is_range_request(req) is True or \
|
||||||
self._is_slo_get_request(req, original_resp, account, \
|
self._is_slo_get_request(req, original_resp,
|
||||||
container, obj) or \
|
account,
|
||||||
self.proxy_only_storlet_execution == True:
|
container, obj) or \
|
||||||
|
self.proxy_only_storlet_execution is True:
|
||||||
# SLO / proxy only case:
|
# SLO / proxy only case:
|
||||||
# storlet to be invoked now at proxy side:
|
# storlet to be invoked now at proxy side:
|
||||||
(out_md, app_iter) = gateway.gatewayProxyGETFlow(req,
|
(out_md, app_iter) = \
|
||||||
container,
|
gateway.gatewayProxyGETFlow(req, container, obj,
|
||||||
obj,
|
original_resp)
|
||||||
original_resp)
|
|
||||||
|
|
||||||
# adapted from non SLO GET flow
|
# adapted from non SLO GET flow
|
||||||
if is_success(original_resp.status_int):
|
if is_success(original_resp.status_int):
|
||||||
old_env = req.environ.copy()
|
old_env = req.environ.copy()
|
||||||
orig_req = Request.blank(old_env['PATH_INFO'], old_env)
|
orig_req = Request.blank(old_env['PATH_INFO'],
|
||||||
|
old_env)
|
||||||
resp_headers = original_resp.headers
|
resp_headers = original_resp.headers
|
||||||
|
|
||||||
resp_headers['Content-Length'] = None
|
resp_headers['Content-Length'] = None
|
||||||
|
|
||||||
return Response(
|
return Response(app_iter=app_iter,
|
||||||
app_iter=app_iter,
|
headers=resp_headers,
|
||||||
headers=resp_headers,
|
request=orig_req,
|
||||||
request=orig_req,
|
conditional_response=True)
|
||||||
conditional_response=True)
|
|
||||||
return original_resp
|
return original_resp
|
||||||
|
|
||||||
else:
|
else:
|
||||||
# Non proxy GET case: Storlet was already invoked at object side
|
# Non proxy GET case: Storlet was already invoked at
|
||||||
|
# object side
|
||||||
if 'Transfer-Encoding' in original_resp.headers:
|
if 'Transfer-Encoding' in original_resp.headers:
|
||||||
original_resp.headers.pop('Transfer-Encoding')
|
original_resp.headers.pop('Transfer-Encoding')
|
||||||
|
|
||||||
if is_success(original_resp.status_int):
|
if is_success(original_resp.status_int):
|
||||||
old_env = req.environ.copy()
|
old_env = req.environ.copy()
|
||||||
orig_req = Request.blank(old_env['PATH_INFO'], old_env)
|
orig_req = Request.blank(old_env['PATH_INFO'],
|
||||||
|
old_env)
|
||||||
resp_headers = original_resp.headers
|
resp_headers = original_resp.headers
|
||||||
|
|
||||||
resp_headers['Content-Length'] = None
|
resp_headers['Content-Length'] = None
|
||||||
return Response(
|
return Response(app_iter=original_resp.app_iter,
|
||||||
app_iter=original_resp.app_iter,
|
headers=resp_headers,
|
||||||
headers=resp_headers,
|
request=orig_req,
|
||||||
request=orig_req,
|
conditional_response=True)
|
||||||
conditional_response=True)
|
|
||||||
return original_resp
|
return original_resp
|
||||||
|
|
||||||
elif req.method == 'PUT':
|
elif req.method == 'PUT':
|
||||||
if (container in self.storlet_containers):
|
if (container in self.storlet_containers):
|
||||||
ret = gateway.validateStorletUpload(req)
|
ret = gateway.validateStorletUpload(req)
|
||||||
if ret:
|
if ret:
|
||||||
return HTTPBadRequest(body = ret)
|
return HTTPBadRequest(body=ret)
|
||||||
else:
|
else:
|
||||||
if not gateway.authorizeStorletExecution(req):
|
if not gateway.authorizeStorletExecution(req):
|
||||||
return HTTPUnauthorized('Storlet: no permissions')
|
return HTTPUnauthorized('Storlet: no permissions')
|
||||||
if storlet_execution:
|
if storlet_execution:
|
||||||
gateway.augmentStorletRequest(req)
|
gateway.augmentStorletRequest(req)
|
||||||
(out_md, app_iter) = gateway.gatewayProxyPutFlow(req,
|
(out_md, app_iter) = \
|
||||||
container,
|
gateway.gatewayProxyPutFlow(req, container, obj)
|
||||||
obj)
|
|
||||||
req.environ['wsgi.input'] = app_iter
|
req.environ['wsgi.input'] = app_iter
|
||||||
if 'CONTENT_LENGTH' in req.environ:
|
if 'CONTENT_LENGTH' in req.environ:
|
||||||
req.environ.pop('CONTENT_LENGTH')
|
req.environ.pop('CONTENT_LENGTH')
|
||||||
@ -229,6 +227,7 @@ class StorletHandlerMiddleware(object):
|
|||||||
args:
|
args:
|
||||||
req: the request
|
req: the request
|
||||||
'''
|
'''
|
||||||
|
|
||||||
def _is_range_request(self, req):
|
def _is_range_request(self, req):
|
||||||
if 'Range' in req.headers:
|
if 'Range' in req.headers:
|
||||||
return True
|
return True
|
||||||
@ -244,39 +243,50 @@ class StorletHandlerMiddleware(object):
|
|||||||
container: the response as extracted from req
|
container: the response as extracted from req
|
||||||
obj: the response as extracted from req
|
obj: the response as extracted from req
|
||||||
'''
|
'''
|
||||||
|
|
||||||
def _is_slo_get_request(self, req, resp, account, container, obj):
|
def _is_slo_get_request(self, req, resp, account, container, obj):
|
||||||
if req.method != 'GET':
|
if req.method != 'GET':
|
||||||
return False
|
return False
|
||||||
if req.params.get('multipart-manifest') == 'get':
|
if req.params.get('multipart-manifest') == 'get':
|
||||||
return False
|
return False
|
||||||
|
|
||||||
self.logger.info( 'Verify if {0}/{1}/{2} is an SLO assembly object'.format(account,container, obj))
|
self.logger.info('Verify if {0}/{1}/{2} is an SLO assembly object'.
|
||||||
|
format(account, container, obj))
|
||||||
|
|
||||||
if resp.status_int < 300 and resp.status_int >= 200 :
|
if resp.status_int < 300 and resp.status_int >= 200:
|
||||||
for key in resp.headers:
|
for key in resp.headers:
|
||||||
if (key.lower() == 'x-static-large-object' and
|
if (key.lower() == 'x-static-large-object'
|
||||||
config_true_value(resp.headers[key])):
|
and config_true_value(resp.headers[key])):
|
||||||
self.logger.info( '{0}/{1}/{2} is indeed an SLO assembly object'.format(account,container, obj))
|
self.logger.info('{0}/{1}/{2} is indeed an SLO assembly '
|
||||||
|
'object'.format(account, container, obj))
|
||||||
return True
|
return True
|
||||||
self.logger.info( '{0}/{1}/{2} is NOT an SLO assembly object'.format(account,container, obj))
|
self.logger.info('{0}/{1}/{2} is NOT an SLO assembly object'.
|
||||||
|
format(account, container, obj))
|
||||||
return False
|
return False
|
||||||
self.logger.error( 'Failed to check if {0}/{1}/{2} is an SLO assembly object. Got status {3}'.format(account,container, obj,resp.status))
|
self.logger.error('Failed to check if {0}/{1}/{2} is an SLO assembly '
|
||||||
raise Exception('Failed to check if {0}/{1}/{2} is an SLO assembly object. Got status {3}'.format(account,container, obj,resp.status))
|
'object. Got status {3}'.
|
||||||
|
format(account, container, obj, resp.status))
|
||||||
|
raise Exception('Failed to check if {0}/{1}/{2} is an SLO assembly '
|
||||||
|
'object. Got status {3}'.format(account, container,
|
||||||
|
obj, resp.status))
|
||||||
|
|
||||||
|
|
||||||
def filter_factory(global_conf, **local_conf):
|
def filter_factory(global_conf, **local_conf):
|
||||||
|
|
||||||
conf = global_conf.copy()
|
conf = global_conf.copy()
|
||||||
conf.update(local_conf)
|
conf.update(local_conf)
|
||||||
storlet_conf = dict()
|
storlet_conf = dict()
|
||||||
storlet_conf['storlet_timeout'] = conf.get('storlet_timeout',40)
|
storlet_conf['storlet_timeout'] = conf.get('storlet_timeout', 40)
|
||||||
storlet_conf['storlet_container'] = conf.get('storlet_container','storlet')
|
storlet_conf['storlet_container'] = \
|
||||||
|
conf.get('storlet_container', 'storlet')
|
||||||
storlet_conf['storlet_dependency'] = conf.get('storlet_dependency',
|
storlet_conf['storlet_dependency'] = conf.get('storlet_dependency',
|
||||||
'dependency')
|
'dependency')
|
||||||
storlet_conf['execution_server'] = conf.get('execution_server', '')
|
storlet_conf['execution_server'] = conf.get('execution_server', '')
|
||||||
storlet_conf['storlet_execute_on_proxy_only'] = config_true_value(conf.get('storlet_execute_on_proxy_only', 'false'))
|
storlet_conf['storlet_execute_on_proxy_only'] = \
|
||||||
|
config_true_value(conf.get('storlet_execute_on_proxy_only', 'false'))
|
||||||
storlet_conf['gateway_conf'] = {}
|
storlet_conf['gateway_conf'] = {}
|
||||||
|
|
||||||
module_name = conf.get('storlet_gateway_module','')
|
module_name = conf.get('storlet_gateway_module', '')
|
||||||
mo = module_name[:module_name.rfind(':')]
|
mo = module_name[:module_name.rfind(':')]
|
||||||
cl = module_name[module_name.rfind(':') + 1:]
|
cl = module_name[module_name.rfind(':') + 1:]
|
||||||
module = __import__(mo, fromlist=[cl])
|
module = __import__(mo, fromlist=[cl])
|
||||||
@ -288,7 +298,7 @@ def filter_factory(global_conf, **local_conf):
|
|||||||
|
|
||||||
additional_items = configParser.items("DEFAULT")
|
additional_items = configParser.items("DEFAULT")
|
||||||
for key, val in additional_items:
|
for key, val in additional_items:
|
||||||
storlet_conf[key]= val
|
storlet_conf[key] = val
|
||||||
|
|
||||||
swift_info = {}
|
swift_info = {}
|
||||||
storlet_conf["gateway_module"] = the_class
|
storlet_conf["gateway_module"] = the_class
|
||||||
@ -297,4 +307,3 @@ def filter_factory(global_conf, **local_conf):
|
|||||||
def storlet_handler_filter(app):
|
def storlet_handler_filter(app):
|
||||||
return StorletHandlerMiddleware(app, conf, storlet_conf)
|
return StorletHandlerMiddleware(app, conf, storlet_conf)
|
||||||
return storlet_handler_filter
|
return storlet_handler_filter
|
||||||
|
|
||||||
|
4
HACKING.rst
Normal file
4
HACKING.rst
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
storlets Style Commandments
|
||||||
|
===============================================
|
||||||
|
|
||||||
|
Read the OpenStack Style Commandments http://docs.openstack.org/developer/hacking/
|
28
LICENSE
28
LICENSE
@ -1,3 +1,4 @@
|
|||||||
|
|
||||||
Apache License
|
Apache License
|
||||||
Version 2.0, January 2004
|
Version 2.0, January 2004
|
||||||
http://www.apache.org/licenses/
|
http://www.apache.org/licenses/
|
||||||
@ -173,30 +174,3 @@
|
|||||||
incurred by, or claims asserted against, such Contributor by reason
|
incurred by, or claims asserted against, such Contributor by reason
|
||||||
of your accepting any such warranty or additional liability.
|
of your accepting any such warranty or additional liability.
|
||||||
|
|
||||||
END OF TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
APPENDIX: How to apply the Apache License to your work.
|
|
||||||
|
|
||||||
To apply the Apache License to your work, attach the following
|
|
||||||
boilerplate notice, with the fields enclosed by brackets "{}"
|
|
||||||
replaced with your own identifying information. (Don't include
|
|
||||||
the brackets!) The text should be enclosed in the appropriate
|
|
||||||
comment syntax for the file format. We also recommend that a
|
|
||||||
file or class name and description of purpose be included on the
|
|
||||||
same "printed page" as the copyright notice for easier
|
|
||||||
identification within third-party archives.
|
|
||||||
|
|
||||||
Copyright {yyyy} {name of copyright owner}
|
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
you may not use this file except in compliance with the License.
|
|
||||||
You may obtain a copy of the License at
|
|
||||||
|
|
||||||
http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
|
|
||||||
Unless required by applicable law or agreed to in writing, software
|
|
||||||
distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
See the License for the specific language governing permissions and
|
|
||||||
limitations under the License.
|
|
||||||
|
|
||||||
|
6
MANIFEST.in
Normal file
6
MANIFEST.in
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
include AUTHORS
|
||||||
|
include ChangeLog
|
||||||
|
exclude .gitignore
|
||||||
|
exclude .gitreview
|
||||||
|
|
||||||
|
global-exclude *.pyc
|
19
README.rst
Normal file
19
README.rst
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
===============================
|
||||||
|
storlets
|
||||||
|
===============================
|
||||||
|
|
||||||
|
Middleware and Compute Engine for an OpenStack Swift compute framework that runs compute within a Swift cluster
|
||||||
|
|
||||||
|
Please feel here a long description which must be at least 3 lines wrapped on
|
||||||
|
80 cols, so that distribution package maintainers can use it in their packages.
|
||||||
|
Note that this is a hard requirement.
|
||||||
|
|
||||||
|
* Free software: Apache license
|
||||||
|
* Documentation: http://docs.openstack.org/developer/storlets
|
||||||
|
* Source: http://git.openstack.org/cgit/openstack/storlets
|
||||||
|
* Bugs: http://bugs.launchpad.net/storlets
|
||||||
|
|
||||||
|
Features
|
||||||
|
--------
|
||||||
|
|
||||||
|
* TODO
|
@ -17,65 +17,77 @@ Limitations under the License.
|
|||||||
@author: gilv / cdoron / evgenyl
|
@author: gilv / cdoron / evgenyl
|
||||||
'''
|
'''
|
||||||
|
|
||||||
from sys_test_params import *
|
from storlets_test_utils import put_storlet_object
|
||||||
from swiftclient import client as c
|
from swiftclient import client as c
|
||||||
|
from sys_test_params import ACCOUNT
|
||||||
|
from sys_test_params import AUTH_IP
|
||||||
|
from sys_test_params import AUTH_PORT
|
||||||
|
from sys_test_params import PASSWORD
|
||||||
|
from sys_test_params import USER_NAME
|
||||||
|
|
||||||
from storlets_test_utils import put_storlet_containers, put_storlet_object
|
EXECDEP_PATH_TO_BUNDLE = '../StorletSamples/ExecDepStorlet/bin/'
|
||||||
|
EXECDEP_STORLET_NAME = 'execdepstorlet-1.0.jar'
|
||||||
EXECDEP_PATH_TO_BUNDLE ='../StorletSamples/ExecDepStorlet/bin/'
|
EXECDEP_STORLET_LOG_NAME = 'execdepstorlet-1.0.log'
|
||||||
EXECDEP_STORLET_NAME='execdepstorlet-1.0.jar'
|
|
||||||
EXECDEP_STORLET_LOG_NAME='execdepstorlet-1.0.log'
|
|
||||||
EXECDEP_JUNK_FILE = 'junk.txt'
|
EXECDEP_JUNK_FILE = 'junk.txt'
|
||||||
EXECDEP_DEPS_NAMES=['get42']
|
EXECDEP_DEPS_NAMES = ['get42']
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
def put_storlet_dependency(url, token, dependency_name, local_path_to_dependency):
|
|
||||||
|
|
||||||
|
def put_storlet_dependency(url, token, dependency_name,
|
||||||
|
local_path_to_dependency):
|
||||||
metadata = {'X-Object-Meta-Storlet-Dependency-Version': '1'}
|
metadata = {'X-Object-Meta-Storlet-Dependency-Version': '1'}
|
||||||
|
|
||||||
f = open('%s/%s'% (local_path_to_dependency, dependency_name), 'r')
|
f = open('%s/%s' % (local_path_to_dependency, dependency_name), 'r')
|
||||||
content_length = None
|
content_length = None
|
||||||
response = dict()
|
response = dict()
|
||||||
c.put_object(url, token, 'dependency', dependency_name, f,
|
c.put_object(url, token, 'dependency', dependency_name, f,
|
||||||
content_length, None, None, "application/octet-stream", metadata, None, None, None, response)
|
content_length, None, None, "application/octet-stream",
|
||||||
|
metadata, None, None, None, response)
|
||||||
f.close()
|
f.close()
|
||||||
status = response.get('status')
|
status = response.get('status')
|
||||||
assert (status == 200 or status == 201)
|
assert (status == 200 or status == 201)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def put_storlet_input_object(url, token):
|
def put_storlet_input_object(url, token):
|
||||||
resp = dict()
|
resp = dict()
|
||||||
f = open('%s/%s' %(EXECDEP_PATH_TO_BUNDLE, EXECDEP_JUNK_FILE),'r')
|
f = open('%s/%s' % (EXECDEP_PATH_TO_BUNDLE, EXECDEP_JUNK_FILE), 'r')
|
||||||
c.put_object(url, token, 'myobjects', EXECDEP_JUNK_FILE, f,
|
c.put_object(url, token, 'myobjects', EXECDEP_JUNK_FILE, f,
|
||||||
content_type = "application/octet-stream",
|
content_type="application/octet-stream",
|
||||||
response_dict = resp)
|
response_dict=resp)
|
||||||
f.close()
|
f.close()
|
||||||
status = resp.get('status')
|
status = resp.get('status')
|
||||||
assert (status == 200 or status == 201)
|
assert (status == 200 or status == 201)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
def deploy_storlet(url,token, name, jarName):
|
|
||||||
#No need to create containers every time
|
|
||||||
#put_storlet_containers(url, token)
|
def deploy_storlet(url, token, name, jarName):
|
||||||
|
# No need to create containers every time
|
||||||
|
# put_storlet_containers(url, token)
|
||||||
put_storlet_object(url, token, jarName,
|
put_storlet_object(url, token, jarName,
|
||||||
'../StorletSamples/'+ name + '/bin/',
|
'../StorletSamples/' + name + '/bin/',
|
||||||
'',
|
'',
|
||||||
'com.ibm.storlet.' + name.lower() + '.' + name)
|
'com.ibm.storlet.' + name.lower() + '.' + name)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def invoke_storlet(url, token, storletName, jarName, objectName, mode):
|
def invoke_storlet(url, token, storletName, jarName, objectName, mode):
|
||||||
resp = dict()
|
resp = dict()
|
||||||
if mode == 'PUT':
|
if mode == 'PUT':
|
||||||
f = open('../StorletSamples/' + storletName + '/sampleData.txt','r')
|
f = open('../StorletSamples/' + storletName + '/sampleData.txt', 'r')
|
||||||
c.put_object(url, token, 'myobjects', objectName, f,
|
c.put_object(url, token, 'myobjects', objectName, f,
|
||||||
headers = {'X-Run-Storlet':jarName},
|
headers={'X-Run-Storlet': jarName},
|
||||||
response_dict = resp)
|
response_dict=resp)
|
||||||
f.close()
|
f.close()
|
||||||
if mode == 'GET':
|
if mode == 'GET':
|
||||||
resp_headers, saved_content = c.get_object(url, token,
|
resp_headers, saved_content = \
|
||||||
'myobjects',
|
c.get_object(url, token, 'myobjects', objectName,
|
||||||
objectName,
|
headers={'X-Run-Storlet': jarName},
|
||||||
headers = {'X-Run-Storlet':jarName},
|
response_dict=resp)
|
||||||
response_dict=resp)
|
|
||||||
|
|
||||||
assert (resp['status'] == 200 or resp['status'] == 201)
|
assert (resp['status'] == 200 or resp['status'] == 201)
|
||||||
|
|
||||||
@ -83,40 +95,43 @@ def invoke_storlet(url, token, storletName, jarName, objectName, mode):
|
|||||||
return resp_headers, saved_content
|
return resp_headers, saved_content
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
os_options = {'tenant_name': ACCOUNT}
|
os_options = {'tenant_name': ACCOUNT}
|
||||||
url, token = c.get_auth( 'http://' + AUTH_IP + ":"
|
url, token = c.get_auth('http://' + AUTH_IP + ":"
|
||||||
+ AUTH_PORT + '/v2.0',
|
+ AUTH_PORT + '/v2.0',
|
||||||
ACCOUNT + ':' + USER_NAME,
|
ACCOUNT + ':' + USER_NAME,
|
||||||
PASSWORD,
|
PASSWORD,
|
||||||
os_options = os_options,
|
os_options=os_options,
|
||||||
auth_version = '2.0' )
|
auth_version='2.0')
|
||||||
|
|
||||||
print 'Deploying ReadHeaders storlet'
|
print('Deploying ReadHeaders storlet')
|
||||||
deploy_storlet(url, token, 'ReadHeadersStorlet',
|
deploy_storlet(url, token, 'ReadHeadersStorlet',
|
||||||
'readheadersstorlet-1.0.jar')
|
'readheadersstorlet-1.0.jar')
|
||||||
|
|
||||||
print 'Deploying ReadHeaders dependency'
|
print('Deploying ReadHeaders dependency')
|
||||||
put_storlet_dependency(url, token, 'json-simple-1.1.1.jar',
|
put_storlet_dependency(url, token, 'json-simple-1.1.1.jar',
|
||||||
'../StorletSamples/ReadHeadersStorlet/lib')
|
'../StorletSamples/ReadHeadersStorlet/lib')
|
||||||
|
|
||||||
print 'Deploying CSS storlet'
|
print('Deploying CSS storlet')
|
||||||
deploy_storlet(url, token, 'CssStorlet', 'cssstorlet-1.0.jar')
|
deploy_storlet(url, token, 'CssStorlet', 'cssstorlet-1.0.jar')
|
||||||
|
|
||||||
print "Invoking CSS storlet in PUT mode"
|
print("Invoking CSS storlet in PUT mode")
|
||||||
invoke_storlet(url, token, 'CssStorlet', 'cssstorlet-1.0.jar',
|
invoke_storlet(url, token, 'CssStorlet', 'cssstorlet-1.0.jar',
|
||||||
'testDataCss', 'PUT')
|
'testDataCss', 'PUT')
|
||||||
|
|
||||||
print "Invoking ReadHeaders storlet in GET mode"
|
print("Invoking ReadHeaders storlet in GET mode")
|
||||||
headers, content = invoke_storlet(url, token, 'ReadHeadersStorlet',
|
headers, content = invoke_storlet(url, token, 'ReadHeadersStorlet',
|
||||||
'readheadersstorlet-1.0.jar', 'testDataCss', 'GET')
|
'readheadersstorlet-1.0.jar',
|
||||||
|
'testDataCss', 'GET')
|
||||||
|
|
||||||
assert '{"Square-Sums":"[2770444.6455999985, 1.9458262030000027E7,' \
|
assert '{"Square-Sums":"[2770444.6455999985, 1.9458262030000027E7,' \
|
||||||
+ ' 95.17999999999981]","Lines-Num":"356","Sums":"[27037.0' \
|
+ ' 95.17999999999981]","Lines-Num":"356","Sums":"[27037.0' \
|
||||||
+ '40000000008, 83229.09999999998, 168.39999999999947]"}'\
|
+ '40000000008, 83229.09999999998, 168.39999999999947]"}' \
|
||||||
== content
|
== content
|
||||||
|
|
||||||
print "ReadHeaders test finished"
|
print("ReadHeaders test finished")
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
main()
|
main()
|
||||||
|
@ -1,52 +1,76 @@
|
|||||||
import os
|
'''-------------------------------------------------------------------------
|
||||||
import sys
|
Copyright IBM Corp. 2015, 2015 All Rights Reserved
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
Limitations under the License.
|
||||||
|
-------------------------------------------------------------------------'''
|
||||||
|
|
||||||
import json
|
import json
|
||||||
|
import os
|
||||||
import random
|
import random
|
||||||
import string
|
import string
|
||||||
from sys_test_params import *
|
|
||||||
from swiftclient import client as c
|
from swiftclient import client as c
|
||||||
|
from sys_test_params import ACCOUNT
|
||||||
|
from sys_test_params import AUTH_IP
|
||||||
|
from sys_test_params import AUTH_PORT
|
||||||
|
from sys_test_params import PASSWORD
|
||||||
|
from sys_test_params import USER_NAME
|
||||||
|
|
||||||
from storlets_test_utils import put_storlet_containers, put_storlet_object, progress, progress_ln, progress_msg
|
|
||||||
from identity_storlet_test import IDENTITY_STORLET_NAME
|
from identity_storlet_test import IDENTITY_STORLET_NAME
|
||||||
|
from storlets_test_utils import progress
|
||||||
|
from storlets_test_utils import progress_ln
|
||||||
|
from storlets_test_utils import progress_msg
|
||||||
|
from storlets_test_utils import put_storlet_object
|
||||||
|
|
||||||
SLOIDENTITY_PATH_TO_BUNDLE ='../StorletSamples/SLOIdentityStorlet/bin'
|
SLOIDENTITY_PATH_TO_BUNDLE = '../StorletSamples/SLOIdentityStorlet/bin'
|
||||||
SLOIDENTITY_STORLET_NAME='sloidentitystorlet-1.0.jar'
|
SLOIDENTITY_STORLET_NAME = 'sloidentitystorlet-1.0.jar'
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
# Test Constants
|
# Test Constants
|
||||||
#PATH_TO_BUNDLE =
|
# PATH_TO_BUNDLE =
|
||||||
#STORLET_NAME =
|
# STORLET_NAME =
|
||||||
#STORLET_LOG_NAME =
|
# STORLET_LOG_NAME =
|
||||||
#SOURCE_FILE =
|
# SOURCE_FILE =
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def create_local_chunks():
|
def create_local_chunks():
|
||||||
for i in range(1,10):
|
for i in range(1, 10):
|
||||||
progress()
|
progress()
|
||||||
oname = '/tmp/slo_chunk_%d' % i
|
oname = '/tmp/slo_chunk_%d' % i
|
||||||
f = open(oname,'w')
|
f = open(oname, 'w')
|
||||||
f.write(''.join(random.choice(string.ascii_uppercase + string.digits) for _ in range(1048576)))
|
f.write(''.join(random.choice(string.ascii_uppercase + string.digits)
|
||||||
|
for _ in range(1048576)))
|
||||||
f.close()
|
f.close()
|
||||||
progress_ln()
|
progress_ln()
|
||||||
|
|
||||||
|
|
||||||
def delete_local_chunks():
|
def delete_local_chunks():
|
||||||
for i in range(1,10):
|
for i in range(1, 10):
|
||||||
oname = '/tmp/slo_chunk_%d' % i
|
oname = '/tmp/slo_chunk_%d' % i
|
||||||
os.remove(oname)
|
os.remove(oname)
|
||||||
|
|
||||||
|
|
||||||
def put_SLO(url, token):
|
def put_SLO(url, token):
|
||||||
# Create temp files
|
# Create temp files
|
||||||
assembly = []
|
assembly = []
|
||||||
for i in range(1,10):
|
for i in range(1, 10):
|
||||||
oname = '/tmp/slo_chunk_%d' % i
|
oname = '/tmp/slo_chunk_%d' % i
|
||||||
f = open(oname,'r')
|
f = open(oname, 'r')
|
||||||
content_length = None
|
content_length = None
|
||||||
response = dict()
|
response = dict()
|
||||||
progress()
|
progress()
|
||||||
c.put_object(url, token, 'myobjects', oname, f,
|
c.put_object(url, token, 'myobjects', oname, f,
|
||||||
content_length, None, None, "application/octet-stream",
|
content_length, None, None, "application/octet-stream",
|
||||||
None, None, None, None, response)
|
None, None, None, None, response)
|
||||||
f.close()
|
f.close()
|
||||||
status = response.get('status')
|
status = response.get('status')
|
||||||
assert (status >= 200 and status < 300)
|
assert (status >= 200 and status < 300)
|
||||||
@ -60,75 +84,82 @@ def put_SLO(url, token):
|
|||||||
|
|
||||||
content_length = None
|
content_length = None
|
||||||
response = dict()
|
response = dict()
|
||||||
headers = {'x-object-meta-prop1' : 'val1'}
|
headers = {'x-object-meta-prop1': 'val1'}
|
||||||
progress()
|
progress()
|
||||||
c.put_object(url, token, 'myobjects', 'assembly', json.dumps(assembly),
|
c.put_object(url, token, 'myobjects', 'assembly', json.dumps(assembly),
|
||||||
content_length=None, etag=None, chunk_size=None, headers=headers,
|
content_length=None, etag=None, chunk_size=None,
|
||||||
query_string='multipart-manifest=put', response_dict=response)
|
headers=headers, query_string='multipart-manifest=put',
|
||||||
|
response_dict=response)
|
||||||
status = response.get('status')
|
status = response.get('status')
|
||||||
assert (status >= 200 and status < 300)
|
assert (status >= 200 and status < 300)
|
||||||
progress_ln()
|
progress_ln()
|
||||||
|
|
||||||
|
|
||||||
def get_SLO(url, token):
|
def get_SLO(url, token):
|
||||||
response = dict()
|
response = dict()
|
||||||
headers, body = c.get_object(url, token, 'myobjects', 'assembly', http_conn=None,
|
headers, body = c.get_object(url, token, 'myobjects', 'assembly',
|
||||||
resp_chunk_size=1048576, query_string=None, response_dict=response, headers=None)
|
http_conn=None, resp_chunk_size=1048576,
|
||||||
|
query_string=None, response_dict=response,
|
||||||
|
headers=None)
|
||||||
|
|
||||||
i = 1
|
i = 1
|
||||||
for chunk in body:
|
for chunk in body:
|
||||||
oname = '/tmp/slo_chunk_%d' % i
|
oname = '/tmp/slo_chunk_%d' % i
|
||||||
f = open(oname,'r')
|
f = open(oname, 'r')
|
||||||
file_content=f.read()
|
file_content = f.read()
|
||||||
#print '%s %s' % (chunk[:10], file_content[:10])
|
# print '%s %s' % (chunk[:10], file_content[:10])
|
||||||
#print '%d %d' % (len(chunk), len(file_content))
|
# print '%d %d' % (len(chunk), len(file_content))
|
||||||
progress()
|
progress()
|
||||||
assert(chunk == file_content)
|
assert(chunk == file_content)
|
||||||
f.close()
|
f.close()
|
||||||
i=i+1
|
i = i + 1
|
||||||
progress_ln()
|
progress_ln()
|
||||||
|
|
||||||
|
|
||||||
def compare_slo_to_chunks(body):
|
def compare_slo_to_chunks(body):
|
||||||
i = 1
|
i = 1
|
||||||
for chunk in body:
|
for chunk in body:
|
||||||
if chunk:
|
if chunk:
|
||||||
if i<10:
|
if i < 10:
|
||||||
progress()
|
progress()
|
||||||
oname = '/tmp/slo_chunk_%d' % i
|
oname = '/tmp/slo_chunk_%d' % i
|
||||||
f = open(oname,'r')
|
f = open(oname, 'r')
|
||||||
file_content=f.read()
|
file_content = f.read()
|
||||||
#print '%s %s' % (chunk[:10], file_content[:10])
|
# print '%s %s' % (chunk[:10], file_content[:10])
|
||||||
#print '%d %d' % (len(chunk), len(file_content))
|
# print '%d %d' % (len(chunk), len(file_content))
|
||||||
assert(chunk == file_content)
|
assert(chunk == file_content)
|
||||||
f.close()
|
f.close()
|
||||||
i=i+1
|
i = i + 1
|
||||||
else:
|
else:
|
||||||
aux_content = ''
|
aux_content = ''
|
||||||
for j in range(1,4):
|
for j in range(1, 4):
|
||||||
oname = '/tmp/aux_file%d' % j
|
oname = '/tmp/aux_file%d' % j
|
||||||
f = open(oname,'r')
|
f = open(oname, 'r')
|
||||||
aux_content+=f.read()
|
aux_content += f.read()
|
||||||
f.close()
|
f.close()
|
||||||
assert(chunk == aux_content)
|
assert(chunk == aux_content)
|
||||||
progress_ln()
|
progress_ln()
|
||||||
|
|
||||||
|
|
||||||
def invoke_identity_on_get_SLO(url, token):
|
def invoke_identity_on_get_SLO(url, token):
|
||||||
metadata = {'X-Run-Storlet': IDENTITY_STORLET_NAME }
|
metadata = {'X-Run-Storlet': IDENTITY_STORLET_NAME}
|
||||||
response = dict()
|
response = dict()
|
||||||
headers, body = c.get_object(url, token,
|
headers, body = c.get_object(url, token,
|
||||||
'myobjects','assembly',
|
'myobjects', 'assembly',
|
||||||
query_string = None,
|
query_string=None,
|
||||||
response_dict=response,
|
response_dict=response,
|
||||||
resp_chunk_size=1048576,
|
resp_chunk_size=1048576,
|
||||||
headers=metadata)
|
headers=metadata)
|
||||||
compare_slo_to_chunks(body)
|
compare_slo_to_chunks(body)
|
||||||
|
|
||||||
|
|
||||||
def invoke_identity_on_get_SLO_double(url, token):
|
def invoke_identity_on_get_SLO_double(url, token):
|
||||||
metadata = {'X-Run-Storlet': IDENTITY_STORLET_NAME }
|
metadata = {'X-Run-Storlet': IDENTITY_STORLET_NAME}
|
||||||
response = dict()
|
response = dict()
|
||||||
headers, body = c.get_object(url, token,
|
headers, body = c.get_object(url, token,
|
||||||
'myobjects',
|
'myobjects',
|
||||||
'assembly',
|
'assembly',
|
||||||
query_string = 'double=true',
|
query_string='double=true',
|
||||||
response_dict=response,
|
response_dict=response,
|
||||||
resp_chunk_size=2048,
|
resp_chunk_size=2048,
|
||||||
headers=metadata)
|
headers=metadata)
|
||||||
@ -136,17 +167,17 @@ def invoke_identity_on_get_SLO_double(url, token):
|
|||||||
i = 1
|
i = 1
|
||||||
progress()
|
progress()
|
||||||
oname = '/tmp/slo_chunk_%d' % i
|
oname = '/tmp/slo_chunk_%d' % i
|
||||||
f = open(oname,'r')
|
f = open(oname, 'r')
|
||||||
file_content=f.read()
|
file_content = f.read()
|
||||||
|
|
||||||
j = 0 # Count chunks in file 1...1024
|
j = 0 # Count chunks in file 1...1024
|
||||||
for chunk in body:
|
for chunk in body:
|
||||||
file_fragment = file_content[j*1024:(j+1)*1024]
|
file_fragment = file_content[j * 1024:(j + 1) * 1024]
|
||||||
chunk_framgment_low = chunk[0:1024]
|
chunk_framgment_low = chunk[0:1024]
|
||||||
chunk_framgment_high = chunk[1024:2048]
|
chunk_framgment_high = chunk[1024:2048]
|
||||||
assert(chunk_framgment_low == file_fragment)
|
assert(chunk_framgment_low == file_fragment)
|
||||||
assert(chunk_framgment_high == file_fragment)
|
assert(chunk_framgment_high == file_fragment)
|
||||||
j = j +1
|
j = j + 1
|
||||||
if j == 1024:
|
if j == 1024:
|
||||||
i = i + 1
|
i = i + 1
|
||||||
if i == 10:
|
if i == 10:
|
||||||
@ -154,21 +185,22 @@ def invoke_identity_on_get_SLO_double(url, token):
|
|||||||
f.close()
|
f.close()
|
||||||
progress()
|
progress()
|
||||||
oname = '/tmp/slo_chunk_%d' % i
|
oname = '/tmp/slo_chunk_%d' % i
|
||||||
f = open(oname,'r')
|
f = open(oname, 'r')
|
||||||
file_content=f.read()
|
file_content = f.read()
|
||||||
j = 0
|
j = 0
|
||||||
assert i == 10
|
assert i == 10
|
||||||
progress_ln()
|
progress_ln()
|
||||||
|
|
||||||
|
|
||||||
def invoke_identity_on_partial_get_SLO(url, token):
|
def invoke_identity_on_partial_get_SLO(url, token):
|
||||||
metadata = {'X-Run-Storlet': IDENTITY_STORLET_NAME }
|
metadata = {'X-Run-Storlet': IDENTITY_STORLET_NAME}
|
||||||
for i in range(5):
|
for i in range(5):
|
||||||
progress()
|
progress()
|
||||||
response = dict()
|
response = dict()
|
||||||
headers, body = c.get_object(url, token,
|
headers, body = c.get_object(url, token,
|
||||||
'myobjects',
|
'myobjects',
|
||||||
'assembly',
|
'assembly',
|
||||||
query_string = None,
|
query_string=None,
|
||||||
response_dict=response,
|
response_dict=response,
|
||||||
resp_chunk_size=1048576,
|
resp_chunk_size=1048576,
|
||||||
headers=metadata)
|
headers=metadata)
|
||||||
@ -180,18 +212,19 @@ def invoke_identity_on_partial_get_SLO(url, token):
|
|||||||
break
|
break
|
||||||
progress_ln()
|
progress_ln()
|
||||||
|
|
||||||
|
|
||||||
# def delete_files():
|
# def delete_files():
|
||||||
# for i in range(1,4):
|
# for i in range(1,4):
|
||||||
# fname = '/tmp/aux_file%d' % i
|
# fname = '/tmp/aux_file%d' % i
|
||||||
# os.remove(fname)
|
# os.remove(fname)
|
||||||
|
|
||||||
|
|
||||||
def create_container(url, token, name):
|
def create_container(url, token, name):
|
||||||
response = dict()
|
response = dict()
|
||||||
c.put_container(url, token, name, headers=None, response_dict = response)
|
c.put_container(url, token, name, headers=None, response_dict=response)
|
||||||
status = response.get('status')
|
status = response.get('status')
|
||||||
assert (status >= 200 or status < 300)
|
assert (status >= 200 or status < 300)
|
||||||
|
|
||||||
|
|
||||||
def deploy_sloidentity_storlet(url, token):
|
def deploy_sloidentity_storlet(url, token):
|
||||||
progress()
|
progress()
|
||||||
response = dict()
|
response = dict()
|
||||||
@ -200,22 +233,24 @@ def deploy_sloidentity_storlet(url, token):
|
|||||||
assert (status >= 200 or status < 300)
|
assert (status >= 200 or status < 300)
|
||||||
|
|
||||||
progress()
|
progress()
|
||||||
put_storlet_object( url, token,
|
put_storlet_object(url, token,
|
||||||
SLOIDENTITY_STORLET_NAME,
|
SLOIDENTITY_STORLET_NAME,
|
||||||
SLOIDENTITY_PATH_TO_BUNDLE,
|
SLOIDENTITY_PATH_TO_BUNDLE,
|
||||||
'',
|
'',
|
||||||
'com.ibm.storlet.sloidentity.SLOIdentityStorlet')
|
'com.ibm.storlet.sloidentity.SLOIdentityStorlet')
|
||||||
progress_ln()
|
progress_ln()
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
os_options = {'tenant_name': ACCOUNT}
|
os_options = {'tenant_name': ACCOUNT}
|
||||||
url, token = c.get_auth( 'http://' + AUTH_IP + ":"
|
url, token = c.get_auth('http://' + AUTH_IP + ":"
|
||||||
+ AUTH_PORT + '/v2.0',
|
+ AUTH_PORT + '/v2.0',
|
||||||
ACCOUNT + ':' + USER_NAME,
|
ACCOUNT + ':' + USER_NAME,
|
||||||
PASSWORD,
|
PASSWORD,
|
||||||
os_options = os_options,
|
os_options=os_options,
|
||||||
auth_version = '2.0' )
|
auth_version='2.0')
|
||||||
# print('Creating containers for auxiliary files')
|
# print('Creating containers for auxiliary files')
|
||||||
create_container(url, token, 'myobjects')
|
create_container(url, token, 'myobjects')
|
||||||
create_container(url, token, 'container1')
|
create_container(url, token, 'container1')
|
||||||
@ -234,8 +269,8 @@ def main():
|
|||||||
progress_msg("Invoking storlet on SLO in GET with double")
|
progress_msg("Invoking storlet on SLO in GET with double")
|
||||||
invoke_identity_on_get_SLO_double(url, token)
|
invoke_identity_on_get_SLO_double(url, token)
|
||||||
|
|
||||||
#progress_msg("Invoking storlet on SLO in partial GET")
|
# progress_msg("Invoking storlet on SLO in partial GET")
|
||||||
#invoke_identity_on_partial_get_SLO(url, token)
|
# invoke_identity_on_partial_get_SLO(url, token)
|
||||||
delete_local_chunks()
|
delete_local_chunks()
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
@ -17,52 +17,61 @@ Limitations under the License.
|
|||||||
@author: gilv / cdoron / evgenyl
|
@author: gilv / cdoron / evgenyl
|
||||||
'''
|
'''
|
||||||
|
|
||||||
from sys_test_params import *
|
from storlets_test_utils import put_file_as_storlet_input_object
|
||||||
|
from storlets_test_utils import put_storlet_object
|
||||||
from swiftclient import client as c
|
from swiftclient import client as c
|
||||||
|
from sys_test_params import ACCOUNT
|
||||||
|
from sys_test_params import AUTH_IP
|
||||||
|
from sys_test_params import AUTH_PORT
|
||||||
|
from sys_test_params import PASSWORD
|
||||||
|
from sys_test_params import USER_NAME
|
||||||
|
|
||||||
from storlets_test_utils import put_storlet_object, \
|
EXECDEP_PATH_TO_BUNDLE = '../StorletSamples/ExecDepStorlet/bin/'
|
||||||
put_file_as_storlet_input_object
|
EXECDEP_STORLET_NAME = 'execdepstorlet-1.0.jar'
|
||||||
|
EXECDEP_STORLET_LOG_NAME = 'execdepstorlet-1.0.log'
|
||||||
EXECDEP_PATH_TO_BUNDLE ='../StorletSamples/ExecDepStorlet/bin/'
|
|
||||||
EXECDEP_STORLET_NAME='execdepstorlet-1.0.jar'
|
|
||||||
EXECDEP_STORLET_LOG_NAME='execdepstorlet-1.0.log'
|
|
||||||
EXECDEP_JUNK_FILE = 'junk.txt'
|
EXECDEP_JUNK_FILE = 'junk.txt'
|
||||||
EXECDEP_DEPS_NAMES=['get42']
|
EXECDEP_DEPS_NAMES = ['get42']
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def put_storlet_executable_dependencies(url, token):
|
def put_storlet_executable_dependencies(url, token):
|
||||||
resp = dict()
|
resp = dict()
|
||||||
for d in EXECDEP_DEPS_NAMES:
|
for d in EXECDEP_DEPS_NAMES:
|
||||||
metadata = {'X-Object-Meta-Storlet-Dependency-Version': '1',
|
metadata = {'X-Object-Meta-Storlet-Dependency-Version': '1',
|
||||||
'X-Object-Meta-Storlet-Dependency-Permissions': '0755' }
|
'X-Object-Meta-Storlet-Dependency-Permissions': '0755'}
|
||||||
|
|
||||||
f = open('%s/%s' %(EXECDEP_PATH_TO_BUNDLE, d),'r')
|
f = open('%s/%s' % (EXECDEP_PATH_TO_BUNDLE, d), 'r')
|
||||||
c.put_object(url, token, 'dependency', d, f,
|
c.put_object(url, token, 'dependency', d, f,
|
||||||
content_type = "application/octet-stream",
|
content_type="application/octet-stream",
|
||||||
headers = metadata,
|
headers=metadata,
|
||||||
response_dict = resp)
|
response_dict=resp)
|
||||||
f.close()
|
f.close()
|
||||||
status = resp.get('status')
|
status = resp.get('status')
|
||||||
assert (status == 200 or status == 201)
|
assert (status == 200 or status == 201)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
def deploy_storlet(url,token):
|
|
||||||
#No need to create containers every time
|
|
||||||
#put_storlet_containers(url, token)
|
def deploy_storlet(url, token):
|
||||||
put_storlet_object( url, token,
|
# No need to create containers every time
|
||||||
EXECDEP_STORLET_NAME,
|
# put_storlet_containers(url, token)
|
||||||
EXECDEP_PATH_TO_BUNDLE,
|
put_storlet_object(url, token,
|
||||||
','.join( str(x) for x in EXECDEP_DEPS_NAMES),
|
EXECDEP_STORLET_NAME,
|
||||||
'com.ibm.storlet.execdep.ExecDepStorlet')
|
EXECDEP_PATH_TO_BUNDLE,
|
||||||
|
','.join(str(x) for x in EXECDEP_DEPS_NAMES),
|
||||||
|
'com.ibm.storlet.execdep.ExecDepStorlet')
|
||||||
put_storlet_executable_dependencies(url, token)
|
put_storlet_executable_dependencies(url, token)
|
||||||
put_file_as_storlet_input_object(url,
|
put_file_as_storlet_input_object(url,
|
||||||
token,
|
token,
|
||||||
EXECDEP_PATH_TO_BUNDLE,
|
EXECDEP_PATH_TO_BUNDLE,
|
||||||
EXECDEP_JUNK_FILE )
|
EXECDEP_JUNK_FILE)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def invoke_storlet(url, token):
|
def invoke_storlet(url, token):
|
||||||
metadata = {'X-Run-Storlet': EXECDEP_STORLET_NAME }
|
metadata = {'X-Run-Storlet': EXECDEP_STORLET_NAME}
|
||||||
resp = dict()
|
resp = dict()
|
||||||
resp_headers, gf = c.get_object(url, token,
|
resp_headers, gf = c.get_object(url, token,
|
||||||
'myobjects',
|
'myobjects',
|
||||||
@ -74,21 +83,22 @@ def invoke_storlet(url, token):
|
|||||||
assert resp_headers['x-object-meta-depend-ret-code'] == '42'
|
assert resp_headers['x-object-meta-depend-ret-code'] == '42'
|
||||||
assert resp['status'] == 200
|
assert resp['status'] == 200
|
||||||
|
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
os_options = {'tenant_name': ACCOUNT}
|
os_options = {'tenant_name': ACCOUNT}
|
||||||
url, token = c.get_auth( 'http://' + AUTH_IP + ":"
|
url, token = c.get_auth('http://' + AUTH_IP + ":"
|
||||||
+ AUTH_PORT + '/v2.0',
|
+ AUTH_PORT + '/v2.0',
|
||||||
ACCOUNT + ':' + USER_NAME,
|
ACCOUNT + ':' + USER_NAME,
|
||||||
PASSWORD,
|
PASSWORD,
|
||||||
os_options = os_options,
|
os_options=os_options,
|
||||||
auth_version = '2.0' )
|
auth_version='2.0')
|
||||||
|
|
||||||
print 'Deploying ExecDep storlet and dependencies'
|
print('Deploying ExecDep storlet and dependencies')
|
||||||
deploy_storlet(url, token)
|
deploy_storlet(url, token)
|
||||||
|
|
||||||
print "Invoking ExecDep storlet"
|
print("Invoking ExecDep storlet")
|
||||||
invoke_storlet(url, token)
|
invoke_storlet(url, token)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
@ -13,51 +13,58 @@ See the License for the specific language governing permissions and
|
|||||||
Limitations under the License.
|
Limitations under the License.
|
||||||
-------------------------------------------------------------------------'''
|
-------------------------------------------------------------------------'''
|
||||||
|
|
||||||
import os
|
|
||||||
import json
|
|
||||||
import random
|
import random
|
||||||
import string
|
import string
|
||||||
from sys_test_params import *
|
|
||||||
from swiftclient import client as c
|
from swiftclient import client as c
|
||||||
|
from sys_test_params import ACCOUNT
|
||||||
|
from sys_test_params import AUTH_IP
|
||||||
|
from sys_test_params import AUTH_PORT
|
||||||
|
from sys_test_params import PASSWORD
|
||||||
|
from sys_test_params import USER_NAME
|
||||||
|
|
||||||
from storlets_test_utils import put_storlet_containers, put_storlet_object, \
|
from storlets_test_utils import put_storlet_object
|
||||||
progress, progress_ln, progress_msg
|
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
# Test Constants
|
# Test Constants
|
||||||
HALF_PATH_TO_BUNDLE ='../StorletSamples/HalfStorlet/bin/'
|
HALF_PATH_TO_BUNDLE = '../StorletSamples/HalfStorlet/bin/'
|
||||||
HALF_STORLET_NAME='halfstorlet-1.0.jar'
|
HALF_STORLET_NAME = 'halfstorlet-1.0.jar'
|
||||||
HALF_SOURCE_FILE = 'source.txt'
|
HALF_SOURCE_FILE = 'source.txt'
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def put_storlet_input_object(url, token):
|
def put_storlet_input_object(url, token):
|
||||||
resp = dict()
|
resp = dict()
|
||||||
metadata = {'X-Object-Meta-Testkey':'tester'}
|
metadata = {'X-Object-Meta-Testkey': 'tester'}
|
||||||
f = open('%s/%s' %(HALF_PATH_TO_BUNDLE, HALF_SOURCE_FILE),'r')
|
f = open('%s/%s' % (HALF_PATH_TO_BUNDLE, HALF_SOURCE_FILE), 'r')
|
||||||
c.put_object(url, token, 'myobjects', HALF_SOURCE_FILE, f,
|
c.put_object(url, token, 'myobjects', HALF_SOURCE_FILE, f,
|
||||||
content_type = "application/octet-stream",
|
content_type="application/octet-stream",
|
||||||
headers = metadata,
|
headers=metadata,
|
||||||
response_dict = resp)
|
response_dict=resp)
|
||||||
f.close()
|
f.close()
|
||||||
status = resp.get('status')
|
status = resp.get('status')
|
||||||
assert (status == 200 or status == 201)
|
assert (status == 200 or status == 201)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
def deploy_storlet(url,token):
|
|
||||||
#No need to create containers every time
|
|
||||||
#put_storlet_containers(url, token)
|
def deploy_storlet(url, token):
|
||||||
put_storlet_object( url, token,
|
# No need to create containers every time
|
||||||
HALF_STORLET_NAME,
|
# put_storlet_containers(url, token)
|
||||||
HALF_PATH_TO_BUNDLE,
|
put_storlet_object(url, token,
|
||||||
'',
|
HALF_STORLET_NAME,
|
||||||
'com.ibm.storlet.half.HalfStorlet')
|
HALF_PATH_TO_BUNDLE,
|
||||||
put_storlet_input_object( url, token )
|
'',
|
||||||
|
'com.ibm.storlet.half.HalfStorlet')
|
||||||
|
put_storlet_input_object(url, token)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
def invoke_storlet(url, token, op, params = None, global_params = None, headers = None):
|
|
||||||
if params != None:
|
def invoke_storlet(url, token, op, params=None, global_params=None,
|
||||||
querystring=''
|
headers=None):
|
||||||
|
if params is not None:
|
||||||
|
querystring = ''
|
||||||
for key in params:
|
for key in params:
|
||||||
querystring += '%s=%s,' % (key, params[key])
|
querystring += '%s=%s,' % (key, params[key])
|
||||||
querystring = querystring[:-1]
|
querystring = querystring[:-1]
|
||||||
@ -70,74 +77,76 @@ def invoke_storlet(url, token, op, params = None, global_params = None, headers
|
|||||||
|
|
||||||
if op == 'GET':
|
if op == 'GET':
|
||||||
# Get original object
|
# Get original object
|
||||||
original_headers, original_content = c.get_object(url, token,
|
original_headers, original_content = \
|
||||||
'myobjects',
|
c.get_object(url, token, 'myobjects', HALF_SOURCE_FILE,
|
||||||
HALF_SOURCE_FILE,
|
response_dict=dict())
|
||||||
response_dict=dict())
|
# print original_headers
|
||||||
#print original_headers
|
|
||||||
file_length = int(original_headers['content-length'])
|
file_length = int(original_headers['content-length'])
|
||||||
processed_headers, returned_content = c.get_object(url, token,
|
processed_headers, returned_content = \
|
||||||
'myobjects',
|
c.get_object(url, token, 'myobjects', HALF_SOURCE_FILE,
|
||||||
HALF_SOURCE_FILE,
|
query_string=querystring, response_dict=dict(),
|
||||||
query_string = querystring,
|
headers=metadata, resp_chunk_size=file_length)
|
||||||
response_dict=dict(),
|
|
||||||
headers=metadata,
|
|
||||||
resp_chunk_size = file_length)
|
|
||||||
processed_content = ''
|
processed_content = ''
|
||||||
for chunk in returned_content:
|
for chunk in returned_content:
|
||||||
if chunk:
|
if chunk:
|
||||||
processed_content+=chunk
|
processed_content += chunk
|
||||||
|
|
||||||
assert(original_headers['X-Object-Meta-Testkey'.lower()] == processed_headers['X-Object-Meta-Testkey'.lower()])
|
assert(original_headers['X-Object-Meta-Testkey'.lower()] ==
|
||||||
|
processed_headers['X-Object-Meta-Testkey'.lower()])
|
||||||
return processed_content
|
return processed_content
|
||||||
|
|
||||||
|
|
||||||
if op == 'PUT':
|
if op == 'PUT':
|
||||||
# PUT a random file
|
# PUT a random file
|
||||||
response = dict()
|
response = dict()
|
||||||
uploaded_content = ''.join(random.choice(string.ascii_uppercase + string.digits) for _ in range(1024))
|
uploaded_content = ''.join(random.choice(string.ascii_uppercase +
|
||||||
random_md = ''.join(random.choice(string.ascii_uppercase + string.digits) for _ in range(32))
|
string.digits) for _ in range(1024))
|
||||||
#content_length = 1024
|
random_md = ''.join(random.choice(string.ascii_uppercase +
|
||||||
|
string.digits) for _ in range(32))
|
||||||
|
# content_length = 1024
|
||||||
content_length = None
|
content_length = None
|
||||||
headers = {'X-Run-Storlet': HALF_STORLET_NAME,
|
headers = {'X-Run-Storlet': HALF_STORLET_NAME,
|
||||||
'X-Object-Meta-Testkey' : random_md }
|
'X-Object-Meta-Testkey': random_md}
|
||||||
c.put_object(url, token, 'myobjects', 'half_random_source', uploaded_content,
|
c.put_object(url, token, 'myobjects', 'half_random_source',
|
||||||
content_length, None, None, "application/octet-stream",
|
uploaded_content, content_length, None, None,
|
||||||
headers, None, None, querystring, response)
|
"application/octet-stream", headers, None, None,
|
||||||
resp_headers, saved_content = c.get_object(url, token,
|
querystring, response)
|
||||||
'myobjects',
|
resp_headers, saved_content = c.get_object(url, token, 'myobjects',
|
||||||
'half_random_source',
|
'half_random_source',
|
||||||
response_dict=dict())
|
response_dict=dict())
|
||||||
|
|
||||||
if params != None and params.get('double',None) == 'true':
|
if params is not None and params.get('double', None) == 'true':
|
||||||
assert(uploaded_content==saved_content[:1024])
|
assert(uploaded_content == saved_content[:1024])
|
||||||
assert(uploaded_content==saved_content[1024:])
|
assert(uploaded_content == saved_content[1024:])
|
||||||
else:
|
else:
|
||||||
assert(uploaded_content == saved_content)
|
assert(uploaded_content == saved_content)
|
||||||
|
|
||||||
if params != None and params.get('execute',None) != None:
|
if params is not None and params.get('execute', None) is not None:
|
||||||
assert(resp_headers['X-Object-Meta-Execution result'.lower()] == '42')
|
assert(resp_headers['X-Object-Meta-Execution result'.lower()] ==
|
||||||
|
'42')
|
||||||
|
|
||||||
assert(resp_headers['X-Object-Meta-Testkey'.lower()] == random_md)
|
assert(resp_headers['X-Object-Meta-Testkey'.lower()] == random_md)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
os_options = {'tenant_name': ACCOUNT}
|
os_options = {'tenant_name': ACCOUNT}
|
||||||
url, token = c.get_auth( 'http://' + AUTH_IP + ":"
|
url, token = c.get_auth('http://' + AUTH_IP + ":"
|
||||||
+ AUTH_PORT + '/v2.0',
|
+ AUTH_PORT + '/v2.0',
|
||||||
ACCOUNT + ':' + USER_NAME,
|
ACCOUNT + ':' + USER_NAME,
|
||||||
PASSWORD,
|
PASSWORD,
|
||||||
os_options = os_options,
|
os_options=os_options,
|
||||||
auth_version = '2.0' )
|
auth_version='2.0')
|
||||||
|
|
||||||
print 'Deploying Half storlet and dependencies'
|
print('Deploying Half storlet and dependencies')
|
||||||
|
|
||||||
deploy_storlet(url, token)
|
deploy_storlet(url, token)
|
||||||
|
|
||||||
print "Invoking Half storlet on GET"
|
print("Invoking Half storlet on GET")
|
||||||
assert (invoke_storlet(url, token,'GET') == 'acegikmn')
|
assert (invoke_storlet(url, token, 'GET') == 'acegikmn')
|
||||||
print "Invoking Half storlet on GET with byte ranges"
|
print("Invoking Half storlet on GET with byte ranges")
|
||||||
assert (invoke_storlet(url, token,'GET', headers = {'range': 'bytes=5-10'}) == 'fhj')
|
assert (invoke_storlet(url, token, 'GET',
|
||||||
|
headers={'range': 'bytes=5-10'}) == 'fhj')
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
@ -14,80 +14,92 @@ Limitations under the License.
|
|||||||
-------------------------------------------------------------------------'''
|
-------------------------------------------------------------------------'''
|
||||||
|
|
||||||
import os
|
import os
|
||||||
import json
|
|
||||||
import random
|
import random
|
||||||
import string
|
import string
|
||||||
from sys_test_params import *
|
|
||||||
from swiftclient import client as c
|
from swiftclient import client as c
|
||||||
|
from sys_test_params import ACCOUNT
|
||||||
|
from sys_test_params import AUTH_IP
|
||||||
|
from sys_test_params import AUTH_PORT
|
||||||
|
from sys_test_params import PASSWORD
|
||||||
|
from sys_test_params import USER_NAME
|
||||||
|
|
||||||
from storlets_test_utils import put_storlet_containers, put_storlet_object, \
|
from storlets_test_utils import progress
|
||||||
progress, progress_ln, progress_msg
|
from storlets_test_utils import progress_ln
|
||||||
|
from storlets_test_utils import progress_msg
|
||||||
|
from storlets_test_utils import put_storlet_object
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
# Test Constants
|
# Test Constants
|
||||||
IDENTITY_PATH_TO_BUNDLE ='../StorletSamples/IdentityStorlet/bin/'
|
IDENTITY_PATH_TO_BUNDLE = '../StorletSamples/IdentityStorlet/bin/'
|
||||||
IDENTITY_STORLET_NAME='identitystorlet-1.0.jar'
|
IDENTITY_STORLET_NAME = 'identitystorlet-1.0.jar'
|
||||||
IDENTITY_STORLET_LOG_NAME='identitystorlet-1.0.log'
|
IDENTITY_STORLET_LOG_NAME = 'identitystorlet-1.0.log'
|
||||||
IDENTITY_SOURCE_FILE = 'source.txt'
|
IDENTITY_SOURCE_FILE = 'source.txt'
|
||||||
IDENTITY_DEPS_NAMES=['get42']
|
IDENTITY_DEPS_NAMES = ['get42']
|
||||||
|
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def put_storlet_executable_dependencies(url, token):
|
def put_storlet_executable_dependencies(url, token):
|
||||||
resp = dict()
|
resp = dict()
|
||||||
for d in IDENTITY_DEPS_NAMES:
|
for d in IDENTITY_DEPS_NAMES:
|
||||||
metadata = {'X-Object-Meta-Storlet-Dependency-Version': '1',
|
metadata = {'X-Object-Meta-Storlet-Dependency-Version': '1',
|
||||||
'X-Object-Meta-Storlet-Dependency-Permissions': '0755' }
|
'X-Object-Meta-Storlet-Dependency-Permissions': '0755'}
|
||||||
|
|
||||||
f = open('%s/%s' %(IDENTITY_PATH_TO_BUNDLE, d),'r')
|
f = open('%s/%s' % (IDENTITY_PATH_TO_BUNDLE, d), 'r')
|
||||||
c.put_object(url, token, 'dependency', d, f,
|
c.put_object(url, token, 'dependency', d, f,
|
||||||
content_type = "application/octet-stream",
|
content_type="application/octet-stream",
|
||||||
headers = metadata,
|
headers=metadata,
|
||||||
response_dict = resp)
|
response_dict=resp)
|
||||||
f.close()
|
f.close()
|
||||||
status = resp.get('status')
|
status = resp.get('status')
|
||||||
assert (status == 200 or status == 201)
|
assert (status == 200 or status == 201)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def put_storlet_input_object(url, token):
|
def put_storlet_input_object(url, token):
|
||||||
resp = dict()
|
resp = dict()
|
||||||
metadata = {'X-Object-Meta-Testkey':'tester'}
|
metadata = {'X-Object-Meta-Testkey': 'tester'}
|
||||||
f = open('%s/%s' %(IDENTITY_PATH_TO_BUNDLE, IDENTITY_SOURCE_FILE),'r')
|
f = open('%s/%s' % (IDENTITY_PATH_TO_BUNDLE, IDENTITY_SOURCE_FILE), 'r')
|
||||||
c.put_object(url, token, 'myobjects', IDENTITY_SOURCE_FILE, f,
|
c.put_object(url, token, 'myobjects', IDENTITY_SOURCE_FILE, f,
|
||||||
content_type = "application/octet-stream",
|
content_type="application/octet-stream",
|
||||||
headers = metadata,
|
headers=metadata,
|
||||||
response_dict = resp)
|
response_dict=resp)
|
||||||
f.close()
|
f.close()
|
||||||
status = resp.get('status')
|
status = resp.get('status')
|
||||||
assert (status == 200 or status == 201)
|
assert (status == 200 or status == 201)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
def deploy_storlet(url,token):
|
|
||||||
#No need to create containers every time
|
|
||||||
#put_storlet_containers(url, token)
|
def deploy_storlet(url, token):
|
||||||
put_storlet_object( url, token,
|
# No need to create containers every time
|
||||||
IDENTITY_STORLET_NAME,
|
# put_storlet_containers(url, token)
|
||||||
IDENTITY_PATH_TO_BUNDLE,
|
put_storlet_object(url, token,
|
||||||
','.join( str(x) for x in IDENTITY_DEPS_NAMES),
|
IDENTITY_STORLET_NAME,
|
||||||
'com.ibm.storlet.identity.IdentityStorlet')
|
IDENTITY_PATH_TO_BUNDLE,
|
||||||
|
','.join(str(x) for x in IDENTITY_DEPS_NAMES),
|
||||||
|
'com.ibm.storlet.identity.IdentityStorlet')
|
||||||
put_storlet_executable_dependencies(url, token)
|
put_storlet_executable_dependencies(url, token)
|
||||||
put_storlet_input_object( url, token )
|
put_storlet_input_object(url, token)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def invoke_storlet_on_1GB_file(url, token):
|
def invoke_storlet_on_1GB_file(url, token):
|
||||||
GBFile = open('/tmp/1GB_file','w')
|
GBFile = open('/tmp/1GB_file', 'w')
|
||||||
for _ in range(128):
|
for _ in range(128):
|
||||||
progress()
|
progress()
|
||||||
uploaded_content = ''.join('1' for _ in range(8*1024*1024))
|
uploaded_content = ''.join('1' for _ in range(8 * 1024 * 1024))
|
||||||
GBFile.write(uploaded_content)
|
GBFile.write(uploaded_content)
|
||||||
GBFile.close()
|
GBFile.close()
|
||||||
|
|
||||||
headers = {'X-Run-Storlet': IDENTITY_STORLET_NAME }
|
headers = {'X-Run-Storlet': IDENTITY_STORLET_NAME}
|
||||||
GBFile = open('/tmp/1GB_file','r')
|
GBFile = open('/tmp/1GB_file', 'r')
|
||||||
response=dict()
|
response = dict()
|
||||||
progress()
|
progress()
|
||||||
c.put_object(url, token, 'myobjects', '1GBFile', GBFile,
|
c.put_object(url, token, 'myobjects', '1GBFile', GBFile,
|
||||||
1024*1024*1024, None, None, "application/octet-stream",
|
1024 * 1024 * 1024, None, None, "application/octet-stream",
|
||||||
headers, None, None, None, response)
|
headers, None, None, None, response)
|
||||||
progress()
|
progress()
|
||||||
status = response.get('status')
|
status = response.get('status')
|
||||||
@ -97,9 +109,9 @@ def invoke_storlet_on_1GB_file(url, token):
|
|||||||
progress_ln()
|
progress_ln()
|
||||||
|
|
||||||
|
|
||||||
def invoke_storlet(url, token, op, params = None, global_params = None):
|
def invoke_storlet(url, token, op, params=None, global_params=None):
|
||||||
if params != None:
|
if params is not None:
|
||||||
querystring=''
|
querystring = ''
|
||||||
for key in params:
|
for key in params:
|
||||||
querystring += '%s=%s,' % (key, params[key])
|
querystring += '%s=%s,' % (key, params[key])
|
||||||
querystring = querystring[:-1]
|
querystring = querystring[:-1]
|
||||||
@ -109,89 +121,92 @@ def invoke_storlet(url, token, op, params = None, global_params = None):
|
|||||||
metadata = {'X-Run-Storlet': IDENTITY_STORLET_NAME}
|
metadata = {'X-Run-Storlet': IDENTITY_STORLET_NAME}
|
||||||
if op == 'GET':
|
if op == 'GET':
|
||||||
# Get original object
|
# Get original object
|
||||||
original_headers, original_content = c.get_object(url, token,
|
original_headers, original_content = \
|
||||||
'myobjects',
|
c.get_object(url, token, 'myobjects', IDENTITY_SOURCE_FILE,
|
||||||
IDENTITY_SOURCE_FILE,
|
response_dict=dict())
|
||||||
response_dict=dict())
|
# print original_headers
|
||||||
#print original_headers
|
|
||||||
file_length = int(original_headers['content-length'])
|
file_length = int(original_headers['content-length'])
|
||||||
processed_headers, returned_content = c.get_object(url, token,
|
processed_headers, returned_content = \
|
||||||
'myobjects',
|
c.get_object(url, token, 'myobjects', IDENTITY_SOURCE_FILE,
|
||||||
IDENTITY_SOURCE_FILE,
|
query_string=querystring, response_dict=dict(),
|
||||||
query_string = querystring,
|
headers=metadata, resp_chunk_size=file_length)
|
||||||
response_dict=dict(),
|
|
||||||
headers=metadata,
|
|
||||||
resp_chunk_size = file_length)
|
|
||||||
processed_content = ''
|
processed_content = ''
|
||||||
for chunk in returned_content:
|
for chunk in returned_content:
|
||||||
if chunk:
|
if chunk:
|
||||||
processed_content+=chunk
|
processed_content += chunk
|
||||||
|
|
||||||
if params != None and params.get('execute',None) != None:
|
if params is not None and params.get('execute', None) is not None:
|
||||||
assert(processed_headers['X-Object-Meta-Execution result'.lower()] == '42')
|
assert(processed_headers['X-Object-Meta-Execution result'.lower()]
|
||||||
if params != None and params.get('double',None) == 'true':
|
== '42')
|
||||||
assert(original_content==processed_content[:file_length])
|
if params is not None and params.get('double', None) == 'true':
|
||||||
assert(original_content==processed_content[file_length:])
|
assert(original_content == processed_content[:file_length])
|
||||||
|
assert(original_content == processed_content[file_length:])
|
||||||
else:
|
else:
|
||||||
assert(original_content == processed_content)
|
assert(original_content == processed_content)
|
||||||
assert(original_headers['X-Object-Meta-Testkey'.lower()] == processed_headers['X-Object-Meta-Testkey'.lower()])
|
assert(original_headers['X-Object-Meta-Testkey'.lower()] ==
|
||||||
|
processed_headers['X-Object-Meta-Testkey'.lower()])
|
||||||
|
|
||||||
if op == 'PUT':
|
if op == 'PUT':
|
||||||
# PUT a random file
|
# PUT a random file
|
||||||
response = dict()
|
response = dict()
|
||||||
uploaded_content = ''.join(random.choice(string.ascii_uppercase + string.digits) for _ in range(1024))
|
uploaded_content = ''.join(random.choice(string.ascii_uppercase +
|
||||||
random_md = ''.join(random.choice(string.ascii_uppercase + string.digits) for _ in range(32))
|
string.digits) for _ in range(1024))
|
||||||
#content_length = 1024
|
random_md = ''.join(random.choice(string.ascii_uppercase +
|
||||||
|
string.digits) for _ in range(32))
|
||||||
content_length = None
|
content_length = None
|
||||||
headers = {'X-Run-Storlet': IDENTITY_STORLET_NAME,
|
headers = {'X-Run-Storlet': IDENTITY_STORLET_NAME,
|
||||||
'X-Object-Meta-Testkey' : random_md }
|
'X-Object-Meta-Testkey': random_md}
|
||||||
c.put_object(url, token, 'myobjects', 'identity_random_source', uploaded_content,
|
c.put_object(url, token, 'myobjects', 'identity_random_source',
|
||||||
content_length, None, None, "application/octet-stream",
|
uploaded_content, content_length, None, None,
|
||||||
headers, None, None, querystring, response)
|
"application/octet-stream", headers, None, None,
|
||||||
|
querystring, response)
|
||||||
resp_headers, saved_content = c.get_object(url, token,
|
resp_headers, saved_content = c.get_object(url, token,
|
||||||
'myobjects',
|
'myobjects',
|
||||||
'identity_random_source',
|
'identity_random_source',
|
||||||
response_dict=dict())
|
response_dict=dict())
|
||||||
|
|
||||||
if params != None and params.get('double',None) == 'true':
|
if params is not None and params.get('double', None) == 'true':
|
||||||
assert(uploaded_content==saved_content[:1024])
|
assert(uploaded_content == saved_content[:1024])
|
||||||
assert(uploaded_content==saved_content[1024:])
|
assert(uploaded_content == saved_content[1024:])
|
||||||
else:
|
else:
|
||||||
assert(uploaded_content == saved_content)
|
assert(uploaded_content == saved_content)
|
||||||
|
|
||||||
if params != None and params.get('execute',None) != None:
|
if params is not None and params.get('execute', None) is not None:
|
||||||
assert(resp_headers['X-Object-Meta-Execution result'.lower()] == '42')
|
assert(resp_headers['X-Object-Meta-Execution result'.lower()] ==
|
||||||
|
'42')
|
||||||
|
|
||||||
assert(resp_headers['X-Object-Meta-Testkey'.lower()] == random_md)
|
assert(resp_headers['X-Object-Meta-Testkey'.lower()] == random_md)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
os_options = {'tenant_name': ACCOUNT}
|
os_options = {'tenant_name': ACCOUNT}
|
||||||
url, token = c.get_auth( 'http://' + AUTH_IP + ":"
|
url, token = c.get_auth('http://' + AUTH_IP + ":" +
|
||||||
+ AUTH_PORT + '/v2.0',
|
AUTH_PORT + '/v2.0',
|
||||||
ACCOUNT + ':' + USER_NAME,
|
ACCOUNT + ':' + USER_NAME,
|
||||||
PASSWORD,
|
PASSWORD,
|
||||||
os_options = os_options,
|
os_options=os_options,
|
||||||
auth_version = '2.0' )
|
auth_version='2.0')
|
||||||
|
|
||||||
print 'Deploying Identity storlet and dependencies'
|
print('Deploying Identity storlet and dependencies')
|
||||||
|
|
||||||
deploy_storlet(url, token)
|
deploy_storlet(url, token)
|
||||||
|
|
||||||
print "Invoking Identity storlet on PUT"
|
print("Invoking Identity storlet on PUT")
|
||||||
invoke_storlet(url, token,'PUT')
|
invoke_storlet(url, token, 'PUT')
|
||||||
progress_msg("Invoking Identity storlet on 1GB file PUT")
|
progress_msg("Invoking Identity storlet on 1GB file PUT")
|
||||||
invoke_storlet_on_1GB_file(url, token)
|
invoke_storlet_on_1GB_file(url, token)
|
||||||
print "Invoking Identity storlet on PUT with execution of dependency"
|
print("Invoking Identity storlet on PUT with execution of dependency")
|
||||||
invoke_storlet(url, token,'PUT', {'execute' : 'true'})
|
invoke_storlet(url, token, 'PUT', {'execute': 'true'})
|
||||||
print "Invoking Identity storlet on PUT with double"
|
print("Invoking Identity storlet on PUT with double")
|
||||||
invoke_storlet(url, token,'PUT', {'double' : 'true'})
|
invoke_storlet(url, token, 'PUT', {'double': 'true'})
|
||||||
print "Invoking Identity storlet on GET"
|
print("Invoking Identity storlet on GET")
|
||||||
invoke_storlet(url, token,'GET')
|
invoke_storlet(url, token, 'GET')
|
||||||
print "Invoking Identity storlet on GET with double"
|
print("Invoking Identity storlet on GET with double")
|
||||||
invoke_storlet(url, token,'GET', {'double' : 'true'})
|
invoke_storlet(url, token, 'GET', {'double': 'true'})
|
||||||
print "Invoking Identity storlet on GET with execution of dependency"
|
print("Invoking Identity storlet on GET with execution of dependency")
|
||||||
invoke_storlet(url, token,'GET',{'execute' : 'true'})
|
invoke_storlet(url, token, 'GET', {'execute': 'true'})
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
@ -13,71 +13,76 @@ See the License for the specific language governing permissions and
|
|||||||
Limitations under the License.
|
Limitations under the License.
|
||||||
-------------------------------------------------------------------------'''
|
-------------------------------------------------------------------------'''
|
||||||
|
|
||||||
import json
|
|
||||||
import random
|
|
||||||
import string
|
|
||||||
from sys_test_params import *
|
|
||||||
from swiftclient import client as c
|
from swiftclient import client as c
|
||||||
|
from sys_test_params import ACCOUNT
|
||||||
|
from sys_test_params import AUTH_IP
|
||||||
|
from sys_test_params import AUTH_PORT
|
||||||
|
from sys_test_params import PASSWORD
|
||||||
|
from sys_test_params import USER_NAME
|
||||||
|
|
||||||
from storlets_test_utils import put_storlet_containers, put_storlet_object
|
from storlets_test_utils import put_storlet_object
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
# Test Constants
|
# Test Constants
|
||||||
PATH_TO_BUNDLE ='../StorletSamples/TestMetadataStorlet/bin/'
|
PATH_TO_BUNDLE = '../StorletSamples/TestMetadataStorlet/bin/'
|
||||||
STORLET_NAME='testmetadatastorlet-1.0.jar'
|
STORLET_NAME = 'testmetadatastorlet-1.0.jar'
|
||||||
STORLET_LOG_NAME='testmetadatastorlet-1.0.log'
|
STORLET_LOG_NAME = 'testmetadatastorlet-1.0.log'
|
||||||
SOURCE_FILE = 'source.txt'
|
SOURCE_FILE = 'source.txt'
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def put_storlet_input_object(url, token):
|
def put_storlet_input_object(url, token):
|
||||||
resp = dict()
|
resp = dict()
|
||||||
metadata = {'X-Object-Meta-key1':'1',
|
metadata = {'X-Object-Meta-key1': '1',
|
||||||
'X-Object-Meta-key2':'2',
|
'X-Object-Meta-key2': '2',
|
||||||
'X-Object-Meta-key3':'3',
|
'X-Object-Meta-key3': '3',
|
||||||
'X-Object-Meta-key4':'4',
|
'X-Object-Meta-key4': '4',
|
||||||
'X-Object-Meta-key5':'5',
|
'X-Object-Meta-key5': '5',
|
||||||
'X-Object-Meta-key6':'6',
|
'X-Object-Meta-key6': '6',
|
||||||
'X-Object-Meta-key7':'7',
|
'X-Object-Meta-key7': '7',
|
||||||
'X-Object-Meta-key8':'8',
|
'X-Object-Meta-key8': '8',
|
||||||
'X-Object-Meta-key9':'9',
|
'X-Object-Meta-key9': '9',
|
||||||
'X-Object-Meta-key10':'10'}
|
'X-Object-Meta-key10': '10'}
|
||||||
f = open('%s/%s' %(PATH_TO_BUNDLE, SOURCE_FILE),'r')
|
f = open('%s/%s' % (PATH_TO_BUNDLE, SOURCE_FILE), 'r')
|
||||||
c.put_object(url, token, 'myobjects', SOURCE_FILE, f,
|
c.put_object(url, token, 'myobjects', SOURCE_FILE, f,
|
||||||
content_type = "application/octet-stream",
|
content_type="application/octet-stream",
|
||||||
headers = metadata,
|
headers=metadata,
|
||||||
response_dict = resp)
|
response_dict=resp)
|
||||||
f.close()
|
f.close()
|
||||||
status = resp.get('status')
|
status = resp.get('status')
|
||||||
assert (status == 200 or status == 201)
|
assert (status == 200 or status == 201)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
def deploy_storlet(url,token):
|
|
||||||
#No need to create containers every time
|
|
||||||
#put_storlet_containers(url, token)
|
def deploy_storlet(url, token):
|
||||||
put_storlet_object( url, token,
|
# No need to create containers every time
|
||||||
STORLET_NAME,
|
# put_storlet_containers(url, token)
|
||||||
PATH_TO_BUNDLE,
|
put_storlet_object(url, token,
|
||||||
'',
|
STORLET_NAME,
|
||||||
'com.ibm.storlet.testmetadatastorlet.MetadataStorlet')
|
PATH_TO_BUNDLE,
|
||||||
put_storlet_input_object( url, token )
|
'',
|
||||||
|
'com.ibm.storlet.testmetadatastorlet.MetadataStorlet')
|
||||||
|
put_storlet_input_object(url, token)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
def invoke_storlet(url, token,op, params = None, global_params = None):
|
|
||||||
if params != None:
|
|
||||||
querystring=''
|
def invoke_storlet(url, token, op, params=None, global_params=None):
|
||||||
|
if params is not None:
|
||||||
|
querystring = ''
|
||||||
for key in params:
|
for key in params:
|
||||||
querystring += '%s=%s,' % (key, params[key])
|
querystring += '%s=%s,' % (key, params[key])
|
||||||
querystring = querystring[:-1]
|
querystring = querystring[:-1]
|
||||||
else:
|
else:
|
||||||
querystring = None
|
querystring = None
|
||||||
|
|
||||||
metadata = {'X-Run-Storlet': STORLET_NAME }
|
metadata = {'X-Run-Storlet': STORLET_NAME}
|
||||||
if op == 'GET':
|
if op == 'GET':
|
||||||
# Get original object
|
# Get original object
|
||||||
original_headers, original_content = c.get_object(url, token,
|
original_headers, original_content = \
|
||||||
'myobjects',
|
c.get_object(url, token, 'myobjects', SOURCE_FILE,
|
||||||
SOURCE_FILE,
|
response_dict=dict(), headers=metadata)
|
||||||
response_dict=dict(),
|
|
||||||
headers=metadata)
|
|
||||||
assert(original_headers['X-Object-Meta-key1'.lower()] == '1')
|
assert(original_headers['X-Object-Meta-key1'.lower()] == '1')
|
||||||
assert(original_headers['X-Object-Meta-key2'.lower()] == '2')
|
assert(original_headers['X-Object-Meta-key2'.lower()] == '2')
|
||||||
assert(original_headers['X-Object-Meta-key3'.lower()] == '3')
|
assert(original_headers['X-Object-Meta-key3'.lower()] == '3')
|
||||||
@ -88,23 +93,26 @@ def invoke_storlet(url, token,op, params = None, global_params = None):
|
|||||||
assert(original_headers['X-Object-Meta-key8'.lower()] == '8')
|
assert(original_headers['X-Object-Meta-key8'.lower()] == '8')
|
||||||
assert(original_headers['X-Object-Meta-key9'.lower()] == '9')
|
assert(original_headers['X-Object-Meta-key9'.lower()] == '9')
|
||||||
assert(original_headers['X-Object-Meta-key10'.lower()] == '10')
|
assert(original_headers['X-Object-Meta-key10'.lower()] == '10')
|
||||||
assert(original_headers['X-Object-Meta-override_key'.lower()] == 'new_value')
|
assert(original_headers['X-Object-Meta-override_key'.lower()] ==
|
||||||
|
'new_value')
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
os_options = {'tenant_name': ACCOUNT}
|
os_options = {'tenant_name': ACCOUNT}
|
||||||
url, token = c.get_auth( 'http://' + AUTH_IP + ":"
|
url, token = c.get_auth('http://' + AUTH_IP + ":"
|
||||||
+ AUTH_PORT + '/v2.0',
|
+ AUTH_PORT + '/v2.0',
|
||||||
ACCOUNT + ':' + USER_NAME,
|
ACCOUNT + ':' + USER_NAME,
|
||||||
PASSWORD,
|
PASSWORD,
|
||||||
os_options = os_options,
|
os_options=os_options,
|
||||||
auth_version = '2.0' )
|
auth_version='2.0')
|
||||||
|
|
||||||
print 'Deploying storlet and dependencies'
|
print('Deploying storlet and dependencies')
|
||||||
deploy_storlet(url, token)
|
deploy_storlet(url, token)
|
||||||
|
|
||||||
print "Invoking storlet on GET"
|
print("Invoking storlet on GET")
|
||||||
invoke_storlet(url, token,'GET')
|
invoke_storlet(url, token, 'GET')
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
@ -14,29 +14,36 @@ Limitations under the License.
|
|||||||
-------------------------------------------------------------------------'''
|
-------------------------------------------------------------------------'''
|
||||||
|
|
||||||
import sys
|
import sys
|
||||||
import json
|
|
||||||
|
|
||||||
from swiftclient import client as c
|
from swiftclient import client as c
|
||||||
from swiftclient.client import encode_utf8, http_connection
|
|
||||||
|
|
||||||
def progress():
|
def progress():
|
||||||
sys.stdout.write('.')
|
sys.stdout.write('.')
|
||||||
sys.stdout.flush()
|
sys.stdout.flush()
|
||||||
|
|
||||||
|
|
||||||
def progress_ln():
|
def progress_ln():
|
||||||
sys.stdout.write('\n')
|
sys.stdout.write('\n')
|
||||||
sys.stdout.flush()
|
sys.stdout.flush()
|
||||||
|
|
||||||
|
|
||||||
def progress_msg(msg):
|
def progress_msg(msg):
|
||||||
sys.stdout.write(msg)
|
sys.stdout.write(msg)
|
||||||
sys.stdout.flush()
|
sys.stdout.flush()
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
def enable_account_for_storlets(url,token):
|
|
||||||
|
|
||||||
|
def enable_account_for_storlets(url, token):
|
||||||
headers = dict()
|
headers = dict()
|
||||||
headers['X-Account-Meta-storlet-enabled'] = 'True'
|
headers['X-Account-Meta-storlet-enabled'] = 'True'
|
||||||
c.post_account(url, token, headers)
|
c.post_account(url, token, headers)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
def put_storlet_containers(url,token):
|
|
||||||
|
|
||||||
|
def put_storlet_containers(url, token):
|
||||||
|
|
||||||
response = dict()
|
response = dict()
|
||||||
c.put_container(url, token, 'storlet', None, None, response)
|
c.put_container(url, token, 'storlet', None, None, response)
|
||||||
@ -54,35 +61,39 @@ def put_storlet_containers(url,token):
|
|||||||
assert (status >= 200 or status < 300)
|
assert (status >= 200 or status < 300)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def put_file_as_storlet_input_object(url, token, local_path, local_file):
|
def put_file_as_storlet_input_object(url, token, local_path, local_file):
|
||||||
resp = dict()
|
resp = dict()
|
||||||
f = open('%s/%s' %(local_path, local_file),'r')
|
f = open('%s/%s' % (local_path, local_file), 'r')
|
||||||
c.put_object(url, token, 'myobjects', local_file, f,
|
c.put_object(url, token, 'myobjects', local_file, f,
|
||||||
content_type = "application/octet-stream",
|
content_type="application/octet-stream",
|
||||||
response_dict = resp)
|
response_dict=resp)
|
||||||
f.close()
|
f.close()
|
||||||
status = resp.get('status')
|
status = resp.get('status')
|
||||||
assert (status == 200 or status == 201)
|
assert (status == 200 or status == 201)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def put_storlet_object(url, token, storlet_name, storlet_path,
|
def put_storlet_object(url, token, storlet_name, storlet_path,
|
||||||
dependency, main_class):
|
dependency, main_class):
|
||||||
# Delete previous storlet
|
# Delete previous storlet
|
||||||
resp = dict()
|
# resp = dict()
|
||||||
'''
|
'''try:
|
||||||
try:
|
|
||||||
c.delete_object(url, token, 'storlet', storlet_name, None,
|
c.delete_object(url, token, 'storlet', storlet_name, None,
|
||||||
None, None, None, resp)
|
None, None, None, resp)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
if (resp.get('status')== 404):
|
if (resp.get('status')== 404):
|
||||||
print 'Nothing to delete'
|
print 'Nothing to delete'
|
||||||
'''
|
'''
|
||||||
metadata = {'X-Object-Meta-Storlet-Language':'Java',
|
metadata = {'X-Object-Meta-Storlet-Language': 'Java',
|
||||||
'X-Object-Meta-Storlet-Interface-Version':'1.0',
|
'X-Object-Meta-Storlet-Interface-Version': '1.0',
|
||||||
'X-Object-Meta-Storlet-Dependency': dependency,
|
'X-Object-Meta-Storlet-Dependency': dependency,
|
||||||
'X-Object-Meta-Storlet-Object-Metadata':'no',
|
'X-Object-Meta-Storlet-Object-Metadata': 'no',
|
||||||
'X-Object-Meta-Storlet-Main': main_class}
|
'X-Object-Meta-Storlet-Main': main_class}
|
||||||
f = open('%s/%s' % (storlet_path, storlet_name),'r')
|
f = open('%s/%s' % (storlet_path, storlet_name), 'r')
|
||||||
content_length = None
|
content_length = None
|
||||||
response = dict()
|
response = dict()
|
||||||
c.put_object(url, token, 'storlet', storlet_name, f,
|
c.put_object(url, token, 'storlet', storlet_name, f,
|
||||||
@ -93,11 +104,12 @@ def put_storlet_object(url, token, storlet_name, storlet_path,
|
|||||||
status = response.get('status')
|
status = response.get('status')
|
||||||
assert (status == 200 or status == 201)
|
assert (status == 200 or status == 201)
|
||||||
|
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def put_dependency(url, token, local_path_to_dep, dep_name):
|
def put_dependency(url, token, local_path_to_dep, dep_name):
|
||||||
metadata = {'X-Object-Meta-Storlet-Dependency-Version': '1'}
|
metadata = {'X-Object-Meta-Storlet-Dependency-Version': '1'}
|
||||||
f = open('%s/%s' %(local_path_to_dep, dep_name),'r')
|
f = open('%s/%s' % (local_path_to_dep, dep_name), 'r')
|
||||||
content_length = None
|
content_length = None
|
||||||
response = dict()
|
response = dict()
|
||||||
c.put_object(url, token, 'dependency', dep_name, f,
|
c.put_object(url, token, 'dependency', dep_name, f,
|
||||||
|
@ -17,30 +17,29 @@ Limitations under the License.
|
|||||||
@author: gilv
|
@author: gilv
|
||||||
'''
|
'''
|
||||||
|
|
||||||
import threading
|
|
||||||
import time
|
|
||||||
import json
|
|
||||||
import os
|
import os
|
||||||
import random
|
import threading
|
||||||
import string
|
|
||||||
import tarfile
|
|
||||||
|
|
||||||
from sys_test_params import *
|
|
||||||
from swiftclient import client as c
|
from swiftclient import client as c
|
||||||
from swiftclient.client import encode_utf8, http_connection
|
from sys_test_params import ACCOUNT
|
||||||
|
from sys_test_params import AUTH_IP
|
||||||
|
from sys_test_params import AUTH_PORT
|
||||||
|
from sys_test_params import PASSWORD
|
||||||
|
from sys_test_params import USER_NAME
|
||||||
|
|
||||||
from storlets_test_utils import enable_account_for_storlets, \
|
from storlets_test_utils import put_storlet_object
|
||||||
put_dependency, put_storlet_containers, put_storlet_object
|
|
||||||
|
|
||||||
TEST_STORLET_NAME='test-10.jar'
|
TEST_STORLET_NAME = 'test-10.jar'
|
||||||
|
|
||||||
PATH_TO_STORLET_GIT_MODULE = ''
|
PATH_TO_STORLET_GIT_MODULE = ''
|
||||||
PATH_TO_STORLETS = ''
|
PATH_TO_STORLETS = ''
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def invokeTestStorlet(url, token, op, withlog=False):
|
def invokeTestStorlet(url, token, op, withlog=False):
|
||||||
headers = {'X-Run-Storlet':TEST_STORLET_NAME}
|
headers = {'X-Run-Storlet': TEST_STORLET_NAME}
|
||||||
if withlog == True:
|
if withlog is True:
|
||||||
headers['X-Storlet-Generate-Log'] = 'True'
|
headers['X-Storlet-Generate-Log'] = 'True'
|
||||||
|
|
||||||
params = 'op={0}¶m2=val2'.format(op)
|
params = 'op={0}¶m2=val2'.format(op)
|
||||||
@ -49,19 +48,18 @@ def invokeTestStorlet(url, token, op, withlog=False):
|
|||||||
resp_headers, gf = c.get_object(url, token, 'myobjects',
|
resp_headers, gf = c.get_object(url, token, 'myobjects',
|
||||||
'test_object', None, None, params,
|
'test_object', None, None, params,
|
||||||
resp_dict, headers)
|
resp_dict, headers)
|
||||||
#print resp_dict
|
# print(resp_dict)
|
||||||
get_text = gf
|
get_text = gf
|
||||||
#print get_text
|
# print(get_text)
|
||||||
get_response_status = resp_dict.get('status')
|
get_response_status = resp_dict.get('status')
|
||||||
|
|
||||||
if withlog == True:
|
if withlog is True:
|
||||||
resp_headers, gf = c.get_object(url, token,
|
resp_headers, gf = c.get_object(url, token,
|
||||||
'storletlog', 'test.log',
|
'storletlog', 'test.log',
|
||||||
None, None, None, None, headers)
|
None, None, None, None, headers)
|
||||||
assert resp_headers.get('status') == 200
|
assert resp_headers.get('status') == 200
|
||||||
text = gf.read()
|
gf.read()
|
||||||
assert resp_headers.get('status') == 200
|
assert resp_headers.get('status') == 200
|
||||||
#print text
|
|
||||||
|
|
||||||
if op == 'print':
|
if op == 'print':
|
||||||
assert get_response_status == 200
|
assert get_response_status == 200
|
||||||
@ -73,27 +71,32 @@ def invokeTestStorlet(url, token, op, withlog=False):
|
|||||||
except Exception as e:
|
except Exception as e:
|
||||||
get_response_status = resp_dict.get('status')
|
get_response_status = resp_dict.get('status')
|
||||||
if op == 'crash':
|
if op == 'crash':
|
||||||
print get_response_status
|
print(get_response_status)
|
||||||
assert get_response_status >= 500 or get_response_status == 404
|
assert get_response_status >= 500 or get_response_status == 404
|
||||||
|
|
||||||
if op == 'hold':
|
if op == 'hold':
|
||||||
#print get_response_status
|
# print(get_response_status)
|
||||||
assert get_response_status >= 500 or get_response_status == 404
|
assert get_response_status >= 500 or get_response_status == 404
|
||||||
|
|
||||||
if op == 'print':
|
if op == 'print':
|
||||||
#print get_response_status
|
# print(get_response_status)
|
||||||
raise e
|
raise e
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
class myTestThread (threading.Thread):
|
class myTestThread (threading.Thread):
|
||||||
def __init__(self, url, token):
|
def __init__(self, url, token):
|
||||||
threading.Thread.__init__(self)
|
threading.Thread.__init__(self)
|
||||||
self.token = token
|
self.token = token
|
||||||
self.url = url
|
self.url = url
|
||||||
|
|
||||||
def run(self):
|
def run(self):
|
||||||
invokeTestStorlet(self.url, self.token, "print", False)
|
invokeTestStorlet(self.url, self.token, "print", False)
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def invokeTestStorletinParallel(url, token):
|
def invokeTestStorletinParallel(url, token):
|
||||||
mythreads = []
|
mythreads = []
|
||||||
|
|
||||||
@ -108,8 +111,10 @@ def invokeTestStorletinParallel(url, token):
|
|||||||
t.join()
|
t.join()
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def testTestStorlet(url, token):
|
def testTestStorlet(url, token):
|
||||||
print "Deploying test storlet"
|
print("Deploying test storlet")
|
||||||
put_storlet_object(url,
|
put_storlet_object(url,
|
||||||
token,
|
token,
|
||||||
TEST_STORLET_NAME,
|
TEST_STORLET_NAME,
|
||||||
@ -117,49 +122,51 @@ def testTestStorlet(url, token):
|
|||||||
'',
|
'',
|
||||||
'com.ibm.storlet.test.test1')
|
'com.ibm.storlet.test.test1')
|
||||||
|
|
||||||
print "uploading object to execute test upon"
|
print("uploading object to execute test upon")
|
||||||
c.put_object(url,
|
c.put_object(url,
|
||||||
token,
|
token,
|
||||||
'myobjects',
|
'myobjects',
|
||||||
'test_object',
|
'test_object',
|
||||||
'some content')
|
'some content')
|
||||||
print "Invoking test storlet to print"
|
print("Invoking test storlet to print")
|
||||||
invokeTestStorlet(url, token, "print", False)
|
invokeTestStorlet(url, token, "print", False)
|
||||||
print "Invoking test storlet to crash"
|
print("Invoking test storlet to crash")
|
||||||
invokeTestStorlet(url, token, "crash")
|
invokeTestStorlet(url, token, "crash")
|
||||||
print "Invoking test storlet to hold"
|
print("Invoking test storlet to hold")
|
||||||
invokeTestStorlet(url, token, "hold")
|
invokeTestStorlet(url, token, "hold")
|
||||||
print "Invoking test storlet to print"
|
print("Invoking test storlet to print")
|
||||||
invokeTestStorlet(url, token, "print", False)
|
invokeTestStorlet(url, token, "print", False)
|
||||||
print "Invoking test storlet in parallel to print"
|
print("Invoking test storlet in parallel to print")
|
||||||
invokeTestStorletinParallel(url, token)
|
invokeTestStorletinParallel(url, token)
|
||||||
|
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def init_path_dependant_params():
|
def init_path_dependant_params():
|
||||||
global PATH_TO_STORLET_GIT_MODULE
|
global PATH_TO_STORLET_GIT_MODULE
|
||||||
global PATH_TO_STORLETS
|
global PATH_TO_STORLETS
|
||||||
PATH_TO_STORLET_GIT_MODULE = '../'
|
PATH_TO_STORLET_GIT_MODULE = '../'
|
||||||
if PATH_TO_STORLET_GIT_MODULE == '':
|
if PATH_TO_STORLET_GIT_MODULE == '':
|
||||||
PATH_TO_STORLET_GIT_MODULE = os.environ['HOME'] + \
|
PATH_TO_STORLET_GIT_MODULE = os.environ['HOME'] + '/workspace/Storlets'
|
||||||
'/workspace/Storlets'
|
PATH_TO_STORLETS = '%s/StorletSamples' % PATH_TO_STORLET_GIT_MODULE
|
||||||
PATH_TO_STORLETS='%s/StorletSamples' % PATH_TO_STORLET_GIT_MODULE
|
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
init_path_dependant_params()
|
init_path_dependant_params()
|
||||||
|
|
||||||
print 'Getting token'
|
print('Getting token')
|
||||||
os_options = {'tenant_name': ACCOUNT}
|
os_options = {'tenant_name': ACCOUNT}
|
||||||
url, token = c.get_auth("http://" + AUTH_IP + ":" + AUTH_PORT \
|
url, token = c.get_auth("http://" + AUTH_IP + ":" + AUTH_PORT
|
||||||
+ "/v2.0", ACCOUNT + ":" + USER_NAME,
|
+ "/v2.0", ACCOUNT + ":" + USER_NAME,
|
||||||
PASSWORD, os_options = os_options,
|
PASSWORD, os_options=os_options,
|
||||||
auth_version="2.0")
|
auth_version="2.0")
|
||||||
|
|
||||||
print 'Creating myobjects container'
|
print('Creating myobjects container')
|
||||||
c.put_container(url, token, 'myobjects')
|
c.put_container(url, token, 'myobjects')
|
||||||
|
|
||||||
print 'Invoking test storlet'
|
print('Invoking test storlet')
|
||||||
testTestStorlet(url, token)
|
testTestStorlet(url, token)
|
||||||
|
|
||||||
os.system('python execdep_test.py')
|
os.system('python execdep_test.py')
|
||||||
|
@ -18,22 +18,21 @@ Limitations under the License.
|
|||||||
13-Jan-2015 evgenyl Initial implementation.
|
13-Jan-2015 evgenyl Initial implementation.
|
||||||
==========================================================================='''
|
==========================================================================='''
|
||||||
|
|
||||||
import time
|
|
||||||
import sys
|
import sys
|
||||||
|
import time
|
||||||
|
|
||||||
'''------------------------------------------------------------------------'''
|
'''------------------------------------------------------------------------'''
|
||||||
|
|
||||||
|
|
||||||
class TextUIProgressBar:
|
class TextUIProgressBar(object):
|
||||||
'''
|
'''@summary: This class simulates Progress Bar GUI widget in UNIX terminal.
|
||||||
@summary: This class simulates Progress Bar GUI widget in UNIX terminal.
|
|
||||||
'''
|
'''
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
'''
|
'''@summary: CTOR, define some constant mapping'''
|
||||||
@summary: CTOR, define some constant mapping
|
|
||||||
'''
|
|
||||||
self.colors = {}
|
self.colors = {}
|
||||||
self.colors['gray'] = '30'
|
self.colors['gray'] = '30'
|
||||||
self.colors['red'] = '31'
|
self.colors['red'] = '31'
|
||||||
@ -45,10 +44,10 @@ class TextUIProgressBar:
|
|||||||
self.colors['white'] = '37'
|
self.colors['white'] = '37'
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
def update_progress_bar(self, complete, total, caption = '', color='' ):
|
|
||||||
'''
|
def update_progress_bar(self, complete, total, caption='', color=''):
|
||||||
@summary: update_progress_bar
|
'''@summary: update_progress_bar Drawing code. The idea is
|
||||||
Drawing code. The idea is
|
|
||||||
- jump to the beginning of the line
|
- jump to the beginning of the line
|
||||||
- print the same amount of characters
|
- print the same amount of characters
|
||||||
but in a different proportion (complete/total)
|
but in a different proportion (complete/total)
|
||||||
@ -65,23 +64,25 @@ class TextUIProgressBar:
|
|||||||
color = self.colors.get(color, self.colors['white'])
|
color = self.colors.get(color, self.colors['white'])
|
||||||
color_start = '\033[01;' + color + 'm'
|
color_start = '\033[01;' + color + 'm'
|
||||||
color_stop = '\033[00m'
|
color_stop = '\033[00m'
|
||||||
print '\r' + color_start + u'\u2591'*complete + \
|
sys.stdout.write('\r' + color_start + u'\u2591' * complete +
|
||||||
u'\u2593'*(total-complete) + color_stop,
|
u'\u2593' * (total - complete) + color_stop)
|
||||||
if 0 < len(caption):
|
if 0 < len(caption):
|
||||||
print '{0}'.format(caption) ,
|
sys.stdout.write('{0}'.format(caption))
|
||||||
sys.stdout.flush()
|
sys.stdout.flush()
|
||||||
|
|
||||||
'''--------------------------------------------------------------------'''
|
'''--------------------------------------------------------------------'''
|
||||||
|
|
||||||
def test(self):
|
def test(self):
|
||||||
|
'''@summary: test
|
||||||
|
|
||||||
|
Unit test. Simulate a process of 10 steps with delay of one
|
||||||
|
second after each step.
|
||||||
'''
|
'''
|
||||||
@summary: test
|
|
||||||
Unit test. Simulate a process of 10 steps with
|
|
||||||
delay of one second after each step.
|
|
||||||
'''
|
|
||||||
k = self.colors.keys()
|
k = self.colors.keys()
|
||||||
l = len(k)
|
l = len(k)
|
||||||
for j in range(1, l+1):
|
for j in range(1, l + 1):
|
||||||
self.update_progress_bar(j, l, str(j), k[j-1])
|
self.update_progress_bar(j, l, str(j), k[j - 1])
|
||||||
time.sleep(1)
|
time.sleep(1)
|
||||||
|
|
||||||
'''============================= END OF FILE =============================='''
|
'''============================= END OF FILE =============================='''
|
||||||
|
75
doc/source/conf.py
Executable file
75
doc/source/conf.py
Executable file
@ -0,0 +1,75 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||||
|
# implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
|
||||||
|
sys.path.insert(0, os.path.abspath('../..'))
|
||||||
|
# -- General configuration ----------------------------------------------------
|
||||||
|
|
||||||
|
# Add any Sphinx extension module names here, as strings. They can be
|
||||||
|
# extensions coming with Sphinx (named 'sphinx.ext.*') or your custom ones.
|
||||||
|
extensions = [
|
||||||
|
'sphinx.ext.autodoc',
|
||||||
|
#'sphinx.ext.intersphinx',
|
||||||
|
'oslosphinx'
|
||||||
|
]
|
||||||
|
|
||||||
|
# autodoc generation is a bit aggressive and a nuisance when doing heavy
|
||||||
|
# text edit cycles.
|
||||||
|
# execute "export SPHINX_DEBUG=1" in your terminal to disable
|
||||||
|
|
||||||
|
# The suffix of source filenames.
|
||||||
|
source_suffix = '.rst'
|
||||||
|
|
||||||
|
# The master toctree document.
|
||||||
|
master_doc = 'index'
|
||||||
|
|
||||||
|
# General information about the project.
|
||||||
|
project = u'storlets'
|
||||||
|
copyright = u'2013, OpenStack Foundation'
|
||||||
|
|
||||||
|
# If true, '()' will be appended to :func: etc. cross-reference text.
|
||||||
|
add_function_parentheses = True
|
||||||
|
|
||||||
|
# If true, the current module name will be prepended to all description
|
||||||
|
# unit titles (such as .. function::).
|
||||||
|
add_module_names = True
|
||||||
|
|
||||||
|
# The name of the Pygments (syntax highlighting) style to use.
|
||||||
|
pygments_style = 'sphinx'
|
||||||
|
|
||||||
|
# -- Options for HTML output --------------------------------------------------
|
||||||
|
|
||||||
|
# The theme to use for HTML and HTML Help pages. Major themes that come with
|
||||||
|
# Sphinx are currently 'default' and 'sphinxdoc'.
|
||||||
|
# html_theme_path = ["."]
|
||||||
|
# html_theme = '_theme'
|
||||||
|
# html_static_path = ['static']
|
||||||
|
|
||||||
|
# Output file base name for HTML help builder.
|
||||||
|
htmlhelp_basename = '%sdoc' % project
|
||||||
|
|
||||||
|
# Grouping the document tree into LaTeX files. List of tuples
|
||||||
|
# (source start file, target name, title, author, documentclass
|
||||||
|
# [howto/manual]).
|
||||||
|
latex_documents = [
|
||||||
|
('index',
|
||||||
|
'%s.tex' % project,
|
||||||
|
u'%s Documentation' % project,
|
||||||
|
u'OpenStack Foundation', 'manual'),
|
||||||
|
]
|
||||||
|
|
||||||
|
# Example configuration for intersphinx: refer to the Python standard library.
|
||||||
|
#intersphinx_mapping = {'http://docs.python.org/': None}
|
4
doc/source/contributing.rst
Normal file
4
doc/source/contributing.rst
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
============
|
||||||
|
Contributing
|
||||||
|
============
|
||||||
|
.. include:: ../../CONTRIBUTING.rst
|
25
doc/source/index.rst
Normal file
25
doc/source/index.rst
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
.. storlets documentation master file, created by
|
||||||
|
sphinx-quickstart on Tue Jul 9 22:26:36 2013.
|
||||||
|
You can adapt this file completely to your liking, but it should at least
|
||||||
|
contain the root `toctree` directive.
|
||||||
|
|
||||||
|
Welcome to storlets's documentation!
|
||||||
|
========================================================
|
||||||
|
|
||||||
|
Contents:
|
||||||
|
|
||||||
|
.. toctree::
|
||||||
|
:maxdepth: 2
|
||||||
|
|
||||||
|
readme
|
||||||
|
installation
|
||||||
|
usage
|
||||||
|
contributing
|
||||||
|
|
||||||
|
Indices and tables
|
||||||
|
==================
|
||||||
|
|
||||||
|
* :ref:`genindex`
|
||||||
|
* :ref:`modindex`
|
||||||
|
* :ref:`search`
|
||||||
|
|
12
doc/source/installation.rst
Normal file
12
doc/source/installation.rst
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
============
|
||||||
|
Installation
|
||||||
|
============
|
||||||
|
|
||||||
|
At the command line::
|
||||||
|
|
||||||
|
$ pip install storlets
|
||||||
|
|
||||||
|
Or, if you have virtualenvwrapper installed::
|
||||||
|
|
||||||
|
$ mkvirtualenv storlets
|
||||||
|
$ pip install storlets
|
1
doc/source/readme.rst
Normal file
1
doc/source/readme.rst
Normal file
@ -0,0 +1 @@
|
|||||||
|
.. include:: ../../README.rst
|
7
doc/source/usage.rst
Normal file
7
doc/source/usage.rst
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
========
|
||||||
|
Usage
|
||||||
|
========
|
||||||
|
|
||||||
|
To use storlets in a project::
|
||||||
|
|
||||||
|
import storlets
|
6
openstack-common.conf
Normal file
6
openstack-common.conf
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
[DEFAULT]
|
||||||
|
|
||||||
|
# The list of modules to copy from oslo-incubator.git
|
||||||
|
|
||||||
|
# The base module to hold the copy of openstack.common
|
||||||
|
base=storlets
|
6
requirements.txt
Normal file
6
requirements.txt
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
# The order of packages is significant, because pip processes them in the order
|
||||||
|
# of appearance. Changing the order has an impact on the overall integration
|
||||||
|
# process, which may cause wedges in the gate later.
|
||||||
|
|
||||||
|
pbr>=0.6,!=0.7,<1.0
|
||||||
|
Babel>=1.3
|
47
setup.cfg
Normal file
47
setup.cfg
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
[metadata]
|
||||||
|
name = storlets
|
||||||
|
summary = Middleware and Compute Engine for an OpenStack Swift compute framework that runs compute within a Swift cluster
|
||||||
|
description-file =
|
||||||
|
README.rst
|
||||||
|
author = OpenStack
|
||||||
|
author-email = openstack-dev@lists.openstack.org
|
||||||
|
home-page = http://www.openstack.org/
|
||||||
|
classifier =
|
||||||
|
Environment :: OpenStack
|
||||||
|
Intended Audience :: Information Technology
|
||||||
|
Intended Audience :: System Administrators
|
||||||
|
License :: OSI Approved :: Apache Software License
|
||||||
|
Operating System :: POSIX :: Linux
|
||||||
|
Programming Language :: Python
|
||||||
|
Programming Language :: Python :: 2
|
||||||
|
Programming Language :: Python :: 2.7
|
||||||
|
Programming Language :: Python :: 2.6
|
||||||
|
Programming Language :: Python :: 3
|
||||||
|
Programming Language :: Python :: 3.3
|
||||||
|
Programming Language :: Python :: 3.4
|
||||||
|
|
||||||
|
[files]
|
||||||
|
packages =
|
||||||
|
storlets
|
||||||
|
|
||||||
|
[build_sphinx]
|
||||||
|
source-dir = doc/source
|
||||||
|
build-dir = doc/build
|
||||||
|
all_files = 1
|
||||||
|
|
||||||
|
[upload_sphinx]
|
||||||
|
upload-dir = doc/build/html
|
||||||
|
|
||||||
|
[compile_catalog]
|
||||||
|
directory = storlets/locale
|
||||||
|
domain = storlets
|
||||||
|
|
||||||
|
[update_catalog]
|
||||||
|
domain = storlets
|
||||||
|
output_dir = storlets/locale
|
||||||
|
input_file = storlets/locale/storlets.pot
|
||||||
|
|
||||||
|
[extract_messages]
|
||||||
|
keywords = _ gettext ngettext l_ lazy_gettext
|
||||||
|
mapping_file = babel.cfg
|
||||||
|
output_file = storlets/locale/storlets.pot
|
29
setup.py
Normal file
29
setup.py
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
# Copyright (c) 2013 Hewlett-Packard Development Company, L.P.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||||
|
# implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
# THIS FILE IS MANAGED BY THE GLOBAL REQUIREMENTS REPO - DO NOT EDIT
|
||||||
|
import setuptools
|
||||||
|
|
||||||
|
# In python < 2.7.4, a lazy loading of package `pbr` will break
|
||||||
|
# setuptools if some other modules registered functions in `atexit`.
|
||||||
|
# solution from: http://bugs.python.org/issue15881#msg170215
|
||||||
|
try:
|
||||||
|
import multiprocessing # noqa
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
setuptools.setup(
|
||||||
|
setup_requires=['pbr'],
|
||||||
|
pbr=True)
|
18
storlets/__init__.py
Normal file
18
storlets/__init__.py
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
import pbr.version
|
||||||
|
|
||||||
|
__version__ = pbr.version.VersionInfo(
|
||||||
|
'storlets').version_string()
|
0
storlets/tests/__init__.py
Normal file
0
storlets/tests/__init__.py
Normal file
23
storlets/tests/base.py
Normal file
23
storlets/tests/base.py
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright 2010-2011 OpenStack Foundation
|
||||||
|
# Copyright (c) 2013 Hewlett-Packard Development Company, L.P.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
from oslotest import base
|
||||||
|
|
||||||
|
|
||||||
|
class TestCase(base.BaseTestCase):
|
||||||
|
|
||||||
|
"""Test case base class for all unit tests."""
|
28
storlets/tests/test_storlets.py
Normal file
28
storlets/tests/test_storlets.py
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
"""
|
||||||
|
test_storlets
|
||||||
|
----------------------------------
|
||||||
|
|
||||||
|
Tests for `storlets` module.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from storlets.tests import base
|
||||||
|
|
||||||
|
|
||||||
|
class TestStorlets(base.TestCase):
|
||||||
|
|
||||||
|
def test_something(self):
|
||||||
|
pass
|
15
test-requirements.txt
Normal file
15
test-requirements.txt
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
# The order of packages is significant, because pip processes them in the order
|
||||||
|
# of appearance. Changing the order has an impact on the overall integration
|
||||||
|
# process, which may cause wedges in the gate later.
|
||||||
|
|
||||||
|
hacking<0.11,>=0.10.0
|
||||||
|
|
||||||
|
coverage>=3.6
|
||||||
|
discover
|
||||||
|
python-subunit>=0.0.18
|
||||||
|
sphinx>=1.1.2,!=1.2.0,!=1.3b1,<1.3
|
||||||
|
oslosphinx>=2.2.0 # Apache-2.0
|
||||||
|
oslotest>=1.2.0 # Apache-2.0
|
||||||
|
testrepository>=0.0.18
|
||||||
|
testscenarios>=0.4
|
||||||
|
testtools>=0.9.36,!=1.2.0
|
35
tox.ini
Normal file
35
tox.ini
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
[tox]
|
||||||
|
minversion = 1.6
|
||||||
|
envlist = py27,pep8
|
||||||
|
skipsdist = True
|
||||||
|
|
||||||
|
[testenv]
|
||||||
|
usedevelop = True
|
||||||
|
install_command = pip install -U {opts} {packages}
|
||||||
|
setenv =
|
||||||
|
VIRTUAL_ENV={envdir}
|
||||||
|
deps = -r{toxinidir}/test-requirements.txt
|
||||||
|
commands = python setup.py test --slowest --testr-args='{posargs}'
|
||||||
|
|
||||||
|
[testenv:pep8]
|
||||||
|
commands = flake8
|
||||||
|
|
||||||
|
[testenv:venv]
|
||||||
|
commands = {posargs}
|
||||||
|
|
||||||
|
[testenv:cover]
|
||||||
|
commands = python setup.py test --coverage --testr-args='{posargs}'
|
||||||
|
|
||||||
|
[testenv:docs]
|
||||||
|
commands = python setup.py build_sphinx
|
||||||
|
|
||||||
|
[testenv:debug]
|
||||||
|
commands = oslo_debug_helper {posargs}
|
||||||
|
|
||||||
|
[flake8]
|
||||||
|
# E123, E125 skipped as they are invalid PEP-8.
|
||||||
|
|
||||||
|
show-source = True
|
||||||
|
ignore = E123,E125
|
||||||
|
builtins = _
|
||||||
|
exclude=.venv,.git,.tox,dist,doc,*openstack/common*,*lib/python*,*egg,build
|
Loading…
Reference in New Issue
Block a user