Initial Cookiecutter Commit.

Change-Id: I58c1914ca033c2c40ba2b63a3c07e0a9a8397ba4
This commit is contained in:
Doron Chen 2015-08-06 11:48:50 +03:00
parent c0809470d1
commit 403102e8e9
56 changed files with 2028 additions and 1353 deletions

7
.coveragerc Normal file
View File

@ -0,0 +1,7 @@
[run]
branch = True
source = storlets
omit = storlets/openstack/*
[report]
ignore-errors = True

54
.gitignore vendored Normal file
View 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
View File

@ -0,0 +1,4 @@
[gerrit]
host=review.openstack.org
port=29418
project=openstack/storlets.git

3
.mailmap Normal file
View 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
View 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
View 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

View File

@ -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

View File

@ -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)

View File

@ -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"

View File

@ -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,6 +73,7 @@ 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():
@ -90,21 +93,21 @@ 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): def unpatch_swift_config_file(conf, conf_file):
storlet_middleware = conf.get('common-confs', 'storlet_middleware') 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): 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
@ -113,34 +116,45 @@ def patch_swift_config_file(conf, conf_file, service):
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
@ -149,18 +163,22 @@ def patch_swift_storlet_proxy_file(conf):
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)
@ -171,8 +189,10 @@ def remove(conf):
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')
@ -182,8 +202,10 @@ def install(conf):
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:

View File

@ -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.
''' '''
@ -74,10 +75,10 @@ class SBus(object):
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.
@ -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".
@ -237,5 +241,4 @@ class SBus(object):
n_params) n_params)
return n_status return n_status
'''============================ END OF FILE ===============================''' '''============================ END OF FILE ==============================='''

View 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.
@ -67,11 +67,12 @@ class SBusDatagram(object):
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

View File

@ -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.

View 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.

View 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");
@ -20,4 +19,3 @@ setup( name = 'SBusPythonFacade',
version='1.0', version='1.0',
package_dir={'SBusPythonFacade': ''}, package_dir={'SBusPythonFacade': ''},
packages=['SBusPythonFacade']) packages=['SBusPythonFacade'])

View File

@ -1,39 +1,43 @@
#!/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)
@ -44,11 +48,11 @@ def main(argv):
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)

View File

@ -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
@ -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,7 +230,7 @@ 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)
@ -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
@ -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
@ -334,9 +343,10 @@ class daemon_factory():
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
@ -358,7 +368,7 @@ class daemon_factory():
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:
@ -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)
@ -402,7 +413,7 @@ class daemon_factory():
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
@ -453,11 +466,11 @@ class daemon_factory():
@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)
@ -465,8 +478,7 @@ class daemon_factory():
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)
@ -474,9 +486,10 @@ class daemon_factory():
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
@ -491,6 +504,7 @@ class daemon_factory():
@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
@ -519,12 +533,12 @@ class daemon_factory():
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()
@ -546,12 +560,14 @@ class daemon_factory():
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)
@ -578,13 +594,14 @@ class daemon_factory():
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()
@ -593,9 +610,10 @@ class daemon_factory():
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':
@ -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
@ -711,7 +734,6 @@ def main(argv):
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)

View File

@ -14,7 +14,8 @@ 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', setup(name='storlet_daemon_factory',
version='1.0', version='1.0',
package_dir={'storlet_daemon_factory': ''}, package_dir={'storlet_daemon_factory': ''},

View File

@ -13,7 +13,8 @@ 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',

View File

@ -1,3 +0,0 @@

View File

@ -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,14 +54,15 @@ 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:
@ -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']):
@ -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
@ -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)
@ -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,8 +352,8 @@ 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.
''' '''
@ -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
@ -484,7 +493,7 @@ class StorletGatewayDocker(StorletGatewayBase):
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
@ -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.

View File

@ -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]
@ -114,7 +122,6 @@ class RunTimePaths():
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
@ -219,7 +233,7 @@ class RunTimeSandbox():
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
@ -271,14 +284,12 @@ class RunTimeSandbox():
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):
@ -304,7 +315,7 @@ class RunTimeSandbox():
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
@ -316,7 +327,8 @@ class RunTimeSandbox():
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)
@ -324,7 +336,7 @@ class RunTimeSandbox():
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
@ -336,32 +348,39 @@ class RunTimeSandbox():
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,7 +391,8 @@ 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 = \
'/home/swift/%s/%s' % (invocation_data['storlet_main_class'],
invocation_data['storlet_name']) 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' % \
@ -380,29 +400,34 @@ class RunTimeSandbox():
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:
@ -461,11 +486,10 @@ class StorletInvocationProtocol():
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)
@ -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
@ -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,17 +576,21 @@ 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)
@ -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()

View File

@ -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):

View File

@ -1,34 +1,32 @@
#----------------------------------------------------------------------------------------------- '''-------------------------------------------------------------------------
# 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)
@ -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,

View File

@ -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):
@ -43,10 +47,10 @@ class StorletHandlerMiddleware(object):
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,39 +87,36 @@ 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,
headers = orig_resp.headers,
request=orig_req, request=orig_req,
conditional_response=True) conditional_response=True)
@ -152,45 +150,46 @@ 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,
account,
container, obj) or \ container, obj) or \
self.proxy_only_storlet_execution == True: 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)
@ -206,9 +205,8 @@ class StorletHandlerMiddleware(object):
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,24 +243,33 @@ 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):
@ -269,11 +277,13 @@ def filter_factory(global_conf, **local_conf):
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', '')
@ -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
View File

@ -0,0 +1,4 @@
storlets Style Commandments
===============================================
Read the OpenStack Style Commandments http://docs.openstack.org/developer/hacking/

28
LICENSE
View File

@ -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
View File

@ -0,0 +1,6 @@
include AUTHORS
include ChangeLog
exclude .gitignore
exclude .gitreview
global-exclude *.pyc

19
README.rst Normal file
View 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

View File

@ -17,10 +17,13 @@ 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 storlets_test_utils import put_storlet_containers, put_storlet_object 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
EXECDEP_PATH_TO_BUNDLE = '../StorletSamples/ExecDepStorlet/bin/' EXECDEP_PATH_TO_BUNDLE = '../StorletSamples/ExecDepStorlet/bin/'
EXECDEP_STORLET_NAME = 'execdepstorlet-1.0.jar' EXECDEP_STORLET_NAME = 'execdepstorlet-1.0.jar'
@ -29,19 +32,25 @@ 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')
@ -53,6 +62,8 @@ def put_storlet_input_object(url, token):
assert (status == 200 or status == 201) assert (status == 200 or status == 201)
'''------------------------------------------------------------------------''' '''------------------------------------------------------------------------'''
def deploy_storlet(url, token, name, jarName): def deploy_storlet(url, token, name, jarName):
# No need to create containers every time # No need to create containers every time
# put_storlet_containers(url, token) # put_storlet_containers(url, token)
@ -62,6 +73,8 @@ def deploy_storlet(url,token, name, jarName):
'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':
@ -71,9 +84,8 @@ def invoke_storlet(url, token, storletName, jarName, objectName, mode):
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)
@ -83,6 +95,8 @@ 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 + ":"
@ -92,31 +106,32 @@ def main():
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()

View File

@ -1,13 +1,34 @@
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'
@ -20,12 +41,14 @@ SLOIDENTITY_STORLET_NAME='sloidentitystorlet-1.0.jar'
# 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()
@ -35,6 +58,7 @@ def delete_local_chunks():
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 = []
@ -63,16 +87,20 @@ def put_SLO(url, token):
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:
@ -87,6 +115,7 @@ def get_SLO(url, token):
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:
@ -111,6 +140,7 @@ def compare_slo_to_chunks(body):
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()
@ -122,6 +152,7 @@ def invoke_identity_on_get_SLO(url, token):
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()
@ -160,6 +191,7 @@ def invoke_identity_on_get_SLO_double(url, token):
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):
@ -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()
@ -208,6 +241,8 @@ def deploy_sloidentity_storlet(url, token):
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 + ":"

View File

@ -17,11 +17,14 @@ 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 storlets_test_utils import put_storlet_object, \ from sys_test_params import AUTH_IP
put_file_as_storlet_input_object from sys_test_params import AUTH_PORT
from sys_test_params import PASSWORD
from sys_test_params import USER_NAME
EXECDEP_PATH_TO_BUNDLE = '../StorletSamples/ExecDepStorlet/bin/' EXECDEP_PATH_TO_BUNDLE = '../StorletSamples/ExecDepStorlet/bin/'
EXECDEP_STORLET_NAME = 'execdepstorlet-1.0.jar' EXECDEP_STORLET_NAME = 'execdepstorlet-1.0.jar'
@ -30,6 +33,8 @@ 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:
@ -46,6 +51,8 @@ def put_storlet_executable_dependencies(url, token):
assert (status == 200 or status == 201) assert (status == 200 or status == 201)
'''------------------------------------------------------------------------''' '''------------------------------------------------------------------------'''
def deploy_storlet(url, token): def deploy_storlet(url, token):
# No need to create containers every time # No need to create containers every time
# put_storlet_containers(url, token) # put_storlet_containers(url, token)
@ -61,6 +68,8 @@ def deploy_storlet(url,token):
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()
@ -74,8 +83,9 @@ 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 + ":"
@ -85,10 +95,10 @@ def main():
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)
'''------------------------------------------------------------------------''' '''------------------------------------------------------------------------'''

View File

@ -13,15 +13,16 @@ 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
@ -30,6 +31,8 @@ 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'}
@ -43,6 +46,8 @@ def put_storlet_input_object(url, token):
assert (status == 200 or status == 201) assert (status == 200 or status == 201)
'''------------------------------------------------------------------------''' '''------------------------------------------------------------------------'''
def deploy_storlet(url, token): def deploy_storlet(url, token):
# No need to create containers every time # No need to create containers every time
# put_storlet_containers(url, token) # put_storlet_containers(url, token)
@ -55,8 +60,10 @@ def deploy_storlet(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,
headers=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])
@ -70,57 +77,58 @@ 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))
random_md = ''.join(random.choice(string.ascii_uppercase +
string.digits) for _ in range(32))
# content_length = 1024 # 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 + ":"
@ -130,14 +138,15 @@ def main():
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__":

View File

@ -14,14 +14,19 @@ 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
@ -31,8 +36,9 @@ 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:
@ -49,6 +55,8 @@ def put_storlet_executable_dependencies(url, token):
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'}
@ -62,6 +70,8 @@ def put_storlet_input_object(url, token):
assert (status == 200 or status == 201) assert (status == 200 or status == 201)
'''------------------------------------------------------------------------''' '''------------------------------------------------------------------------'''
def deploy_storlet(url, token): def deploy_storlet(url, token):
# No need to create containers every time # No need to create containers every time
# put_storlet_containers(url, token) # put_storlet_containers(url, token)
@ -74,6 +84,8 @@ def deploy_storlet(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):
@ -98,7 +110,7 @@ def invoke_storlet_on_1GB_file(url, token):
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])
@ -109,88 +121,91 @@ 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')
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:]) 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'})
'''------------------------------------------------------------------------''' '''------------------------------------------------------------------------'''

View File

@ -13,13 +13,14 @@ 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
@ -28,6 +29,8 @@ 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',
@ -50,6 +53,8 @@ def put_storlet_input_object(url, token):
assert (status == 200 or status == 201) assert (status == 200 or status == 201)
'''------------------------------------------------------------------------''' '''------------------------------------------------------------------------'''
def deploy_storlet(url, token): def deploy_storlet(url, token):
# No need to create containers every time # No need to create containers every time
# put_storlet_containers(url, token) # put_storlet_containers(url, token)
@ -61,8 +66,10 @@ def deploy_storlet(url,token):
put_storlet_input_object(url, token) put_storlet_input_object(url, token)
'''------------------------------------------------------------------------''' '''------------------------------------------------------------------------'''
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])
@ -73,11 +80,9 @@ def invoke_storlet(url, token,op, params = None, global_params = 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,9 +93,12 @@ 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 + ":"
@ -100,10 +108,10 @@ def main():
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')
'''------------------------------------------------------------------------''' '''------------------------------------------------------------------------'''

View File

@ -14,28 +14,35 @@ 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()
@ -54,6 +61,8 @@ 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')
@ -65,12 +74,14 @@ def put_file_as_storlet_input_object(url, token, local_path, local_file):
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:
@ -93,8 +104,9 @@ 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')

View File

@ -17,20 +17,17 @@ 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'
@ -38,9 +35,11 @@ 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}&param2=val2'.format(op) params = 'op={0}&param2=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')

View File

@ -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 Drawing code. The idea is
@summary: update_progress_bar
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,19 +64,21 @@ 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):

2
babel.cfg Normal file
View File

@ -0,0 +1,2 @@
[python: **.py]

75
doc/source/conf.py Executable file
View 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}

View File

@ -0,0 +1,4 @@
============
Contributing
============
.. include:: ../../CONTRIBUTING.rst

25
doc/source/index.rst Normal file
View 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`

View 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
View File

@ -0,0 +1 @@
.. include:: ../../README.rst

7
doc/source/usage.rst Normal file
View File

@ -0,0 +1,7 @@
========
Usage
========
To use storlets in a project::
import storlets

6
openstack-common.conf Normal file
View 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
View 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
View 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
View 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
View 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()

View File

23
storlets/tests/base.py Normal file
View 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."""

View 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
View 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
View 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