mirror of
https://github.com/samba-team/samba.git
synced 2025-11-25 00:23:52 +03:00
r18880: JSON-RPC work in progress
This commit is contained in:
committed by
Gerald (Jerry) Carter
parent
5a2e103ea0
commit
34bffbaebf
@@ -114,6 +114,7 @@ typedef struct
|
||||
char *szWINS_URL;
|
||||
char *szPrivateDir;
|
||||
char **jsInclude;
|
||||
char *jsonrpcBase;
|
||||
char **szPasswordServers;
|
||||
char *szSocketOptions;
|
||||
char *szRealm;
|
||||
@@ -542,6 +543,7 @@ static struct parm_struct parm_table[] = {
|
||||
{"modules dir", P_STRING, P_GLOBAL, &Globals.szModulesDir, NULL, NULL, FLAG_ADVANCED | FLAG_DEVELOPER},
|
||||
{"pid directory", P_STRING, P_GLOBAL, &Globals.szPidDir, NULL, NULL, FLAG_ADVANCED | FLAG_DEVELOPER},
|
||||
{"js include", P_LIST, P_GLOBAL, &Globals.jsInclude, NULL, NULL, FLAG_ADVANCED | FLAG_DEVELOPER},
|
||||
{"jsonrpc base", P_STRING, P_GLOBAL, &Globals.jsonrpcBase, NULL, NULL, FLAG_ADVANCED | FLAG_DEVELOPER},
|
||||
{"setup directory", P_STRING, P_GLOBAL, &Globals.szSetupDir, NULL, NULL, FLAG_ADVANCED | FLAG_DEVELOPER},
|
||||
|
||||
{"socket address", P_STRING, P_GLOBAL, &Globals.szSocketAddress, NULL, NULL, FLAG_DEVELOPER},
|
||||
@@ -911,6 +913,7 @@ _PUBLIC_ FN_GLOBAL_LIST(lp_auth_methods, &Globals.AuthMethods)
|
||||
_PUBLIC_ FN_GLOBAL_BOOL(lp_paranoid_server_security, &Globals.paranoid_server_security)
|
||||
static FN_GLOBAL_INTEGER(lp_announce_as, &Globals.announce_as)
|
||||
_PUBLIC_ FN_GLOBAL_LIST(lp_js_include, &Globals.jsInclude)
|
||||
_PUBLIC_ FN_GLOBAL_STRING(lp_jsonrpc_base, &Globals.jsonrpcBase)
|
||||
_PUBLIC_
|
||||
_PUBLIC_
|
||||
_PUBLIC_ FN_LOCAL_STRING(lp_servicename, szService)
|
||||
|
||||
@@ -66,6 +66,45 @@ static int ejs_typeof(MprVarHandle eid, int argc, struct MprVar **argv)
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
return the native type of a variable
|
||||
*/
|
||||
static int ejs_typeof_native(MprVarHandle eid, int argc, struct MprVar **argv)
|
||||
{
|
||||
const struct {
|
||||
MprType type;
|
||||
const char *name;
|
||||
} types[] = {
|
||||
{ MPR_TYPE_UNDEFINED, "undefined" },
|
||||
{ MPR_TYPE_NULL, "null" },
|
||||
{ MPR_TYPE_BOOL, "boolean" },
|
||||
{ MPR_TYPE_CFUNCTION, "c_function" },
|
||||
{ MPR_TYPE_FLOAT, "float" },
|
||||
{ MPR_TYPE_INT, "integer" },
|
||||
{ MPR_TYPE_INT64, "integer64" },
|
||||
{ MPR_TYPE_OBJECT, "object" },
|
||||
{ MPR_TYPE_FUNCTION, "function" },
|
||||
{ MPR_TYPE_STRING, "string" },
|
||||
{ MPR_TYPE_STRING_CFUNCTION, "string_c_function" },
|
||||
{ MPR_TYPE_PTR, "pointer" }
|
||||
};
|
||||
int i;
|
||||
const char *type = NULL;
|
||||
|
||||
if (argc != 1) return -1;
|
||||
|
||||
for (i=0;i<ARRAY_SIZE(types);i++) {
|
||||
if (argv[0]->type == types[i].type) {
|
||||
type = types[i].name;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (type == NULL) return -1;
|
||||
|
||||
mpr_ReturnString(eid, type);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
libinclude() allows you to include js files using a search path specified
|
||||
in "js include =" in smb.conf.
|
||||
@@ -112,6 +151,47 @@ static int ejs_libinclude(int eid, int argc, char **argv)
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
jsonrpc_include() allows you to include jsonrpc files from a path
|
||||
based at "jsonrpc base =" in smb.conf.
|
||||
*/
|
||||
static int ejs_jsonrpc_include(int eid, int argc, char **argv)
|
||||
{
|
||||
int ret = -1;
|
||||
char *path;
|
||||
char *emsg;
|
||||
const char *jsonrpc_base = lp_jsonrpc_base();
|
||||
struct MprVar result;
|
||||
|
||||
|
||||
if (jsonrpc_base == NULL || jsonrpc_base == NULL) {
|
||||
ejsSetErrorMsg(eid, "js include path not set");
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (argc != 1) {
|
||||
mpr_Return(eid, mprCreateIntegerVar(-1));
|
||||
return 0;
|
||||
}
|
||||
|
||||
path = talloc_asprintf(mprMemCtx(), "%s/%s", jsonrpc_base, argv[0]);
|
||||
if (path == NULL) {
|
||||
mpr_Return(eid, mprCreateIntegerVar(-1));
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (file_exist(path)) {
|
||||
ret = ejsEvalFile(eid, path, &result, &emsg);
|
||||
if (ret < 0) {
|
||||
printf("file found; ret=%d (%s)\n", ret, emsg);
|
||||
}
|
||||
}
|
||||
|
||||
mpr_Return(eid, mprCreateIntegerVar(ret));
|
||||
talloc_free(path);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
return the current version
|
||||
*/
|
||||
@@ -153,7 +233,9 @@ void smb_setup_ejs_functions(void (*exception_handler)(const char *))
|
||||
talloc_free(shared_init);
|
||||
|
||||
ejsDefineCFunction(-1, "typeof", ejs_typeof, NULL, MPR_VAR_SCRIPT_HANDLE);
|
||||
ejsDefineCFunction(-1, "nativeTypeOf", ejs_typeof_native, NULL, MPR_VAR_SCRIPT_HANDLE);
|
||||
ejsDefineStringCFunction(-1, "libinclude", ejs_libinclude, NULL, MPR_VAR_SCRIPT_HANDLE);
|
||||
ejsDefineStringCFunction(-1, "jsonrpc_include", ejs_jsonrpc_include, NULL, MPR_VAR_SCRIPT_HANDLE);
|
||||
ejsDefineCFunction(-1, "version", ejs_version, NULL, MPR_VAR_SCRIPT_HANDLE);
|
||||
}
|
||||
|
||||
|
||||
@@ -57,6 +57,22 @@ static int ejs_sys_hostname(MprVarHandle eid, int argc, struct MprVar **argv)
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
return current time as seconds and microseconds
|
||||
*/
|
||||
static int ejs_sys_gettimeofday(MprVarHandle eid, int argc, struct MprVar **argv)
|
||||
{
|
||||
struct timeval tv = timeval_current();
|
||||
struct MprVar v = mprObject("timeval");
|
||||
struct MprVar sec = mprCreateIntegerVar(tv.tv_sec);
|
||||
struct MprVar usec = mprCreateIntegerVar(tv.tv_usec);
|
||||
|
||||
mprCreateProperty(&v, "sec", &sec);
|
||||
mprCreateProperty(&v, "usec", &usec);
|
||||
mpr_Return(eid, v);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
return current time as a 64 bit nttime value
|
||||
*/
|
||||
@@ -85,6 +101,35 @@ static int ejs_sys_unix2nttime(MprVarHandle eid, int argc, struct MprVar **argv)
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
return the GMT time represented by the struct tm argument, as a time_t value
|
||||
*/
|
||||
static int ejs_sys_gmmktime(MprVarHandle eid, int argc, struct MprVar **argv)
|
||||
{
|
||||
struct MprVar *o;
|
||||
struct tm tm;
|
||||
if (argc != 1 || !mprVarIsObject(argv[0]->type)) {
|
||||
ejsSetErrorMsg(eid, "sys_gmmktime invalid arguments");
|
||||
return -1;
|
||||
}
|
||||
|
||||
o = argv[0];
|
||||
#define TM_EL(n) tm.n = mprVarToNumber(mprGetProperty(o, #n, NULL))
|
||||
TM_EL(tm_sec);
|
||||
TM_EL(tm_min);
|
||||
TM_EL(tm_hour);
|
||||
TM_EL(tm_mday);
|
||||
TM_EL(tm_mon);
|
||||
TM_EL(tm_year);
|
||||
TM_EL(tm_wday);
|
||||
TM_EL(tm_yday);
|
||||
TM_EL(tm_isdst);
|
||||
#undef TM_EL
|
||||
|
||||
mpr_Return(eid, mprCreateIntegerVar(mktime(&tm)));
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
return the given time as a gmtime structure
|
||||
*/
|
||||
@@ -97,6 +142,41 @@ static int ejs_sys_gmtime(MprVarHandle eid, int argc, struct MprVar **argv)
|
||||
ejsSetErrorMsg(eid, "sys_gmtime invalid arguments");
|
||||
return -1;
|
||||
}
|
||||
t = (time_t) mprVarToNumber(argv[0]);
|
||||
tm = gmtime(&t);
|
||||
if (tm == NULL) {
|
||||
mpr_Return(eid, mprCreateUndefinedVar());
|
||||
return 0;
|
||||
}
|
||||
ret = mprObject("gmtime");
|
||||
#define TM_EL(n) mprSetVar(&ret, #n, mprCreateIntegerVar(tm->n))
|
||||
TM_EL(tm_sec);
|
||||
TM_EL(tm_min);
|
||||
TM_EL(tm_hour);
|
||||
TM_EL(tm_mday);
|
||||
TM_EL(tm_mon);
|
||||
TM_EL(tm_year);
|
||||
TM_EL(tm_wday);
|
||||
TM_EL(tm_yday);
|
||||
TM_EL(tm_isdst);
|
||||
#undef TM_EL
|
||||
|
||||
mpr_Return(eid, ret);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
return the given NT time as a gmtime structure
|
||||
*/
|
||||
static int ejs_sys_ntgmtime(MprVarHandle eid, int argc, struct MprVar **argv)
|
||||
{
|
||||
time_t t;
|
||||
struct MprVar ret;
|
||||
struct tm *tm;
|
||||
if (argc != 1 || !mprVarIsNumber(argv[0]->type)) {
|
||||
ejsSetErrorMsg(eid, "sys_ntgmtime invalid arguments");
|
||||
return -1;
|
||||
}
|
||||
t = nt_time_to_unix(mprVarToNumber(argv[0]));
|
||||
tm = gmtime(&t);
|
||||
if (tm == NULL) {
|
||||
@@ -114,6 +194,7 @@ static int ejs_sys_gmtime(MprVarHandle eid, int argc, struct MprVar **argv)
|
||||
TM_EL(tm_wday);
|
||||
TM_EL(tm_yday);
|
||||
TM_EL(tm_isdst);
|
||||
#undef TM_EL
|
||||
|
||||
mpr_Return(eid, ret);
|
||||
return 0;
|
||||
@@ -332,8 +413,11 @@ static int ejs_sys_init(MprVarHandle eid, int argc, struct MprVar **argv)
|
||||
mprSetCFunction(obj, "interfaces", ejs_sys_interfaces);
|
||||
mprSetCFunction(obj, "hostname", ejs_sys_hostname);
|
||||
mprSetCFunction(obj, "nttime", ejs_sys_nttime);
|
||||
mprSetCFunction(obj, "getTimeOfDay", ejs_sys_gettimeofday);
|
||||
mprSetCFunction(obj, "unix2nttime", ejs_sys_unix2nttime);
|
||||
mprSetCFunction(obj, "gmmktime", ejs_sys_gmmktime);
|
||||
mprSetCFunction(obj, "gmtime", ejs_sys_gmtime);
|
||||
mprSetCFunction(obj, "ntgmtime", ejs_sys_ntgmtime);
|
||||
mprSetCFunction(obj, "ldaptime", ejs_sys_ldaptime);
|
||||
mprSetCFunction(obj, "httptime", ejs_sys_httptime);
|
||||
mprSetStringCFunction(obj, "unlink", ejs_sys_unlink);
|
||||
|
||||
@@ -113,7 +113,7 @@ function ldaptime()
|
||||
*/
|
||||
function datestring()
|
||||
{
|
||||
var t = sys.gmtime(sys.nttime());
|
||||
var t = sys.ntgmtime(sys.nttime());
|
||||
return sprintf("%04u%02u%02u%02u",
|
||||
t.tm_year+1900, t.tm_mon+1, t.tm_mday, t.tm_hour);
|
||||
}
|
||||
|
||||
@@ -35,6 +35,7 @@
|
||||
|
||||
#define SWAT_SESSION_KEY "SwatSessionId"
|
||||
#define HTTP_PREAUTH_URI "/scripting/preauth.esp"
|
||||
#define JSONRPC_REQUEST "/services"
|
||||
|
||||
/* state of the esp subsystem for a specific request */
|
||||
struct esp_state {
|
||||
@@ -414,6 +415,9 @@ static void http_setup_arrays(struct esp_state *esp)
|
||||
SETVAR(ESP_REQUEST_OBJ, "CONTENT_LENGTH",
|
||||
talloc_asprintf(esp, "%u", web->input.content_length));
|
||||
SETVAR(ESP_REQUEST_OBJ, "QUERY_STRING", web->input.query_string);
|
||||
#if 0 /* djl -- not yet. need to track down the compiler warning */
|
||||
SETVAR(ESP_REQUEST_OBJ, "POST_DATA", web->input.partial);
|
||||
#endif
|
||||
SETVAR(ESP_REQUEST_OBJ, "REQUEST_METHOD", web->input.post_request?"POST":"GET");
|
||||
SETVAR(ESP_REQUEST_OBJ, "REQUEST_URI", web->input.url);
|
||||
p = strrchr(web->input.url, '/');
|
||||
@@ -518,7 +522,6 @@ static void esp_request(struct esp_state *esp, const char *url)
|
||||
talloc_free(buf);
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
perform pre-authentication on every page is /scripting/preauth.esp
|
||||
exists. If this script generates any non-whitepace output at all,
|
||||
@@ -541,9 +544,9 @@ static BOOL http_preauth(struct esp_state *esp)
|
||||
esp_request(esp, HTTP_PREAUTH_URI);
|
||||
for (i=0;i<esp->web->output.content.length;i++) {
|
||||
if (!isspace(esp->web->output.content.data[i])) {
|
||||
/* if the preauth has generated content, then force it to be
|
||||
html, so that we can show the login page for failed
|
||||
access to images */
|
||||
/* if the preauth has generated content, then force it
|
||||
to be html, so that we can show the login page for
|
||||
failed access to images */
|
||||
http_setHeader(esp->web, "Content-Type: text/html", 0);
|
||||
return False;
|
||||
}
|
||||
@@ -763,11 +766,16 @@ void http_process_input(struct websrv_context *web)
|
||||
void *ejs_save = ejs_save_state();
|
||||
int i;
|
||||
const char *file_type = NULL;
|
||||
BOOL esp_enable = False;
|
||||
enum page_type {
|
||||
page_type_simple,
|
||||
page_type_esp,
|
||||
page_type_jsonrpc
|
||||
};
|
||||
enum page_type page_type;
|
||||
const struct {
|
||||
const char *extension;
|
||||
const char *mime_type;
|
||||
BOOL esp_enable;
|
||||
enum page_type page_type;
|
||||
} mime_types[] = {
|
||||
{"gif", "image/gif"},
|
||||
{"png", "image/png"},
|
||||
@@ -847,20 +855,29 @@ void http_process_input(struct websrv_context *web)
|
||||
esp->req = espCreateRequest(web, web->input.url, esp->variables);
|
||||
if (esp->req == NULL) goto internal_error;
|
||||
|
||||
/* work out the mime type */
|
||||
p = strrchr(web->input.url, '.');
|
||||
if (p == NULL) {
|
||||
esp_enable = True;
|
||||
}
|
||||
for (i=0;p && i<ARRAY_SIZE(mime_types);i++) {
|
||||
/*
|
||||
* Work out the mime type. First, we see if the request is a JSON-RPC
|
||||
* service request. If not, we look at the extension.
|
||||
*/
|
||||
if (strcmp(web->input.url, JSONRPC_REQUEST) == 0) {
|
||||
page_type = page_type_jsonrpc;
|
||||
file_type = "text/json";
|
||||
|
||||
} else {
|
||||
p = strrchr(web->input.url, '.');
|
||||
if (p == NULL) {
|
||||
page_type = page_type_esp;
|
||||
}
|
||||
for (i=0;p && i<ARRAY_SIZE(mime_types);i++) {
|
||||
if (strcmp(mime_types[i].extension, p+1) == 0) {
|
||||
file_type = mime_types[i].mime_type;
|
||||
esp_enable = mime_types[i].esp_enable;
|
||||
file_type = mime_types[i].mime_type;
|
||||
page_type = mime_types[i].page_type;
|
||||
}
|
||||
}
|
||||
if (file_type == NULL) {
|
||||
}
|
||||
if (file_type == NULL) {
|
||||
file_type = "text/html";
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/* setup basic headers */
|
||||
http_setResponseCode(web, 200);
|
||||
@@ -872,14 +889,32 @@ void http_process_input(struct websrv_context *web)
|
||||
|
||||
http_setup_arrays(esp);
|
||||
|
||||
/* possibly do pre-authentication */
|
||||
if (http_preauth(esp)) {
|
||||
if (esp_enable) {
|
||||
esp_request(esp, web->input.url);
|
||||
} else {
|
||||
http_simple_request(web);
|
||||
}
|
||||
}
|
||||
/*
|
||||
* Do pre-authentication. If pre-authentication succeeds, do
|
||||
* page-type-specific processing.
|
||||
*/
|
||||
switch(page_type)
|
||||
{
|
||||
case page_type_simple:
|
||||
if (http_preauth(esp)) {
|
||||
http_simple_request(web);
|
||||
}
|
||||
break;
|
||||
|
||||
case page_type_esp:
|
||||
if (http_preauth(esp)) {
|
||||
esp_request(esp, web->input.url);
|
||||
}
|
||||
break;
|
||||
|
||||
case page_type_jsonrpc:
|
||||
#if 0 /* djl -- not yet */
|
||||
if (! jsonrpc_request(esp)) {
|
||||
http_error(web, 500, "Out of memory");
|
||||
}
|
||||
#endif
|
||||
break;
|
||||
}
|
||||
|
||||
if (web->conn == NULL) {
|
||||
/* the connection has been terminated above us, probably
|
||||
|
||||
Reference in New Issue
Block a user