diff options
author | jsdelfino <jsdelfino@13f79535-47bb-0310-9956-ffa450edef68> | 2009-10-11 00:01:09 +0000 |
---|---|---|
committer | jsdelfino <jsdelfino@13f79535-47bb-0310-9956-ffa450edef68> | 2009-10-11 00:01:09 +0000 |
commit | 4a4c32fea2f0cc36b01ee1382964b10936e1d7d0 (patch) | |
tree | f9059ef059f45a28c7379a7f101035f7b20557a1 | |
parent | a64c1ec2a50f62d63872eac6bc740966602e87bd (diff) |
Added support for JSON-RPC to httpd module. Fixed issues with double numbers in json.hpp. Added store .html and .js files to store test case.
git-svn-id: http://svn.us.apache.org/repos/asf/tuscany@823982 13f79535-47bb-0310-9956-ffa450edef68
-rw-r--r-- | cpp/sca/modules/httpd/Makefile.am | 4 | ||||
-rw-r--r-- | cpp/sca/modules/httpd/mod.cpp | 278 | ||||
-rw-r--r-- | cpp/sca/modules/json/json-test.cpp | 24 | ||||
-rw-r--r-- | cpp/sca/modules/json/json.hpp | 52 | ||||
-rw-r--r-- | cpp/sca/test/store-script/htdocs/.htaccess | 19 | ||||
-rw-r--r-- | cpp/sca/test/store-script/htdocs/store.html | 162 | ||||
-rw-r--r-- | cpp/sca/test/store-script/htdocs/store.js | 661 | ||||
-rw-r--r-- | cpp/sca/test/store-script/store-script-test.cpp | 33 | ||||
-rw-r--r-- | cpp/sca/test/store-script/store.scm (renamed from cpp/sca/test/store-script/store-script.scm) | 11 |
9 files changed, 1121 insertions, 123 deletions
diff --git a/cpp/sca/modules/httpd/Makefile.am b/cpp/sca/modules/httpd/Makefile.am index f7906d09ac..13cb3045f6 100644 --- a/cpp/sca/modules/httpd/Makefile.am +++ b/cpp/sca/modules/httpd/Makefile.am @@ -18,7 +18,7 @@ libdir=$(prefix)/lib lib_LTLIBRARIES = libmod_tuscany.la -INCLUDES = -I. -I$(top_builddir)/kernel -I${HTTPD_INCLUDE} -I${APR_INCLUDE} +INCLUDES = -I. -I$(top_builddir)/kernel -I${HTTPD_INCLUDE} -I${APR_INCLUDE} -I${LIBXML2_INCLUDE} -I${LIBMOZJS_INCLUDE} libmod_tuscany_la_SOURCES = mod.cpp -libmod_tuscany_la_LIBADD = -lpthread +libmod_tuscany_la_LIBADD = -lpthread -L${LIBXML2_LIB} -lxml2 -L${LIBMOZJS_LIB} -lmozjs diff --git a/cpp/sca/modules/httpd/mod.cpp b/cpp/sca/modules/httpd/mod.cpp index 7ecf2840bd..feebca4b5c 100644 --- a/cpp/sca/modules/httpd/mod.cpp +++ b/cpp/sca/modules/httpd/mod.cpp @@ -26,6 +26,7 @@ #include <string> #include <iostream> #include <sstream> +#include <fstream> #include "apr_strings.h" #include "apr_fnmatch.h" @@ -48,6 +49,10 @@ #include "list.hpp" #include "slist.hpp" +#include "value.hpp" +#include "monad.hpp" +#include "../json/json.hpp" +#include "../eval/driver.hpp" extern "C" { extern module AP_MODULE_DECLARE_DATA mod_tuscany; @@ -66,13 +71,38 @@ struct ServerConf { * Directory configuration. */ struct DirConf { - const char* root; - const char* path; - const char* uri; + const char* contribution; const char* component; + const char* implementation; }; /** + * Returns the server conf for a request. + */ +const ServerConf& serverConf(const request_rec* r) { + return *(ServerConf*)ap_get_module_config(r->server->module_config, &mod_tuscany); +} +const std::string home(request_rec* r) { + return serverConf(r).home; +} + +/** + * Returns the dir conf for a request. + */ +const DirConf& dirConf(const request_rec* r) { + return *(DirConf*)ap_get_module_config(r->per_dir_config, &mod_tuscany); +} +const std::string contribution(request_rec* r) { + return dirConf(r).contribution; +} +const std::string component(request_rec* r) { + return dirConf(r).component; +} +const std::string implementation(request_rec* r) { + return dirConf(r).implementation; +} + +/** * Returns an HTTP request path as a list of strings. */ const list<std::string> path(const request_rec* r) { @@ -83,17 +113,6 @@ const list<std::string> path(const request_rec* r) { } /** - * Returns an HTTP query string as a list of lists of strings. - */ -const list<list<std::string> > args(const request_rec* r) { - const char* a = r->args; - if (a == NULL) - return list<list<std::string> >(); - const lambda<list<std::string>(std::string, std::string)> tok(tokenize); - return map(curry(tok, std::string("=")), tokenize("&", a)); -} - -/** * Log HTTP request info to standard out for now, for debugging purposes. */ bool logRequests = true; @@ -109,12 +128,11 @@ const char* optional(const char* s) { return s; } -const bool logRequest(request_rec* r, const ServerConf& sc, const DirConf& dc) { +const bool logRequest(request_rec* r) { std::cout << "mod-tuscany..." << std::endl; - std::cout << "tuscany home: " << sc.home << std::endl; - std::cout << "tuscany root: " << dc.root << std::endl; - std::cout << "tuscany path: " << dc.path << std::endl; - std::cout << "component: " << dc.component << std::endl; + std::cout << "tuscany home: " << home(r) << std::endl; + std::cout << "contribution: " << contribution(r) << std::endl; + std::cout << "component: " << component(r) << std::endl; std::cout << "protocol: " << optional(r->protocol) << std::endl; std::cout << "method: " << optional(r->method) << std::endl; std::cout << "method number: " << r->method_number << std::endl; @@ -124,41 +142,172 @@ const bool logRequest(request_rec* r, const ServerConf& sc, const DirConf& dc) { std::cout << "uri: " << optional(r->uri) << std::endl; std::cout << "path info: " << optional(r->path_info) << std::endl; std::cout << "path: " << path(r) << std::endl; - std::cout << "args info: " << optional(r->args) << std::endl; - std::cout << "args: " << args(r) << std::endl; + std::cout << "args: " << optional(r->args) << std::endl; + std::cout.flush(); return true; } +const value evalLoop(std::istream& is, const value& req, Env& globalEnv) { + value in = read(is); + if (isNil(in)) + return eval(req, globalEnv); + eval(in, globalEnv); + return evalLoop(is, req, globalEnv); +} + +/** + * Returns a list of key value pairs from the args in a query string. + */ +const list<value> queryArg(std::string s) { + const list<std::string> t = tokenize("=", s); + return makeList<value>(car(t).c_str(), cadr(t)); +} + +const list<list<value> > queryArgs(const request_rec* r) { + const char* a = r->args; + if (a == NULL) + return list<list<value> >(); + return map<std::string, list<value>>(queryArg, tokenize("&", a)); +} + +/** + * Returns a list of param values other than the id and method args from a list + * of key value pairs. + */ +const list<value> queryParams(list<list<value> > a) { + if (isNil(a)) + return list<value>(); + if (car(a) == value("id") || car(a) == value("method")) + return queryParams(cdr(a)); + return cons(cadr(car(a)), queryParams(cdr(a))); +} + /** * Handle an HTTP GET request. */ const int get(request_rec* r) { - std::string str("<result>OK</result>"); - if (false) { - r->status = HTTP_NOT_FOUND; - return OK; - } - // Handle a conditional GET - std::string etag(ap_md5(r->pool, (const unsigned char*)str.c_str())); + // Setup the script evaluator + Env globalEnv = setupEnvironment(); + std::ostringstream nullos; + setupEvalOut(nullos); + + // Open the component implementation + const std::string impl = contribution(r) + implementation(r); + std::ifstream is(impl.c_str(), std::ios_base::in); + if (is.fail() || is.bad()) + return HTTP_NOT_FOUND; + + // Extract the request id, method and params from the query string + const list<list<value> > args = queryArgs(r); + const value id = cadr(assoc(value("id"), args)); + const value method = std::string(cadr(assoc(value("method"), args))).c_str(); + const list<value> params = queryParams(args); + + // Build expr to evaluate + const value expr = cons<value>(method, params); + std::cout<< "expr: " << expr << std::endl; + std::cout.flush(); + + // Evaluate the expr + const tuscany::value val = evalLoop(is, expr, globalEnv); + if (isNil(val)) + return HTTP_INTERNAL_SERVER_ERROR; + std::cout<< "val: " << val << std::endl; + std::cout.flush(); + + // Convert the expr value to JSON + const JSONContext cx; + failable<list<std::string>, std::string> jsval = writeJSON(cx, makeList<value>(makeList<value>("id", id), makeList<value>("result", val))); + if (!hasValue(jsval)) + return HTTP_INTERNAL_SERVER_ERROR; + + // Send the response + ap_set_content_type(r, "application/json-rpc"); + std::ostringstream os; + write(jsval, os); + std::string sval = os.str(); + std::string etag(ap_md5(r->pool, (const unsigned char*)sval.c_str())); const char* match = apr_table_get(r->headers_in, "If-None-Match"); - if (match != NULL && etag == match) { + if (match != NULL && etag == match) r->status = HTTP_NOT_MODIFIED; - return OK; - } - - // Send response - ap_set_content_type(r, "text/xml"); apr_table_setn(r->headers_out, "ETag", etag.c_str()); - ap_rputs(str.c_str(), r); + ap_rputs(sval.c_str(), r); return OK; } /** + * Read the content of a POST. + */ +const list<std::string> read(request_rec* r) { + char b[2048]; + const int n = ap_get_client_block(r, b, 2048); + if (n <= 0) + return list<std::string>(); + return cons(std::string(b, n), read(r)); +} + +/** + * Converts the args received in a POST to a list of key value pairs. + */ +const list<list<value> > postArgs(list<value> a) { + if (isNil(a)) + return list<list<value> >(); + const list<value> l = car(a); + return cons(l, postArgs(cdr(a))); +} + +/** * Handle an HTTP POST request. */ const int post(request_rec* r) { + + // Setup the script evaluator + Env globalEnv = setupEnvironment(); + std::ostringstream nullos; + setupEvalOut(nullos); + + // Open the component implementation + const std::string impl = contribution(r) + implementation(r); + std::ifstream is(impl.c_str(), std::ios_base::in); + if (is.fail() || is.bad()) + return HTTP_NOT_FOUND; + + // Read the JSON request + const list<std::string> req = read(r); + JSONContext cx; + const list<value> json = readJSON(cx, req); + const list<list<value> > args = postArgs(json); + + // Extract the request id, method and params + const value id = cadr(assoc(value("id"), args)); + const value method = std::string(cadr(assoc(value("method"), args))).c_str(); + const list<value> params = (list<value>)cadr(assoc(value("params"), args)); + + // Build expr to evaluate + const value expr = cons<value>(method, params); + std::cout<< "expr: " << expr << std::endl; + std::cout.flush(); + + // Evaluate the expr + const tuscany::value val = evalLoop(is, expr, globalEnv); + if (isNil(val)) + return HTTP_INTERNAL_SERVER_ERROR; + std::cout<< "val: " << val << std::endl; + std::cout.flush(); + + // Convert the expr value to JSON + failable<list<std::string>, std::string> jsval = writeJSON(cx, makeList<value>(makeList<value>("id", id), makeList<value>("result", val))); + if (!hasValue(jsval)) + return HTTP_INTERNAL_SERVER_ERROR; + + // Send the JSON response + ap_set_content_type(r, "application/json-rpc"); + std::ostringstream os; + write(jsval, os); + ap_rputs(os.str().c_str(), r); + return OK; } @@ -166,26 +315,6 @@ const int post(request_rec* r) { * Handle an HTTP PUT request. */ const int put(request_rec* r) { - std::ostringstream sos; - char buffer[2049]; - for ( ; ; ) - { - int size = ap_get_client_block(r, buffer, 2048); - if (size > 0) - { - buffer[size] = '\0'; - sos << buffer; - } - else if (size == 0) - { - break; - } - else if (size < 0) - { - return HTTP_INTERNAL_SERVER_ERROR; - } - } - std::string input = sos.str(); return OK; } @@ -203,13 +332,9 @@ int handler(request_rec *r) { if(strcmp(r->handler, "mod_tuscany")) return DECLINED; - // Get the server and dir config - ServerConf& serverConf = *(ServerConf*)ap_get_module_config(r->server->module_config, &mod_tuscany); - DirConf& dirConf = *(DirConf*)ap_get_module_config(r->per_dir_config, &mod_tuscany); - // Log the request if(logRequests) - logRequest(r, serverConf, dirConf); + logRequest(r); // Set up the read policy const int rc = ap_setup_client_block(r, REQUEST_CHUNKED_DECHUNK); @@ -220,10 +345,9 @@ int handler(request_rec *r) { r->chunked = true; apr_table_setn(r->headers_out, "Connection", "close"); + // Handle HTTP method if (r->header_only) return OK; - - // Handle HTTP method if(r->method_number == M_GET) return get(r); if(r->method_number == M_POST) @@ -243,31 +367,24 @@ const char *confHome(cmd_parms *cmd, void *dummy, const char *arg) { conf->home = apr_pstrdup(cmd->pool, arg); return NULL; } -const char *confPath(cmd_parms *cmd, void *c, const char *arg) { +const char *confContribution(cmd_parms *cmd, void *c, const char *arg) { DirConf *conf = (DirConf*)c; - conf->path = apr_pstrdup(cmd->pool, arg); + conf->contribution = apr_pstrdup(cmd->pool, arg); return NULL; } -const char *confRoot(cmd_parms *cmd, void *c, const char *arg) { - DirConf *conf = (DirConf*)c; - conf->root = apr_pstrdup(cmd->pool, arg); - return NULL; -} -const char *confURI(cmd_parms *cmd, void *c, const char *arg) { +const char *confComponent(cmd_parms *cmd, void *c, const char *arg) { DirConf *conf = (DirConf*)c; - conf->uri = apr_pstrdup(cmd->pool, arg); + conf->component = apr_pstrdup(cmd->pool, arg); return NULL; } -const char *confComponent(cmd_parms *cmd, void *c, const char *arg) { +const char *confImplementation(cmd_parms *cmd, void *c, const char *arg) { DirConf *conf = (DirConf*)c; - conf->component = apr_pstrdup(cmd->pool, arg); + conf->implementation = apr_pstrdup(cmd->pool, arg); return NULL; } void *makeDirConf(apr_pool_t *p, char *dirspec) { DirConf* conf = (DirConf*)apr_palloc(p, sizeof(*conf)); - conf->path = ""; - conf->root = ""; - conf->uri = ""; + conf->contribution = ""; conf->component = ""; return conf; } @@ -281,11 +398,10 @@ void* makeServerConf(apr_pool_t *p, server_rec *s) { * HTTP server module declarations. */ const command_rec commands[] = { - AP_INIT_TAKE1("home", (const char*(*)())confHome, NULL, RSRC_CONF, "Tuscany home directory"), - AP_INIT_TAKE1("path", (const char*(*)())confPath, NULL, ACCESS_CONF, "Tuscany SCA composite search path"), - AP_INIT_TAKE1("root", (const char*(*)())confRoot, NULL, ACCESS_CONF, "Tuscany root SCA configuration path"), - AP_INIT_TAKE1("uri", (const char*(*)())confURI, NULL, ACCESS_CONF, "Tuscany SCA system base URI"), - AP_INIT_TAKE1("component", (const char*(*)())confComponent, NULL, ACCESS_CONF, "SCA component name"), + AP_INIT_TAKE1("TuscanyHome", (const char*(*)())confHome, NULL, RSRC_CONF, "Tuscany home directory"), + AP_INIT_TAKE1("SCAContribution", (const char*(*)())confContribution, NULL, ACCESS_CONF, "SCA contribution location"), + AP_INIT_TAKE1("SCAComponent", (const char*(*)())confComponent, NULL, ACCESS_CONF, "SCA component name"), + AP_INIT_TAKE1("SCAImplementation", (const char*(*)())confImplementation, NULL, ACCESS_CONF, "SCA component implementation"), {NULL} }; diff --git a/cpp/sca/modules/json/json-test.cpp b/cpp/sca/modules/json/json-test.cpp index 5a1dcc5d51..1ebece647b 100644 --- a/cpp/sca/modules/json/json-test.cpp +++ b/cpp/sca/modules/json/json-test.cpp @@ -34,7 +34,7 @@ namespace tuscany { bool testJSEval() { JSONContext cx; - std::string script("(function testJSON(n){ return JSON.parse(JSON.stringify(n)) })(5)"); + const std::string script("(function testJSON(n){ return JSON.parse(JSON.stringify(n)) })(5)"); jsval rval; assert(JS_EvaluateScript(cx, cx.getGlobal(), script.c_str(), script.length(), "testJSON.js", 1, &rval)); const std::string r(JS_GetStringBytes(JS_ValueToString(cx, rval))); @@ -48,21 +48,20 @@ std::ostringstream* jsonWriter(std::ostringstream* os, const std::string& s) { } bool testJSON() { - JSONContext cx; + const JSONContext cx; - list<value> phones = makeList<value> (std::string("408-1234"), std::string("650-1234")); - list<value> l = makeList<value> (makeList<value> ("phones", phones), makeList<value> ("lastName", std::string("test\ttab")), makeList<value> ("firstName", std::string("test1"))); + const list<value> phones = makeList<value> (std::string("408-1234"), std::string("650-1234")); + const list<value> l = makeList<value> (makeList<value> ("phones", phones), makeList<value> ("lastName", std::string("test\ttab")), makeList<value> ("firstName", std::string("test1"))); print(l, std::cout); std::cout << std::endl; std::ostringstream os; - lambda<std::ostringstream*(std::ostringstream*, std::string)> writer(jsonWriter); - writeJSON(cx, writer, &os, l); + writeJSON<std::ostringstream*>(cx, jsonWriter, &os, l); std::cout << os.str() << std::endl; std::istringstream is(os.str()); - list<std::string> il = makeStreamList(is); - list<value> r = readJSON(cx, il); + const list<std::string> il = makeStreamList(is); + const list<value> r = readJSON(cx, il); print(r, std::cout); std::cout << std::endl; assert(r == l); @@ -74,6 +73,14 @@ bool testJSON() { return true; } +bool testJSONRPC() { + const std::string lm("{\"id\": 1, \"method\": \"system.listMethods\", \"params\": []}"); + JSONContext cx; + const list<value> v = readJSON(cx, makeList(lm)); + std::cout << v << std::endl; + return true; +} + } int main() { @@ -81,6 +88,7 @@ int main() { tuscany::testJSEval(); tuscany::testJSON(); + tuscany::testJSONRPC(); std::cout << "OK" << std::endl; diff --git a/cpp/sca/modules/json/json.hpp b/cpp/sca/modules/json/json.hpp index e9938672c5..f6ed9202c3 100644 --- a/cpp/sca/modules/json/json.hpp +++ b/cpp/sca/modules/json/json.hpp @@ -32,6 +32,7 @@ #include <string> #include "list.hpp" #include "value.hpp" +#include "monad.hpp" namespace tuscany { @@ -172,8 +173,9 @@ const value jsValToValue(const JSONContext& cx, const jsval& jsv) { return value((bool)JSVAL_TO_BOOLEAN(jsv)); } case JSTYPE_NUMBER: { - jsdouble* jsd = JSVAL_TO_DOUBLE(jsv); - return value((double)*jsd); + jsdouble jsd; + JS_ValueToNumber(cx, jsv, &jsd); + return value((double)jsd); } case JSTYPE_OBJECT: { JSObject* o = JSVAL_TO_OBJECT(jsv); @@ -191,39 +193,39 @@ const value jsValToValue(const JSONContext& cx, const jsval& jsv) { /** * Consumes JSON strings and populates a JS object. */ -bool consumeJSON(const JSONContext& cx, JSONParser* parser, const list<std::string>& ilist) { +failable<bool, std::string> consumeJSON(const JSONContext& cx, JSONParser* parser, const list<std::string>& ilist) { if (isNil(ilist)) return true; JSString* jstr = JS_NewStringCopyZ(cx, car(ilist).c_str()); if(!JS_ConsumeJSONText(cx, parser, JS_GetStringChars(jstr), JS_GetStringLength(jstr))) - return false; + return "JS_ConsumeJSONText failed"; return consumeJSON(cx, parser, cdr(ilist)); } /** * Read JSON tokens from list of strings. */ -const list<value> readJSON(const JSONContext& cx, const list<std::string>& ilist) { +const failable<list<value>, std::string> readJSON(const JSONContext& cx, const list<std::string>& ilist) { jsval val; JSONParser* parser = JS_BeginJSONParse(cx, &val); if(parser == NULL) - return list<value> (); + return std::string("JS_BeginJSONParse failed"); - bool ok = consumeJSON(cx, parser, ilist); + const failable<bool, std::string> consumed = consumeJSON(cx, parser, ilist); if(!JS_FinishJSONParse(cx, parser, JSVAL_NULL)) - return list<value> (); - if(!ok) - return list<value> (); + return std::string("JS_FinishJSONParse failed"); + if(!hasValue(consumed)) + return std::string(consumed); - return jsValToValue(cx, val); + return list<value>(jsValToValue(cx, val)); } /** * Returns true if a list represents a JS array. */ const bool isJSArray(const list<value>& l) { - if(l == list<value> ()) + if(isNil(l)) return false; value v = car(l); if(isList(v)) { @@ -240,8 +242,7 @@ const bool isJSArray(const list<value>& l) { */ JSObject* valuesToJSElements(const JSONContext& cx, JSObject* a, const list<value>& l, int i) { const jsval valueToJSVal(const JSONContext& cx, const value& val); - - if (l == list<value>()) + if (isNil(l)) return a; jsval pv = valueToJSVal(cx, car(l)); JS_SetElement(cx, a, i, &pv); @@ -253,8 +254,7 @@ JSObject* valuesToJSElements(const JSONContext& cx, JSObject* a, const list<valu */ JSObject* valuesToJSProperties(const JSONContext& cx, JSObject* o, const list<value>& l) { const jsval valueToJSVal(const JSONContext& cx, const value& val); - - if(l == list<value> ()) + if(isNil(l)) return o; const list<value> p = car(l); jsval pv = valueToJSVal(cx, cadr(p)); @@ -274,8 +274,7 @@ const jsval valueToJSVal(const JSONContext& cx, const value& val) { return BOOLEAN_TO_JSVAL((bool)val); } case value::Number: { - jsdouble d = (double)val; - return DOUBLE_TO_JSVAL(&d); + return DOUBLE_TO_JSVAL(JS_NewDouble(cx, (double)val)); } case value::List: { if (isJSArray(val)) { @@ -312,12 +311,13 @@ template<typename R> JSBool writeCallback(const jschar *buf, uint32 len, void *d } /** - * Write a list of values as a JSON document. + * Convert a list of values to a JSON document. */ -template<typename R> const R writeJSON(const JSONContext& cx, const lambda<R(R, std::string)>& reduce, const R& initial, const list<value>& l) { +template<typename R> const failable<R, std::string> writeJSON(const JSONContext& cx, const lambda<R(R, std::string)>& reduce, const R& initial, const list<value>& l) { jsval val = valueToJSVal(cx, l); JSONWriteContext<R> wcx(cx, reduce, initial); - JS_Stringify(cx, &val, NULL, JSVAL_NULL, writeCallback<R>, &wcx); + if (!JS_Stringify(cx, &val, NULL, JSVAL_NULL, writeCallback<R>, &wcx)) + return std::string("JS_Stringify failed"); return wcx.accum; } @@ -326,11 +326,13 @@ const list<std::string> writeJSONList(const list<std::string>& listSoFar, const } /** - * Write a list of values as a JSON document represented as a list of strings. + * Convert a list of values to a JSON document represented as a list of strings. */ -const list<std::string> writeJSON(const JSONContext& cx, const list<value>& l) { - lambda<list<std::string>(list<std::string>, std::string)> writer(writeJSONList); - return reverse(writeJSON(cx, writer, list<std::string>(), l)); +const failable<list<std::string>, std::string> writeJSON(const JSONContext& cx, const list<value>& l) { + const failable<list<std::string>, std::string> ls = writeJSON<list<std::string> >(cx, writeJSONList, list<std::string>(), l); + if (!hasValue(ls)) + return ls; + return reverse(list<std::string>(ls)); } } diff --git a/cpp/sca/test/store-script/htdocs/.htaccess b/cpp/sca/test/store-script/htdocs/.htaccess new file mode 100644 index 0000000000..e2e343b6b2 --- /dev/null +++ b/cpp/sca/test/store-script/htdocs/.htaccess @@ -0,0 +1,19 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you 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. + +DirectoryIndex store.html diff --git a/cpp/sca/test/store-script/htdocs/store.html b/cpp/sca/test/store-script/htdocs/store.html new file mode 100644 index 0000000000..42f58ab40c --- /dev/null +++ b/cpp/sca/test/store-script/htdocs/store.html @@ -0,0 +1,162 @@ +<!--
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements. See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership. The ASF licenses this file
+ * to you 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.
+-->
+<html>
+<head>
+<title>Store</title>
+
+<script type="text/javascript" src="store.js"></script>
+
+<script language="JavaScript">
+
+ //@Reference
+ var catalog = new tuscany.sca.Reference("catalog");
+
+ //@Reference
+ //var shoppingCart = new tuscany.sca.Reference("shoppingCart");
+
+ //@Reference
+ //var shoppingTotal = new tuscany.sca.Reference("shoppingTotal");
+
+ var catalogItems;
+
+ function catalog_getResponse(items,exception) {
+ if(exception){
+ alert(exception.message);
+ return;
+ }
+ var catalog = "";
+
+ for (var i=0; i<items.length; i++) {
+ var item = items[i].name + ' - ' + items[i].price;
+ catalog += '<input name="items" type="checkbox" value="' +
+ item + '">' + item + ' <br>';
+ }
+ document.getElementById('catalog').innerHTML=catalog;
+ catalogItems = items;
+ }
+
+ function shoppingCart_getResponse(feed) {
+ if (feed != null) {
+ var entries = feed.getElementsByTagName("entry");
+ var list = "";
+ for (var i=0; i<entries.length; i++) {
+ var content = entries[i].getElementsByTagName("content")[0];
+ var name = content.getElementsByTagName("name")[0].firstChild.nodeValue;
+ var price = content.getElementsByTagName("price")[0].firstChild.nodeValue;
+ list += name + ' - ' + price + ' <br>';
+ }
+ document.getElementById("shoppingCart").innerHTML = list;
+
+ if (entries.length != 0) {
+ try {
+ //shoppingTotal.getTotal(shoppingTotal_getTotalResponse);
+ }
+ catch(e){
+ alert(e);
+ }
+ }
+ }
+ }
+
+ function shoppingTotal_getTotalResponse(total,exception) {
+ if(exception) {
+ alert(exception.message);
+ return;
+ }
+ document.getElementById('total').innerHTML = total;
+ }
+
+ function shoppingCart_postResponse(entry) {
+ //shoppingCart.get("", shoppingCart_getResponse);
+ }
+
+ function addToCart() {
+ var items = document.catalogForm.items;
+ var j = 0;
+ for (var i=0; i<items.length; i++)
+ if (items[i].checked) {
+
+ var entry = '<entry xmlns="http://www.w3.org/2005/Atom"><title>item</title><content type="text/xml">' +
+ '<Item xmlns="http://services/">' +
+ '<name xmlns="">' + catalogItems[i].name + '</name>' + '<price xmlns="">' + catalogItems[i].price + '</price>' +
+ '</Item>' + '</content></entry>';
+ shoppingCart.post(entry, shoppingCart_postResponse);
+ items[i].checked = false;
+ }
+ }
+ function checkoutCart() {
+ document.getElementById('store').innerHTML='<h2>' +
+ 'Thanks for Shopping With Us!</h2>'+
+ '<h2>Your Order</h2>'+
+ '<form name="orderForm">'+
+ document.getElementById('shoppingCart').innerHTML+
+ '<br>'+
+ document.getElementById('total').innerHTML+
+ '<br>'+
+ '<br>'+
+ '<input type="submit" value="Continue Shopping">'+
+ '</form>';
+ shoppingCart.del("", null);
+ }
+ function deleteCart() {
+ shoppingCart.del("", null);
+ document.getElementById('shoppingCart').innerHTML = "";
+ document.getElementById('total').innerHTML = "";
+ }
+
+ function init() {
+
+ try {
+ catalog.get(catalog_getResponse);
+ //shoppingCart.get("", shoppingCart_getResponse);
+ }
+ catch(e){
+ alert(e);
+ }
+ }
+
+</script>
+
+</head>
+
+<body onload="init()">
+<h1>Store</h1>
+ <div id="store">
+ <h2>Catalog</h2>
+ <form name="catalogForm">
+ <div id="catalog" ></div>
+ <br>
+ <input type="button" onClick="addToCart()" value="Add to Cart">
+ </form>
+
+ <br>
+
+ <h2>Your Shopping Cart</h2>
+ <form name="shoppingCartForm">
+ <div id="shoppingCart"></div>
+ <br>
+ <div id="total"></div>
+ <br>
+ <input type="button" onClick="checkoutCart()" value="Checkout">
+ <input type="button" onClick="deleteCart()" value="Empty">
+ <a href="../ShoppingCart/Cart/">(feed)</a>
+ </form>
+ </div>
+</body>
+</html>
diff --git a/cpp/sca/test/store-script/htdocs/store.js b/cpp/sca/test/store-script/htdocs/store.js new file mode 100644 index 0000000000..47af7c01f7 --- /dev/null +++ b/cpp/sca/test/store-script/htdocs/store.js @@ -0,0 +1,661 @@ + +/* Apache Tuscany SCA Widget header */ + +/* + * JSON-RPC JavaScript client + * + * $Id: jsonrpc.js,v 1.36.2.3 2006/03/08 15:09:37 mclark Exp $ + * + * Copyright (c) 2003-2004 Jan-Klaas Kollhof + * Copyright (c) 2005 Michael Clark, Metaparadigm Pte Ltd + * + * This code is based on Jan-Klaas' JavaScript o lait library (jsolait). + * + * 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. + * + */ + +/* + * Modifications for Apache Tuscany: + * - JSONRpcClient_createMethod changed so callback is last arg + */ + +/* escape a character */ + +escapeJSONChar = +function escapeJSONChar(c) +{ + if(c == "\"" || c == "\\") return "\\" + c; + else if (c == "\b") return "\\b"; + else if (c == "\f") return "\\f"; + else if (c == "\n") return "\\n"; + else if (c == "\r") return "\\r"; + else if (c == "\t") return "\\t"; + var hex = c.charCodeAt(0).toString(16); + if(hex.length == 1) return "\\u000" + hex; + else if(hex.length == 2) return "\\u00" + hex; + else if(hex.length == 3) return "\\u0" + hex; + else return "\\u" + hex; +}; + + +/* encode a string into JSON format */ + +escapeJSONString = +function escapeJSONString(s) +{ + /* The following should suffice but Safari's regex is b0rken + (doesn't support callback substitutions) + return "\"" + s.replace(/([^\u0020-\u007f]|[\\\"])/g, + escapeJSONChar) + "\""; + */ + + /* Rather inefficient way to do it */ + var parts = s.split(""); + for(var i=0; i < parts.length; i++) { + var c =parts[i]; + if(c == '"' || + c == '\\' || + c.charCodeAt(0) < 32 || + c.charCodeAt(0) >= 128) + parts[i] = escapeJSONChar(parts[i]); + } + return "\"" + parts.join("") + "\""; +}; + + +/* Marshall objects to JSON format */ + +toJSON = function toJSON(o) +{ + if(o == null) { + return "null"; + } else if(o.constructor == String) { + return escapeJSONString(o); + } else if(o.constructor == Number) { + return o.toString(); + } else if(o.constructor == Boolean) { + return o.toString(); + } else if(o.constructor == Date) { + return '{javaClass: "java.util.Date", time: ' + o.valueOf() +'}'; + } else if(o.constructor == Array) { + var v = []; + for(var i = 0; i < o.length; i++) v.push(toJSON(o[i])); + return "[" + v.join(", ") + "]"; + } else { + var v = []; + for(attr in o) { + if(o[attr] == null) v.push("\"" + attr + "\": null"); + else if(typeof o[attr] == "function"); /* skip */ + else v.push(escapeJSONString(attr) + ": " + toJSON(o[attr])); + } + return "{" + v.join(", ") + "}"; + } +}; + + +/* JSONRpcClient constructor */ + +JSONRpcClient = +function JSONRpcClient_ctor(serverURL, user, pass, objectID) +{ + this.serverURL = serverURL; + this.user = user; + this.pass = pass; + this.objectID = objectID; + + /* Add standard methods */ + if(this.objectID) { + this._addMethods(["listMethods"]); + var req = this._makeRequest("listMethods", []); + } else { + this._addMethods(["system.listMethods"]); + var req = this._makeRequest("system.listMethods", []); + } + var m = this._sendRequest(req); + this._addMethods(m); +}; + + +/* JSONRpcCLient.Exception */ + +JSONRpcClient.Exception = +function JSONRpcClient_Exception_ctor(code, message, javaStack) +{ + this.code = code; + var name; + if(javaStack) { + this.javaStack = javaStack; + var m = javaStack.match(/^([^:]*)/); + if(m) name = m[0]; + } + if(name) this.name = name; + else this.name = "JSONRpcClientException"; + this.message = message; +}; + +JSONRpcClient.Exception.CODE_REMOTE_EXCEPTION = 490; +JSONRpcClient.Exception.CODE_ERR_CLIENT = 550; +JSONRpcClient.Exception.CODE_ERR_PARSE = 590; +JSONRpcClient.Exception.CODE_ERR_NOMETHOD = 591; +JSONRpcClient.Exception.CODE_ERR_UNMARSHALL = 592; +JSONRpcClient.Exception.CODE_ERR_MARSHALL = 593; + +JSONRpcClient.Exception.prototype = new Error(); + +JSONRpcClient.Exception.prototype.toString = +function JSONRpcClient_Exception_toString(code, msg) +{ + return this.name + ": " + this.message; +}; + + +/* Default top level exception handler */ + +JSONRpcClient.default_ex_handler = +function JSONRpcClient_default_ex_handler(e) { alert(e); }; + + +/* Client settable variables */ + +JSONRpcClient.toplevel_ex_handler = JSONRpcClient.default_ex_handler; +JSONRpcClient.profile_async = false; +JSONRpcClient.max_req_active = 1; +JSONRpcClient.requestId = 1; + + +/* JSONRpcClient implementation */ + +JSONRpcClient.prototype._createMethod = +function JSONRpcClient_createMethod(methodName) +{ + var fn=function() + { + var args = []; + var callback = null; + for(var i=0;i<arguments.length;i++) args.push(arguments[i]); + +/* TUSCANY change callback to be last arg instead of first to match binding.ajax + if(typeof args[0] == "function") callback = args.shift(); +*/ + if(typeof args[arguments.length-1] == "function") callback = args.pop(); + + var req = fn.client._makeRequest.call(fn.client, fn.methodName, + args, callback); + if(callback == null) { + return fn.client._sendRequest.call(fn.client, req); + } else { + JSONRpcClient.async_requests.push(req); + JSONRpcClient.kick_async(); + return req.requestId; + } + }; + fn.client = this; + fn.methodName = methodName; + return fn; +}; + +JSONRpcClient.prototype._addMethods = +function JSONRpcClient_addMethods(methodNames) +{ + for(var i=0; i<methodNames.length; i++) { + var obj = this; + var names = methodNames[i].split("."); + for(var n=0; n<names.length-1; n++) { + var name = names[n]; + if(obj[name]) { + obj = obj[name]; + } else { + obj[name] = new Object(); + obj = obj[name]; + } + } + var name = names[names.length-1]; + if(!obj[name]) { + var method = this._createMethod(methodNames[i]); + obj[name] = method; + } + } +}; + +JSONRpcClient._getCharsetFromHeaders = +function JSONRpcClient_getCharsetFromHeaders(http) +{ + try { + var contentType = http.getResponseHeader("Content-type"); + var parts = contentType.split(/\s*;\s*/); + for(var i =0; i < parts.length; i++) { + if(parts[i].substring(0, 8) == "charset=") + return parts[i].substring(8, parts[i].length); + } + } catch (e) {} + return "UTF-8"; /* default */ +}; + +/* Async queue globals */ +JSONRpcClient.async_requests = []; +JSONRpcClient.async_inflight = {}; +JSONRpcClient.async_responses = []; +JSONRpcClient.async_timeout = null; +JSONRpcClient.num_req_active = 0; + +JSONRpcClient._async_handler = +function JSONRpcClient_async_handler() +{ + JSONRpcClient.async_timeout = null; + + while(JSONRpcClient.async_responses.length > 0) { + var res = JSONRpcClient.async_responses.shift(); + if(res.canceled) continue; + if(res.profile) res.profile.dispatch = new Date(); + try { + res.cb(res.result, res.ex, res.profile); + } catch(e) { + JSONRpcClient.toplevel_ex_handler(e); + } + } + + while(JSONRpcClient.async_requests.length > 0 && + JSONRpcClient.num_req_active < JSONRpcClient.max_req_active) { + var req = JSONRpcClient.async_requests.shift(); + if(req.canceled) continue; + req.client._sendRequest.call(req.client, req); + } +}; + +JSONRpcClient.kick_async = +function JSONRpcClient_kick_async() +{ + if(JSONRpcClient.async_timeout == null) + JSONRpcClient.async_timeout = + setTimeout(JSONRpcClient._async_handler, 0); +}; + +JSONRpcClient.cancelRequest = +function JSONRpcClient_cancelRequest(requestId) +{ + /* If it is in flight then mark it as canceled in the inflight map + and the XMLHttpRequest callback will discard the reply. */ + if(JSONRpcClient.async_inflight[requestId]) { + JSONRpcClient.async_inflight[requestId].canceled = true; + return true; + } + + /* If its not in flight yet then we can just mark it as canceled in + the the request queue and it will get discarded before being sent. */ + for(var i in JSONRpcClient.async_requests) { + if(JSONRpcClient.async_requests[i].requestId == requestId) { + JSONRpcClient.async_requests[i].canceled = true; + return true; + } + } + + /* It may have returned from the network and be waiting for its callback + to be dispatched, so mark it as canceled in the response queue + and the response will get discarded before calling the callback. */ + for(var i in JSONRpcClient.async_responses) { + if(JSONRpcClient.async_responses[i].requestId == requestId) { + JSONRpcClient.async_responses[i].canceled = true; + return true; + } + } + + return false; +}; + +JSONRpcClient.prototype._makeRequest = +function JSONRpcClient_makeRequest(methodName, args, cb) +{ + var req = {}; + req.client = this; + req.requestId = JSONRpcClient.requestId++; + + var obj = {}; + obj.id = req.requestId; + if (this.objectID) + obj.method = ".obj#" + this.objectID + "." + methodName; + else + obj.method = methodName; + obj.params = args; + + if (cb) req.cb = cb; + if (JSONRpcClient.profile_async) + req.profile = { "submit": new Date() }; + req.data = toJSON(obj); + + return req; +}; + +JSONRpcClient.prototype._sendRequest = +function JSONRpcClient_sendRequest(req) +{ + if(req.profile) req.profile.start = new Date(); + + /* Get free http object from the pool */ + var http = JSONRpcClient.poolGetHTTPRequest(); + JSONRpcClient.num_req_active++; + + /* Send the request */ + if (typeof(this.user) == "undefined") { + http.open("POST", this.serverURL, (req.cb != null)); + } else { + http.open("POST", this.serverURL, (req.cb != null), this.user, this.pass); + } + + /* setRequestHeader is missing in Opera 8 Beta */ + try { http.setRequestHeader("Content-type", "text/plain"); } catch(e) {} + + /* Construct call back if we have one */ + if(req.cb) { + var self = this; + http.onreadystatechange = function() { + if(http.readyState == 4) { + http.onreadystatechange = function () {}; + var res = { "cb": req.cb, "result": null, "ex": null}; + if (req.profile) { + res.profile = req.profile; + res.profile.end = new Date(); + } + try { res.result = self._handleResponse(http); } + catch(e) { res.ex = e; } + if(!JSONRpcClient.async_inflight[req.requestId].canceled) + JSONRpcClient.async_responses.push(res); + delete JSONRpcClient.async_inflight[req.requestId]; + JSONRpcClient.kick_async(); + } + }; + } else { + http.onreadystatechange = function() {}; + } + + JSONRpcClient.async_inflight[req.requestId] = req; + + try { + http.send(req.data); + } catch(e) { + JSONRpcClient.poolReturnHTTPRequest(http); + JSONRpcClient.num_req_active--; + throw new JSONRpcClient.Exception + (JSONRpcClient.Exception.CODE_ERR_CLIENT, "Connection failed"); + } + + if(!req.cb) return this._handleResponse(http); +}; + +JSONRpcClient.prototype._handleResponse = +function JSONRpcClient_handleResponse(http) +{ + /* Get the charset */ + if(!this.charset) { + this.charset = JSONRpcClient._getCharsetFromHeaders(http); + } + + /* Get request results */ + var status, statusText, data; + try { + status = http.status; + statusText = http.statusText; + data = http.responseText; + } catch(e) { + JSONRpcClient.poolReturnHTTPRequest(http); + JSONRpcClient.num_req_active--; + JSONRpcClient.kick_async(); + throw new JSONRpcClient.Exception + (JSONRpcClient.Exception.CODE_ERR_CLIENT, "Connection failed"); + } + + /* Return http object to the pool; */ + JSONRpcClient.poolReturnHTTPRequest(http); + JSONRpcClient.num_req_active--; + + /* Unmarshall the response */ + if(status != 200) { + throw new JSONRpcClient.Exception(status, statusText); + } + var obj; + try { + eval("obj = " + data); + } catch(e) { + throw new JSONRpcClient.Exception(550, "error parsing result"); + } + if(obj.error) + throw new JSONRpcClient.Exception(obj.error.code, obj.error.msg, + obj.error.trace); + var res = obj.result; + + /* Handle CallableProxy */ + if(res && res.objectID && res.JSONRPCType == "CallableReference") + return new JSONRpcClient(this.serverURL, this.user, + this.pass, res.objectID); + + return res; +}; + + +/* XMLHttpRequest wrapper code */ + +/* XMLHttpRequest pool globals */ +JSONRpcClient.http_spare = []; +JSONRpcClient.http_max_spare = 8; + +JSONRpcClient.poolGetHTTPRequest = +function JSONRpcClient_pool_getHTTPRequest() +{ + if(JSONRpcClient.http_spare.length > 0) { + return JSONRpcClient.http_spare.pop(); + } + return JSONRpcClient.getHTTPRequest(); +}; + +JSONRpcClient.poolReturnHTTPRequest = +function JSONRpcClient_poolReturnHTTPRequest(http) +{ + if(JSONRpcClient.http_spare.length >= JSONRpcClient.http_max_spare) + delete http; + else + JSONRpcClient.http_spare.push(http); +}; + +JSONRpcClient.msxmlNames = [ "MSXML2.XMLHTTP.5.0", + "MSXML2.XMLHTTP.4.0", + "MSXML2.XMLHTTP.3.0", + "MSXML2.XMLHTTP", + "Microsoft.XMLHTTP" ]; + +JSONRpcClient.getHTTPRequest = +function JSONRpcClient_getHTTPRequest() +{ + /* Mozilla XMLHttpRequest */ + try { + JSONRpcClient.httpObjectName = "XMLHttpRequest"; + return new XMLHttpRequest(); + } catch(e) {} + + /* Microsoft MSXML ActiveX */ + for (var i=0;i < JSONRpcClient.msxmlNames.length; i++) { + try { + JSONRpcClient.httpObjectName = JSONRpcClient.msxmlNames[i]; + return new ActiveXObject(JSONRpcClient.msxmlNames[i]); + } catch (e) {} + } + + /* None found */ + JSONRpcClient.httpObjectName = null; + throw new JSONRpcClient.Exception(0, "Can't create XMLHttpRequest object"); +}; + + +/* + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you 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. + */ + +function AtomClient(uri) { + + this.msxmlNames = [ "MSXML2.XMLHTTP.5.0", + "MSXML2.XMLHTTP.4.0", + "MSXML2.XMLHTTP.3.0", + "MSXML2.XMLHTTP", + "Microsoft.XMLHTTP" ]; + + this.uri=uri; + + this.get = function(id, responseFunction) { + var xhr = this.createXMLHttpRequest(); + xhr.onreadystatechange = function() { + if (xhr.readyState == 4) { + if (xhr.status == 200) { + var strDocument = xhr.responseText; + var xmlDocument = xhr.responseXML; + if(!xmlDocument || xmlDocument.childNodes.length==0){ + xmlDocument = (new DOMParser()).parseFromString(strDocument, "text/xml"); + } + if (responseFunction != null) responseFunction(xmlDocument); + } else { + alert("get - Error getting data from the server"); + } + } + } + xhr.open("GET", uri + '/' + id, true); + xhr.send(null); + } + + this.post = function (entry, responseFunction) { + var xhr = this.createXMLHttpRequest(); + xhr.onreadystatechange = function() { + if (xhr.readyState == 4) { + if (xhr.status == 201) { + var strDocument = xhr.responseText; + var xmlDocument = xhr.responseXML; + if(!xmlDocument || xmlDocument.childNodes.length==0){ + xmlDocument = (new DOMParser()).parseFromString(strDocument, "text/xml"); + } + if (responseFunction != null) responseFunction(xmlDocument); + } else { + alert("post - Error getting data from the server"); + } + } + } + xhr.open("POST", uri, true); + xhr.setRequestHeader("Content-Type", "application/atom+xml"); + xhr.send(entry); + } + + this.put = function (id, entry, responseFunction) { + var xhr = this.createXMLHttpRequest(); + xhr.onreadystatechange = function() { + if (xhr.readyState == 4) { + if (xhr.status == 200) { + var strDocument = xhr.responseText; + var xmlDocument = xhr.responseXML; + if(!xmlDocument || xmlDocument.childNodes.length==0){ + xmlDocument = (new DOMParser()).parseFromString(strDocument, "text/xml"); + } + if (responseFunction != null) responseFunction(xmlDocument); + } else { + alert("put - Error getting data from the server"); + } + } + } + xhr.open("PUT", uri + '/' + id, true); + xhr.setRequestHeader("Content-Type", "application/atom+xml"); + xhr.send(entry); + } + + this.del = function (id, responseFunction) { + var xhr = this.createXMLHttpRequest(); + xhr.onreadystatechange = function() { + if (xhr.readyState == 4) { + if (xhr.status == 200) { + if (responseFunction != null) responseFunction(); + } else { + alert("delete - Error getting data from the server"); + } + } + } + xhr.open("DELETE", uri + '/' + id, true); + xhr.send(null); + } + this.createXMLHttpRequest = function () { + /* Mozilla XMLHttpRequest */ + try {return new XMLHttpRequest();} catch(e) {} + + /* Microsoft MSXML ActiveX */ + for (var i=0;i < this.msxmlNames.length; i++) { + try {return new ActiveXObject(this.msxmlNames[i]);} catch (e) {} + } + alert("XML http request not supported"); + return null; + } + if (typeof DOMParser == "undefined") { + DOMParser = function () {} + + DOMParser.prototype.parseFromString = function (str, contentType) { + if (typeof ActiveXObject != "undefined") { + var d = new ActiveXObject("MSXML.DomDocument"); + d.loadXML(str); + return d; + } else if (typeof XMLHttpRequest != "undefined") { + var req = new XMLHttpRequest; + req.open("GET", "data:" + (contentType || "application/xml") + + ";charset=utf-8," + encodeURIComponent(str), false); + if (req.overrideMimeType) { + req.overrideMimeType(contentType); + } + req.send(null); + return req.responseXML; + } + } + } +} + + + +/* Tuscany Reference/Property injection code */ + +if (!tuscany) { +var tuscany = {}; +} +if (!tuscany.sca) { +tuscany.sca = {}; +} + +tuscany.sca.propertyMap = new String(); +tuscany.sca.Property = function (name) { + return tuscany.sca.propertyMap[name]; +} + +tuscany.sca.referenceMap = new Object(); +tuscany.sca.referenceMap.catalog = new JSONRpcClient("/Catalog").Service; +//tuscany.sca.referenceMap.shoppingCart = new AtomClient("/ShoppingCart/Cart"); +//tuscany.sca.referenceMap.shoppingTotal = new JSONRpcClient("/ShoppingCart/Total").Service; +tuscany.sca.Reference = function (name) { + return tuscany.sca.referenceMap[name]; +} + +/** End of Apache Tuscany SCA Widget */ + diff --git a/cpp/sca/test/store-script/store-script-test.cpp b/cpp/sca/test/store-script/store-script-test.cpp index dff802b714..95e7ba0929 100644 --- a/cpp/sca/test/store-script/store-script-test.cpp +++ b/cpp/sca/test/store-script/store-script-test.cpp @@ -29,18 +29,42 @@ #include <string> #include "driver.hpp" -namespace store -{ +namespace store { bool contains(const std::string& str, const std::string& pattern) { return str.find(pattern) != str.npos; } bool testScript() { - std::ifstream is("store-script.scm", std::ios_base::in); + std::ifstream is("store.scm", std::ios_base::in); std::ostringstream os; tuscany::evalDriverRun(is, os); - assert(contains(os.str(), "List::(List::(String::'apple', (String::'USD', (String::'$', (Number::2.99, ())))), ())")); + assert(contains(os.str(), "List::(List::(List::(Symbol::name, (String::\"apple\", ())), (List::(Symbol::currency, (String::\"USD\", ())), (List::(Symbol::symbol, (String::\"$\", ())), (List::(Symbol::price, (Number::2.99, ())), ())))), ())")); + return true; +} + +const tuscany::value evalLoop(std::istream& is, const tuscany::value& req, tuscany::Env& globalEnv) { + tuscany::value in = tuscany::read(is); + if(tuscany::isNil(in)) + return tuscany::eval(req, globalEnv); + tuscany::eval(in, globalEnv); + return evalLoop(is, req, globalEnv); +} + +bool testEval() { + std::ifstream is("store.scm", std::ios_base::in); + std::ostringstream os; + + tuscany::setupEvalOut(os); + tuscany::Env globalEnv = tuscany::setupEnvironment(); + + const tuscany::value req(tuscany::makeList<tuscany::value>("storeui_service", std::string("getcatalog"))); + const tuscany::value res = evalLoop(is, req, globalEnv); + + std::ostringstream rs; + rs << res; + assert(contains(rs.str(), "List::(List::(List::(Symbol::name, (String::\"apple\", ())), (List::(Symbol::currency, (String::\"USD\", ())), (List::(Symbol::symbol, (String::\"$\", ())), (List::(Symbol::price, (Number::2.99, ())), ())))), (List::(List::(Symbol::name, (String::\"orange\", ())), (List::(Symbol::currency, (String::\"USD\", ())), (List::(Symbol::symbol, (String::\"$\", ())), (List::(Symbol::price, (Number::3.55, ())), ())))), (List::(List::(Symbol::name, (String::\"pear\", ())), (List::(Symbol::currency, (String::\"USD\", ())), (List::(Symbol::symbol, (String::\"$\", ())), (List::(Symbol::price, (Number::1.55, ())), ())))), ())))")); + return true; } @@ -51,6 +75,7 @@ int main() { std::cout << "Testing..." << std::endl; store::testScript(); + store::testEval(); std::cout << "OK" << std::endl; diff --git a/cpp/sca/test/store-script/store-script.scm b/cpp/sca/test/store-script/store.scm index aa4ea962f2..5bda81ca69 100644 --- a/cpp/sca/test/store-script/store-script.scm +++ b/cpp/sca/test/store-script/store.scm @@ -28,9 +28,9 @@ (define code "USD") (define symbol (converter "symbol" code)) - (list (list "apple" code symbol (convert 2.99)) - (list "orange" code symbol (convert 3.55)) - (list "pear" code symbol (convert 1.55)) + (list (list (list 'name "apple") (list 'currency code) (list 'symbol symbol) (list 'price (convert 2.99))) + (list (list 'name "orange") (list 'currency code) (list 'symbol symbol) (list 'price (convert 3.55))) + (list (list 'name "pear") (list 'currency code) (list 'symbol symbol) (list 'price (convert 1.55))) ) ) @@ -97,3 +97,8 @@ (define full (storeui_service "post" empty apple)) (display (storeui_service "getall" full)) +(; "Store UI JSON-RPC interop test case") + +(define (system.listMethods) (list "Service.get")) +(define (Service.get) (storeui_service "getcatalog")) + |