mirror of
https://github.com/Ylianst/MeshAgent
synced 2025-12-16 00:03:45 +00:00
2334 lines
125 KiB
C
2334 lines
125 KiB
C
/*
|
|
Copyright 2006 - 2018 Intel Corporation
|
|
|
|
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.
|
|
*/
|
|
|
|
#include "duktape.h"
|
|
#include "ILibDuktape_Helpers.h"
|
|
#include "ILibDuktapeModSearch.h"
|
|
#include "ILibDuktape_DuplexStream.h"
|
|
#include "ILibDuktape_EventEmitter.h"
|
|
#include "ILibDuktape_Debugger.h"
|
|
#include "../microstack/ILibParsers.h"
|
|
#include "../microstack/ILibCrypto.h"
|
|
#include "../microstack/ILibRemoteLogging.h"
|
|
|
|
|
|
#define ILibDuktape_Timer_Ptrs "\xFF_DuktapeTimer_PTRS"
|
|
#define ILibDuktape_Queue_Ptr "\xFF_Queue"
|
|
#define ILibDuktape_Stream_Buffer "\xFF_BUFFER"
|
|
#define ILibDuktape_Stream_ReadablePtr "\xFF_ReadablePtr"
|
|
#define ILibDuktape_Stream_WritablePtr "\xFF_WritablePtr"
|
|
#define ILibDuktape_Console_Destination "\xFF_Console_Destination"
|
|
#define ILibDuktape_Console_LOG_Destination "\xFF_Console_Destination"
|
|
#define ILibDuktape_Console_WARN_Destination "\xFF_Console_WARN_Destination"
|
|
#define ILibDuktape_Console_ERROR_Destination "\xFF_Console_ERROR_Destination"
|
|
#define ILibDuktape_Console_INFO_Level "\xFF_Console_INFO_Level"
|
|
#define ILibDuktape_Console_SessionID "\xFF_Console_SessionID"
|
|
|
|
#define ILibDuktape_DescriptorEvents_Table "\xFF_DescriptorEvents_Table"
|
|
#define ILibDuktape_DescriptorEvents_FD "\xFF_DescriptorEvents_FD"
|
|
#define ILibDuktape_DescriptorEvents_Options "\xFF_DescriptorEvents_Options"
|
|
|
|
typedef enum ILibDuktape_Console_DestinationFlags
|
|
{
|
|
ILibDuktape_Console_DestinationFlags_DISABLED = 0,
|
|
ILibDuktape_Console_DestinationFlags_StdOut = 1,
|
|
ILibDuktape_Console_DestinationFlags_ServerConsole = 2,
|
|
ILibDuktape_Console_DestinationFlags_WebLog = 4,
|
|
ILibDuktape_Console_DestinationFlags_LogFile = 8
|
|
}ILibDuktape_Console_DestinationFlags;
|
|
|
|
int g_displayStreamPipeMessages = 0;
|
|
int g_displayFinalizerMessages = 0;
|
|
|
|
duk_ret_t ILibDuktape_Pollyfills_Buffer_slice(duk_context *ctx)
|
|
{
|
|
int nargs = duk_get_top(ctx);
|
|
char *buffer;
|
|
char *out;
|
|
duk_size_t bufferLen;
|
|
int offset = 0;
|
|
duk_push_this(ctx);
|
|
|
|
buffer = Duktape_GetBuffer(ctx, -1, &bufferLen);
|
|
if (nargs >= 1)
|
|
{
|
|
offset = duk_require_int(ctx, 0);
|
|
bufferLen -= offset;
|
|
}
|
|
if (nargs == 2)
|
|
{
|
|
bufferLen = (duk_size_t)duk_require_int(ctx, 1) - offset;
|
|
}
|
|
duk_push_fixed_buffer(ctx, bufferLen);
|
|
out = Duktape_GetBuffer(ctx, -1, NULL);
|
|
memcpy_s(out, bufferLen, buffer + offset, bufferLen);
|
|
return 1;
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_Buffer_toString(duk_context *ctx)
|
|
{
|
|
int nargs = duk_get_top(ctx);
|
|
char *buffer, *tmpBuffer;
|
|
duk_size_t bufferLen = 0;
|
|
char *cType;
|
|
|
|
duk_push_this(ctx); // [buffer]
|
|
buffer = Duktape_GetBuffer(ctx, -1, &bufferLen);
|
|
|
|
if (nargs == 0)
|
|
{
|
|
if (bufferLen == 0 || buffer == NULL)
|
|
{
|
|
duk_push_null(ctx);
|
|
}
|
|
else
|
|
{
|
|
// Just convert to a string
|
|
duk_push_lstring(ctx, buffer, strnlen_s(buffer, bufferLen)); // [buffer][string]
|
|
}
|
|
}
|
|
else
|
|
{
|
|
cType = (char*)duk_require_string(ctx, 0);
|
|
if (strcmp(cType, "base64") == 0)
|
|
{
|
|
duk_push_fixed_buffer(ctx, ILibBase64EncodeLength((int)bufferLen));
|
|
tmpBuffer = Duktape_GetBuffer(ctx, -1, NULL);
|
|
ILibBase64Encode((unsigned char*)buffer, (int)bufferLen, (unsigned char**)&tmpBuffer);
|
|
duk_push_string(ctx, tmpBuffer);
|
|
}
|
|
else if (strcmp(cType, "hex") == 0)
|
|
{
|
|
duk_push_fixed_buffer(ctx, 1 + (bufferLen * 2));
|
|
tmpBuffer = Duktape_GetBuffer(ctx, -1, NULL);
|
|
util_tohex(buffer, (int)bufferLen, tmpBuffer);
|
|
duk_push_string(ctx, tmpBuffer);
|
|
}
|
|
else if (strcmp(cType, "hex:") == 0)
|
|
{
|
|
duk_push_fixed_buffer(ctx, 1 + (bufferLen * 3));
|
|
tmpBuffer = Duktape_GetBuffer(ctx, -1, NULL);
|
|
util_tohex2(buffer, (int)bufferLen, tmpBuffer);
|
|
duk_push_string(ctx, tmpBuffer);
|
|
}
|
|
else
|
|
{
|
|
return(ILibDuktape_Error(ctx, "Unrecognized parameter"));
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_Buffer_from(duk_context *ctx)
|
|
{
|
|
int nargs = duk_get_top(ctx);
|
|
char *str;
|
|
duk_size_t strlength;
|
|
char *encoding;
|
|
char *buffer;
|
|
int bufferLen;
|
|
|
|
if (nargs == 1)
|
|
{
|
|
str = (char*)duk_get_lstring(ctx, 0, &strlength);
|
|
buffer = duk_push_fixed_buffer(ctx, strlength);
|
|
memcpy_s(buffer, strlength, str, strlength);
|
|
duk_push_buffer_object(ctx, -1, 0, strlength, DUK_BUFOBJ_NODEJS_BUFFER);
|
|
return(1);
|
|
}
|
|
else if(!(nargs == 2 && duk_is_string(ctx, 0) && duk_is_string(ctx, 1)))
|
|
{
|
|
return(ILibDuktape_Error(ctx, "usage not supported yet"));
|
|
}
|
|
|
|
str = (char*)duk_get_lstring(ctx, 0, &strlength);
|
|
encoding = (char*)duk_require_string(ctx, 1);
|
|
|
|
if (strcmp(encoding, "base64") == 0)
|
|
{
|
|
// Base64
|
|
buffer = duk_push_fixed_buffer(ctx, ILibBase64DecodeLength((int)strlength));
|
|
bufferLen = ILibBase64Decode((unsigned char*)str, (int)strlength, (unsigned char**)&buffer);
|
|
duk_push_buffer_object(ctx, -1, 0, bufferLen, DUK_BUFOBJ_NODEJS_BUFFER);
|
|
}
|
|
else if (strcmp(encoding, "hex") == 0)
|
|
{
|
|
if (ILibString_StartsWith(str, (int)strlength, "0x", 2) != 0)
|
|
{
|
|
str += 2;
|
|
strlength -= 2;
|
|
}
|
|
buffer = duk_push_fixed_buffer(ctx, strlength / 2);
|
|
bufferLen = util_hexToBuf(str, (int)strlength, buffer);
|
|
duk_push_buffer_object(ctx, -1, 0, bufferLen, DUK_BUFOBJ_NODEJS_BUFFER);
|
|
}
|
|
else
|
|
{
|
|
return(ILibDuktape_Error(ctx, "unsupported encoding"));
|
|
}
|
|
return 1;
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_Buffer_readInt32BE(duk_context *ctx)
|
|
{
|
|
int offset = duk_require_int(ctx, 0);
|
|
char *buffer;
|
|
duk_size_t bufferLen;
|
|
|
|
duk_push_this(ctx);
|
|
buffer = Duktape_GetBuffer(ctx, -1, &bufferLen);
|
|
|
|
duk_push_int(ctx, ntohl(((int*)(buffer + offset))[0]));
|
|
return(1);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_Buffer_alloc(duk_context *ctx)
|
|
{
|
|
int sz = duk_require_int(ctx, 0);
|
|
int fill = 0;
|
|
|
|
if (duk_is_number(ctx, 1)) { fill = duk_require_int(ctx, 1); }
|
|
|
|
duk_push_fixed_buffer(ctx, sz);
|
|
char *buffer = Duktape_GetBuffer(ctx, -1, NULL);
|
|
memset(buffer, fill, sz);
|
|
duk_push_buffer_object(ctx, -1, 0, sz, DUK_BUFOBJ_NODEJS_BUFFER);
|
|
return(1);
|
|
}
|
|
|
|
void ILibDuktape_Polyfills_Buffer(duk_context *ctx)
|
|
{
|
|
char extras[] =
|
|
"Object.defineProperty(Buffer.prototype, \"swap32\",\
|
|
{\
|
|
value: function swap32()\
|
|
{\
|
|
var a = this.readUInt16BE(0);\
|
|
var b = this.readUInt16BE(2);\
|
|
this.writeUInt16LE(a, 2);\
|
|
this.writeUInt16LE(b, 0);\
|
|
return(this);\
|
|
}\
|
|
});";
|
|
|
|
duk_eval_string(ctx, extras); duk_pop(ctx);
|
|
|
|
// Polyfill Buffer.from()
|
|
duk_get_prop_string(ctx, -1, "Buffer"); // [g][Buffer]
|
|
duk_push_c_function(ctx, ILibDuktape_Polyfills_Buffer_from, DUK_VARARGS); // [g][Buffer][func]
|
|
duk_put_prop_string(ctx, -2, "from"); // [g][Buffer]
|
|
duk_pop(ctx); // [g]
|
|
|
|
// Polyfill Buffer.alloc() for Node Buffers)
|
|
duk_get_prop_string(ctx, -1, "Buffer"); // [g][Buffer]
|
|
duk_push_c_function(ctx, ILibDuktape_Polyfills_Buffer_alloc, DUK_VARARGS); // [g][Buffer][func]
|
|
duk_put_prop_string(ctx, -2, "alloc"); // [g][Buffer]
|
|
duk_pop(ctx); // [g]
|
|
|
|
|
|
// Polyfill Buffer.toString() for Node Buffers
|
|
duk_get_prop_string(ctx, -1, "Buffer"); // [g][Buffer]
|
|
duk_get_prop_string(ctx, -1, "prototype"); // [g][Buffer][prototype]
|
|
duk_push_c_function(ctx, ILibDuktape_Polyfills_Buffer_toString, DUK_VARARGS); // [g][Buffer][prototype][func]
|
|
duk_put_prop_string(ctx, -2, "toString"); // [g][Buffer][prototype]
|
|
duk_pop_2(ctx); // [g]
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_String_startsWith(duk_context *ctx)
|
|
{
|
|
duk_size_t tokenLen;
|
|
char *token = Duktape_GetBuffer(ctx, 0, &tokenLen);
|
|
char *buffer;
|
|
duk_size_t bufferLen;
|
|
|
|
duk_push_this(ctx);
|
|
buffer = Duktape_GetBuffer(ctx, -1, &bufferLen);
|
|
|
|
if (ILibString_StartsWith(buffer, (int)bufferLen, token, (int)tokenLen) != 0)
|
|
{
|
|
duk_push_true(ctx);
|
|
}
|
|
else
|
|
{
|
|
duk_push_false(ctx);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_String_endsWith(duk_context *ctx)
|
|
{
|
|
duk_size_t tokenLen;
|
|
char *token = Duktape_GetBuffer(ctx, 0, &tokenLen);
|
|
char *buffer;
|
|
duk_size_t bufferLen;
|
|
|
|
duk_push_this(ctx);
|
|
buffer = Duktape_GetBuffer(ctx, -1, &bufferLen);
|
|
|
|
if (ILibString_EndsWith(buffer, (int)bufferLen, token, (int)tokenLen) != 0)
|
|
{
|
|
duk_push_true(ctx);
|
|
}
|
|
else
|
|
{
|
|
duk_push_false(ctx);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_String_padStart(duk_context *ctx)
|
|
{
|
|
int totalLen = (int)duk_require_int(ctx, 0);
|
|
|
|
duk_size_t padcharLen;
|
|
duk_size_t bufferLen;
|
|
|
|
char *padchars;
|
|
if (duk_get_top(ctx) > 1)
|
|
{
|
|
padchars = (char*)duk_get_lstring(ctx, 1, &padcharLen);
|
|
}
|
|
else
|
|
{
|
|
padchars = " ";
|
|
padcharLen = 1;
|
|
}
|
|
|
|
duk_push_this(ctx);
|
|
char *buffer = Duktape_GetBuffer(ctx, -1, &bufferLen);
|
|
|
|
if ((int)bufferLen > totalLen)
|
|
{
|
|
duk_push_lstring(ctx, buffer, bufferLen);
|
|
return(1);
|
|
}
|
|
else
|
|
{
|
|
duk_size_t needs = totalLen - bufferLen;
|
|
|
|
duk_push_array(ctx); // [array]
|
|
while(needs > 0)
|
|
{
|
|
if (needs > padcharLen)
|
|
{
|
|
duk_push_string(ctx, padchars); // [array][pad]
|
|
duk_put_prop_index(ctx, -2, duk_get_length(ctx, -2)); // [array]
|
|
needs -= padcharLen;
|
|
}
|
|
else
|
|
{
|
|
duk_push_lstring(ctx, padchars, needs); // [array][pad]
|
|
duk_put_prop_index(ctx, -2, duk_get_length(ctx, -2)); // [array]
|
|
needs = 0;
|
|
}
|
|
}
|
|
duk_push_lstring(ctx, buffer, bufferLen); // [array][pad]
|
|
duk_put_prop_index(ctx, -2, duk_get_length(ctx, -2)); // [array]
|
|
duk_get_prop_string(ctx, -1, "join"); // [array][join]
|
|
duk_swap_top(ctx, -2); // [join][this]
|
|
duk_push_string(ctx, ""); // [join][this]['']
|
|
duk_call_method(ctx, 1); // [result]
|
|
return(1);
|
|
}
|
|
}
|
|
void ILibDuktape_Polyfills_String(duk_context *ctx)
|
|
{
|
|
// Polyfill 'String.startsWith'
|
|
duk_get_prop_string(ctx, -1, "String"); // [string]
|
|
duk_get_prop_string(ctx, -1, "prototype"); // [string][proto]
|
|
duk_push_c_function(ctx, ILibDuktape_Polyfills_String_startsWith, DUK_VARARGS); // [string][proto][func]
|
|
duk_put_prop_string(ctx, -2, "startsWith"); // [string][proto]
|
|
duk_push_c_function(ctx, ILibDuktape_Polyfills_String_endsWith, DUK_VARARGS); // [string][proto][func]
|
|
duk_put_prop_string(ctx, -2, "endsWith"); // [string][proto]
|
|
duk_push_c_function(ctx, ILibDuktape_Polyfills_String_padStart, DUK_VARARGS); // [string][proto][func]
|
|
duk_put_prop_string(ctx, -2, "padStart");
|
|
duk_pop_2(ctx);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_Console_log(duk_context *ctx)
|
|
{
|
|
int numargs = duk_get_top(ctx);
|
|
int i, x;
|
|
int len = 0;
|
|
duk_size_t strLen;
|
|
char *str;
|
|
char *PREFIX = NULL;
|
|
char *DESTINATION = NULL;
|
|
duk_push_current_function(ctx);
|
|
ILibDuktape_LogTypes logType = (ILibDuktape_LogTypes)Duktape_GetIntPropertyValue(ctx, -1, "logType", ILibDuktape_LogType_Normal);
|
|
switch (logType)
|
|
{
|
|
case ILibDuktape_LogType_Warn:
|
|
PREFIX = (char*)"WARNING: "; // LENGTH MUST BE <= 9
|
|
DESTINATION = ILibDuktape_Console_WARN_Destination;
|
|
break;
|
|
case ILibDuktape_LogType_Error:
|
|
PREFIX = (char*)"ERROR: "; // LENGTH MUST BE <= 9
|
|
DESTINATION = ILibDuktape_Console_ERROR_Destination;
|
|
break;
|
|
case ILibDuktape_LogType_Info1:
|
|
case ILibDuktape_LogType_Info2:
|
|
case ILibDuktape_LogType_Info3:
|
|
duk_push_this(ctx);
|
|
i = Duktape_GetIntPropertyValue(ctx, -1, ILibDuktape_Console_INFO_Level, 0);
|
|
duk_pop(ctx);
|
|
PREFIX = NULL;
|
|
if (i >= (((int)logType + 1) - (int)ILibDuktape_LogType_Info1))
|
|
{
|
|
DESTINATION = ILibDuktape_Console_LOG_Destination;
|
|
}
|
|
else
|
|
{
|
|
return(0);
|
|
}
|
|
break;
|
|
default:
|
|
PREFIX = NULL;
|
|
DESTINATION = ILibDuktape_Console_LOG_Destination;
|
|
break;
|
|
}
|
|
duk_pop(ctx);
|
|
|
|
// Calculate total length of string
|
|
for (i = 0; i < numargs; ++i)
|
|
{
|
|
if (duk_is_string(ctx, i))
|
|
{
|
|
len += (i == 0 ? 0 : 2);
|
|
duk_get_lstring(ctx, i, &strLen);
|
|
len += (int)strLen;
|
|
}
|
|
else
|
|
{
|
|
duk_dup(ctx, i);
|
|
if (strcmp("[object Object]", duk_to_string(ctx, -1)) == 0)
|
|
{
|
|
duk_pop(ctx);
|
|
duk_dup(ctx, i);
|
|
len += (i == 0 ? 1 : 3);
|
|
duk_enum(ctx, -1, DUK_ENUM_OWN_PROPERTIES_ONLY);
|
|
int propNum = 0;
|
|
while (duk_next(ctx, -1, 1))
|
|
{
|
|
len += 2;
|
|
len += (propNum++ == 0 ? 1 : 2);
|
|
duk_to_lstring(ctx, -2, &strLen); len += (int)strLen;
|
|
duk_to_lstring(ctx, -1, &strLen); len += (int)strLen;
|
|
duk_pop_2(ctx);
|
|
}
|
|
duk_pop(ctx);
|
|
len += 2;
|
|
}
|
|
else
|
|
{
|
|
len += (i == 0 ? 0 : 2);
|
|
duk_get_lstring(ctx, -1, &strLen); len += (int)strLen;
|
|
}
|
|
}
|
|
}
|
|
len += 2; // NULL Terminator and final carriage return
|
|
strLen = len;
|
|
|
|
str = ILibMemory_AllocateA(strLen + ((PREFIX != NULL) ? strnlen_s(PREFIX, 9) : 0));
|
|
x = (int)(ILibMemory_AllocateA_Size(str) - strLen);
|
|
if (x != 0)
|
|
{
|
|
strLen += sprintf_s(str, strLen, PREFIX);
|
|
}
|
|
for (i = 0; i < numargs; ++i)
|
|
{
|
|
if (duk_is_string(ctx, i))
|
|
{
|
|
x += sprintf_s(str + x, strLen - x, "%s%s", (i == 0 ? "" : ", "), duk_require_string(ctx, i));
|
|
}
|
|
else
|
|
{
|
|
duk_dup(ctx, i);
|
|
if (strcmp("[object Object]", duk_to_string(ctx, -1)) == 0)
|
|
{
|
|
duk_pop(ctx);
|
|
duk_dup(ctx, i);
|
|
x += sprintf_s(str+x, strLen - x, "%s", (i == 0 ? "{" : ", {"));
|
|
duk_enum(ctx, -1, DUK_ENUM_OWN_PROPERTIES_ONLY);
|
|
int propNum = 0;
|
|
while (duk_next(ctx, -1, 1))
|
|
{
|
|
x += sprintf_s(str + x, strLen - x, "%s%s: %s", ((propNum++ == 0) ? " " : ", "), (char*)duk_to_string(ctx, -2), (char*)duk_to_string(ctx, -1));
|
|
duk_pop_2(ctx);
|
|
}
|
|
duk_pop(ctx);
|
|
x += sprintf_s(str + x, strLen - x, " }");
|
|
}
|
|
else
|
|
{
|
|
x += sprintf_s(str + x, strLen - x, "%s%s", (i == 0 ? "" : ", "), duk_to_string(ctx, -1));
|
|
}
|
|
}
|
|
}
|
|
x += sprintf_s(str + x, strLen - x, "\n");
|
|
|
|
duk_push_this(ctx); // [console]
|
|
int dest = Duktape_GetIntPropertyValue(ctx, -1, DESTINATION, ILibDuktape_Console_DestinationFlags_StdOut);
|
|
|
|
if ((dest & ILibDuktape_Console_DestinationFlags_StdOut) == ILibDuktape_Console_DestinationFlags_StdOut)
|
|
{
|
|
#ifdef WIN32
|
|
DWORD writeLen;
|
|
WriteFile(GetStdHandle(STD_OUTPUT_HANDLE), (void*)str, x, &writeLen, NULL);
|
|
#else
|
|
ignore_result(write(STDOUT_FILENO, str, x));
|
|
#endif
|
|
}
|
|
if ((dest & ILibDuktape_Console_DestinationFlags_WebLog) == ILibDuktape_Console_DestinationFlags_WebLog)
|
|
{
|
|
ILibRemoteLogging_printf(ILibChainGetLogger(Duktape_GetChain(ctx)), ILibRemoteLogging_Modules_Microstack_Generic, ILibRemoteLogging_Flags_VerbosityLevel_1, "%s", str);
|
|
}
|
|
if ((dest & ILibDuktape_Console_DestinationFlags_ServerConsole) == ILibDuktape_Console_DestinationFlags_ServerConsole)
|
|
{
|
|
if (duk_peval_string(ctx, "require('MeshAgent');") == 0)
|
|
{
|
|
duk_get_prop_string(ctx, -1, "SendCommand"); // [console][agent][SendCommand]
|
|
duk_swap_top(ctx, -2); // [console][SendCommand][this]
|
|
duk_push_object(ctx); // [console][SendCommand][this][options]
|
|
duk_push_string(ctx, "msg"); duk_put_prop_string(ctx, -2, "action");
|
|
duk_push_string(ctx, "console"); duk_put_prop_string(ctx, -2, "type");
|
|
duk_push_string(ctx, str); duk_put_prop_string(ctx, -2, "value");
|
|
if (duk_has_prop_string(ctx, -4, ILibDuktape_Console_SessionID))
|
|
{
|
|
duk_get_prop_string(ctx, -4, ILibDuktape_Console_SessionID);
|
|
duk_put_prop_string(ctx, -2, "sessionid");
|
|
}
|
|
duk_call_method(ctx, 1);
|
|
}
|
|
}
|
|
if ((dest & ILibDuktape_Console_DestinationFlags_LogFile) == ILibDuktape_Console_DestinationFlags_LogFile)
|
|
{
|
|
duk_size_t pathLen;
|
|
char *path;
|
|
char *tmp = ILibMemory_AllocateA(x + 32);
|
|
int tmpx = ILibGetLocalTime(tmp + 1, (int)ILibMemory_AllocateA_Size(tmp) - 1) + 1;
|
|
tmp[0] = '[';
|
|
tmp[tmpx] = ']';
|
|
tmp[tmpx + 1] = ':';
|
|
tmp[tmpx + 2] = ' ';
|
|
memcpy_s(tmp + tmpx + 3, ILibMemory_AllocateA_Size(tmp) - tmpx - 3, str, x);
|
|
duk_eval_string(ctx, "require('fs');");
|
|
duk_get_prop_string(ctx, -1, "writeFileSync"); // [fs][writeFileSync]
|
|
duk_swap_top(ctx, -2); // [writeFileSync][this]
|
|
duk_push_heapptr(ctx, ILibDuktape_GetProcessObject(ctx)); // [writeFileSync][this][process]
|
|
duk_get_prop_string(ctx, -1, "execPath"); // [writeFileSync][this][process][execPath]
|
|
path = (char*)duk_get_lstring(ctx, -1, &pathLen);
|
|
if (path != NULL)
|
|
{
|
|
if (ILibString_EndsWithEx(path, (int)pathLen, ".exe", 4, 0))
|
|
{
|
|
duk_get_prop_string(ctx, -1, "substring"); // [writeFileSync][this][process][execPath][substring]
|
|
duk_swap_top(ctx, -2); // [writeFileSync][this][process][substring][this]
|
|
duk_push_int(ctx, 0); // [writeFileSync][this][process][substring][this][0]
|
|
duk_push_int(ctx, (int)(pathLen - 4)); // [writeFileSync][this][process][substring][this][0][len]
|
|
duk_call_method(ctx, 2); // [writeFileSync][this][process][path]
|
|
}
|
|
duk_get_prop_string(ctx, -1, "concat"); // [writeFileSync][this][process][path][concat]
|
|
duk_swap_top(ctx, -2); // [writeFileSync][this][process][concat][this]
|
|
duk_push_string(ctx, ".jlog"); // [writeFileSync][this][process][concat][this][.jlog]
|
|
duk_call_method(ctx, 1); // [writeFileSync][this][process][logPath]
|
|
duk_remove(ctx, -2); // [writeFileSync][this][logPath]
|
|
duk_push_string(ctx, tmp); // [writeFileSync][this][logPath][log]
|
|
duk_push_object(ctx); // [writeFileSync][this][logPath][log][options]
|
|
duk_push_string(ctx, "a"); duk_put_prop_string(ctx, -2, "flags");
|
|
duk_call_method(ctx, 3);
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_Console_enableWebLog(duk_context *ctx)
|
|
{
|
|
#ifdef _REMOTELOGGING
|
|
void *chain = Duktape_GetChain(ctx);
|
|
int port = duk_require_int(ctx, 0);
|
|
duk_size_t pLen;
|
|
if (duk_peval_string(ctx, "process.argv0") != 0) { return(ILibDuktape_Error(ctx, "console.enableWebLog(): Couldn't fetch argv0")); }
|
|
char *p = (char*)duk_get_lstring(ctx, -1, &pLen);
|
|
if (ILibString_EndsWith(p, (int)pLen, ".js", 3) != 0)
|
|
{
|
|
memcpy_s(ILibScratchPad2, sizeof(ILibScratchPad2), p, pLen - 3);
|
|
sprintf_s(ILibScratchPad2 + (pLen - 3), sizeof(ILibScratchPad2) - 3, ".wlg");
|
|
}
|
|
else if (ILibString_EndsWith(p, (int)pLen, ".exe", 3) != 0)
|
|
{
|
|
memcpy_s(ILibScratchPad2, sizeof(ILibScratchPad2), p, pLen - 4);
|
|
sprintf_s(ILibScratchPad2 + (pLen - 3), sizeof(ILibScratchPad2) - 4, ".wlg");
|
|
}
|
|
else
|
|
{
|
|
sprintf_s(ILibScratchPad2, sizeof(ILibScratchPad2), "%s.wlg", p);
|
|
}
|
|
ILibStartDefaultLoggerEx(chain, (unsigned short)port, ILibScratchPad2);
|
|
#endif
|
|
return (0);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_Console_displayStreamPipe_getter(duk_context *ctx)
|
|
{
|
|
duk_push_int(ctx, g_displayStreamPipeMessages);
|
|
return(1);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_Console_displayStreamPipe_setter(duk_context *ctx)
|
|
{
|
|
g_displayStreamPipeMessages = duk_require_int(ctx, 0);
|
|
return(0);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_Console_displayFinalizer_getter(duk_context *ctx)
|
|
{
|
|
duk_push_int(ctx, g_displayFinalizerMessages);
|
|
return(1);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_Console_displayFinalizer_setter(duk_context *ctx)
|
|
{
|
|
g_displayFinalizerMessages = duk_require_int(ctx, 0);
|
|
return(0);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_Console_logRefCount(duk_context *ctx)
|
|
{
|
|
printf("Reference Count => %s[%p]:%d\n", Duktape_GetStringPropertyValue(ctx, 0, ILibDuktape_OBJID, "UNKNOWN"), duk_require_heapptr(ctx, 0), ILibDuktape_GetReferenceCount(ctx, 0) - 1);
|
|
return(0);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_Console_setDestination(duk_context *ctx)
|
|
{
|
|
int nargs = duk_get_top(ctx);
|
|
int dest = duk_require_int(ctx, 0);
|
|
|
|
duk_push_this(ctx); // console
|
|
if ((dest & ILibDuktape_Console_DestinationFlags_ServerConsole) == ILibDuktape_Console_DestinationFlags_ServerConsole)
|
|
{
|
|
// Mesh Server Console
|
|
if (duk_peval_string(ctx, "require('MeshAgent');") != 0) { return(ILibDuktape_Error(ctx, "Unable to set destination to Mesh Console ")); }
|
|
duk_pop(ctx);
|
|
if (nargs > 1)
|
|
{
|
|
duk_dup(ctx, 1);
|
|
duk_put_prop_string(ctx, -2, ILibDuktape_Console_SessionID);
|
|
}
|
|
else
|
|
{
|
|
duk_del_prop_string(ctx, -1, ILibDuktape_Console_SessionID);
|
|
}
|
|
}
|
|
duk_dup(ctx, 0);
|
|
duk_put_prop_string(ctx, -2, ILibDuktape_Console_Destination);
|
|
return(0);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_Console_setInfoLevel(duk_context *ctx)
|
|
{
|
|
int val = duk_require_int(ctx, 0);
|
|
if (val < 0) { return(ILibDuktape_Error(ctx, "Invalid Info Level: %d", val)); }
|
|
|
|
duk_push_this(ctx);
|
|
duk_push_int(ctx, val);
|
|
duk_put_prop_string(ctx, -2, ILibDuktape_Console_INFO_Level);
|
|
|
|
return(0);
|
|
}
|
|
void ILibDuktape_Polyfills_Console(duk_context *ctx)
|
|
{
|
|
// Polyfill console.log()
|
|
#ifdef WIN32
|
|
SetConsoleOutputCP(CP_UTF8);
|
|
#endif
|
|
|
|
if (duk_has_prop_string(ctx, -1, "console"))
|
|
{
|
|
duk_get_prop_string(ctx, -1, "console"); // [g][console]
|
|
}
|
|
else
|
|
{
|
|
duk_push_object(ctx); // [g][console]
|
|
duk_dup(ctx, -1); // [g][console][console]
|
|
duk_put_prop_string(ctx, -3, "console"); // [g][console]
|
|
}
|
|
|
|
ILibDuktape_CreateInstanceMethodWithIntProperty(ctx, "logType", (int)ILibDuktape_LogType_Normal, "log", ILibDuktape_Polyfills_Console_log, DUK_VARARGS);
|
|
ILibDuktape_CreateInstanceMethodWithIntProperty(ctx, "logType", (int)ILibDuktape_LogType_Warn, "warn", ILibDuktape_Polyfills_Console_log, DUK_VARARGS);
|
|
ILibDuktape_CreateInstanceMethodWithIntProperty(ctx, "logType", (int)ILibDuktape_LogType_Error, "error", ILibDuktape_Polyfills_Console_log, DUK_VARARGS);
|
|
ILibDuktape_CreateInstanceMethodWithIntProperty(ctx, "logType", (int)ILibDuktape_LogType_Info1, "info1", ILibDuktape_Polyfills_Console_log, DUK_VARARGS);
|
|
ILibDuktape_CreateInstanceMethodWithIntProperty(ctx, "logType", (int)ILibDuktape_LogType_Info2, "info2", ILibDuktape_Polyfills_Console_log, DUK_VARARGS);
|
|
ILibDuktape_CreateInstanceMethodWithIntProperty(ctx, "logType", (int)ILibDuktape_LogType_Info3, "info3", ILibDuktape_Polyfills_Console_log, DUK_VARARGS);
|
|
|
|
|
|
ILibDuktape_CreateInstanceMethod(ctx, "enableWebLog", ILibDuktape_Polyfills_Console_enableWebLog, 1);
|
|
ILibDuktape_CreateEventWithGetterAndSetterEx(ctx, "displayStreamPipeMessages", ILibDuktape_Polyfills_Console_displayStreamPipe_getter, ILibDuktape_Polyfills_Console_displayStreamPipe_setter);
|
|
ILibDuktape_CreateEventWithGetterAndSetterEx(ctx, "displayFinalizerMessages", ILibDuktape_Polyfills_Console_displayFinalizer_getter, ILibDuktape_Polyfills_Console_displayFinalizer_setter);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "logReferenceCount", ILibDuktape_Polyfills_Console_logRefCount, 1);
|
|
|
|
ILibDuktape_CreateInstanceMethod(ctx, "setDestination", ILibDuktape_Polyfills_Console_setDestination, DUK_VARARGS);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "setInfoLevel", ILibDuktape_Polyfills_Console_setInfoLevel, 1);
|
|
|
|
duk_push_object(ctx);
|
|
duk_push_int(ctx, ILibDuktape_Console_DestinationFlags_DISABLED); duk_put_prop_string(ctx, -2, "DISABLED");
|
|
duk_push_int(ctx, ILibDuktape_Console_DestinationFlags_StdOut); duk_put_prop_string(ctx, -2, "STDOUT");
|
|
duk_push_int(ctx, ILibDuktape_Console_DestinationFlags_ServerConsole); duk_put_prop_string(ctx, -2, "SERVERCONSOLE");
|
|
duk_push_int(ctx, ILibDuktape_Console_DestinationFlags_WebLog); duk_put_prop_string(ctx, -2, "WEBLOG");
|
|
duk_push_int(ctx, ILibDuktape_Console_DestinationFlags_LogFile); duk_put_prop_string(ctx, -2, "LOGFILE");
|
|
ILibDuktape_CreateReadonlyProperty(ctx, "Destinations");
|
|
|
|
duk_push_int(ctx, ILibDuktape_Console_DestinationFlags_StdOut | ILibDuktape_Console_DestinationFlags_LogFile);
|
|
duk_put_prop_string(ctx, -2, ILibDuktape_Console_ERROR_Destination);
|
|
|
|
duk_push_int(ctx, ILibDuktape_Console_DestinationFlags_StdOut | ILibDuktape_Console_DestinationFlags_LogFile);
|
|
duk_put_prop_string(ctx, -2, ILibDuktape_Console_WARN_Destination);
|
|
|
|
duk_push_int(ctx, 0); duk_put_prop_string(ctx, -2, ILibDuktape_Console_INFO_Level);
|
|
|
|
duk_pop(ctx); // [g]
|
|
}
|
|
duk_ret_t ILibDuktape_ntohl(duk_context *ctx)
|
|
{
|
|
duk_size_t bufferLen;
|
|
char *buffer = Duktape_GetBuffer(ctx, 0, &bufferLen);
|
|
int offset = duk_require_int(ctx, 1);
|
|
|
|
if ((int)bufferLen < (4 + offset)) { return(ILibDuktape_Error(ctx, "buffer too small")); }
|
|
duk_push_int(ctx, ntohl(((unsigned int*)(buffer + offset))[0]));
|
|
return 1;
|
|
}
|
|
duk_ret_t ILibDuktape_ntohs(duk_context *ctx)
|
|
{
|
|
duk_size_t bufferLen;
|
|
char *buffer = Duktape_GetBuffer(ctx, 0, &bufferLen);
|
|
int offset = duk_require_int(ctx, 1);
|
|
|
|
if ((int)bufferLen < 2 + offset) { return(ILibDuktape_Error(ctx, "buffer too small")); }
|
|
duk_push_int(ctx, ntohs(((unsigned short*)(buffer + offset))[0]));
|
|
return 1;
|
|
}
|
|
duk_ret_t ILibDuktape_htonl(duk_context *ctx)
|
|
{
|
|
duk_size_t bufferLen;
|
|
char *buffer = Duktape_GetBuffer(ctx, 0, &bufferLen);
|
|
int offset = duk_require_int(ctx, 1);
|
|
unsigned int val = (unsigned int)duk_require_int(ctx, 2);
|
|
|
|
if ((int)bufferLen < (4 + offset)) { return(ILibDuktape_Error(ctx, "buffer too small")); }
|
|
((unsigned int*)(buffer + offset))[0] = htonl(val);
|
|
return 0;
|
|
}
|
|
duk_ret_t ILibDuktape_htons(duk_context *ctx)
|
|
{
|
|
duk_size_t bufferLen;
|
|
char *buffer = Duktape_GetBuffer(ctx, 0, &bufferLen);
|
|
int offset = duk_require_int(ctx, 1);
|
|
unsigned int val = (unsigned int)duk_require_int(ctx, 2);
|
|
|
|
if ((int)bufferLen < (2 + offset)) { return(ILibDuktape_Error(ctx, "buffer too small")); }
|
|
((unsigned short*)(buffer + offset))[0] = htons(val);
|
|
return 0;
|
|
}
|
|
void ILibDuktape_Polyfills_byte_ordering(duk_context *ctx)
|
|
{
|
|
ILibDuktape_CreateInstanceMethod(ctx, "ntohl", ILibDuktape_ntohl, 2);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "ntohs", ILibDuktape_ntohs, 2);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "htonl", ILibDuktape_htonl, 3);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "htons", ILibDuktape_htons, 3);
|
|
}
|
|
|
|
typedef enum ILibDuktape_Timer_Type
|
|
{
|
|
ILibDuktape_Timer_Type_TIMEOUT = 0,
|
|
ILibDuktape_Timer_Type_INTERVAL = 1,
|
|
ILibDuktape_Timer_Type_IMMEDIATE = 2
|
|
}ILibDuktape_Timer_Type;
|
|
typedef struct ILibDuktape_Timer
|
|
{
|
|
duk_context *ctx;
|
|
void *object;
|
|
void *callback;
|
|
void *args;
|
|
int timeout;
|
|
ILibDuktape_Timer_Type timerType;
|
|
}ILibDuktape_Timer;
|
|
|
|
duk_ret_t ILibDuktape_Polyfills_timer_finalizer(duk_context *ctx)
|
|
{
|
|
// Make sure we remove any timers just in case, so we don't leak resources
|
|
ILibDuktape_Timer *ptrs;
|
|
if (duk_has_prop_string(ctx, 0, ILibDuktape_Timer_Ptrs))
|
|
{
|
|
duk_get_prop_string(ctx, 0, ILibDuktape_Timer_Ptrs);
|
|
if (duk_has_prop_string(ctx, 0, "\xFF_callback"))
|
|
{
|
|
duk_del_prop_string(ctx, 0, "\xFF_callback");
|
|
}
|
|
if (duk_has_prop_string(ctx, 0, "\xFF_argArray"))
|
|
{
|
|
duk_del_prop_string(ctx, 0, "\xFF_argArray");
|
|
}
|
|
ptrs = (ILibDuktape_Timer*)Duktape_GetBuffer(ctx, -1, NULL);
|
|
|
|
ILibLifeTime_Remove(ILibGetBaseTimer(Duktape_GetChain(ctx)), ptrs);
|
|
}
|
|
return 0;
|
|
}
|
|
void ILibDuktape_Polyfills_timer_elapsed(void *obj)
|
|
{
|
|
ILibDuktape_Timer *ptrs = (ILibDuktape_Timer*)obj;
|
|
int argCount, i;
|
|
duk_context *ctx = ptrs->ctx;
|
|
char *funcName;
|
|
|
|
if (!ILibMemory_CanaryOK(ptrs)) { return; }
|
|
if (duk_check_stack(ctx, 3) == 0) { return; }
|
|
|
|
duk_push_heapptr(ctx, ptrs->callback); // [func]
|
|
funcName = Duktape_GetStringPropertyValue(ctx, -1, "name", "unknown_method");
|
|
duk_push_heapptr(ctx, ptrs->object); // [func][this]
|
|
duk_push_heapptr(ctx, ptrs->args); // [func][this][argArray]
|
|
|
|
if (ptrs->timerType == ILibDuktape_Timer_Type_INTERVAL)
|
|
{
|
|
ILibLifeTime_AddEx(ILibGetBaseTimer(Duktape_GetChain(ctx)), ptrs, ptrs->timeout, ILibDuktape_Polyfills_timer_elapsed, NULL);
|
|
}
|
|
else
|
|
{
|
|
if (ptrs->timerType == ILibDuktape_Timer_Type_IMMEDIATE)
|
|
{
|
|
duk_push_heap_stash(ctx);
|
|
duk_del_prop_string(ctx, -1, Duktape_GetStashKey(ptrs->object));
|
|
duk_pop(ctx);
|
|
}
|
|
|
|
duk_del_prop_string(ctx, -2, "\xFF_callback");
|
|
duk_del_prop_string(ctx, -2, "\xFF_argArray");
|
|
duk_del_prop_string(ctx, -2, ILibDuktape_Timer_Ptrs);
|
|
}
|
|
|
|
argCount = (int)duk_get_length(ctx, -1);
|
|
for (i = 0; i < argCount; ++i)
|
|
{
|
|
duk_get_prop_index(ctx, -1, i); // [func][this][argArray][arg]
|
|
duk_swap_top(ctx, -2); // [func][this][arg][argArray]
|
|
}
|
|
duk_pop(ctx); // [func][this][...arg...]
|
|
if (duk_pcall_method(ctx, argCount) != 0) { ILibDuktape_Process_UncaughtExceptionEx(ctx, "timers.onElapsed() callback handler on '%s()' ", funcName); }
|
|
duk_pop(ctx); // ...
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_timer_set(duk_context *ctx)
|
|
{
|
|
int nargs = duk_get_top(ctx);
|
|
ILibDuktape_Timer *ptrs;
|
|
ILibDuktape_Timer_Type timerType;
|
|
void *chain = Duktape_GetChain(ctx);
|
|
int argx;
|
|
|
|
duk_push_current_function(ctx);
|
|
duk_get_prop_string(ctx, -1, "type");
|
|
timerType = (ILibDuktape_Timer_Type)duk_get_int(ctx, -1);
|
|
|
|
duk_push_object(ctx); //[retVal]
|
|
switch (timerType)
|
|
{
|
|
case ILibDuktape_Timer_Type_IMMEDIATE:
|
|
ILibDuktape_WriteID(ctx, "Timers.immediate");
|
|
// We're only saving a reference for immediates
|
|
duk_push_heap_stash(ctx); //[retVal][stash]
|
|
duk_dup(ctx, -2); //[retVal][stash][immediate]
|
|
duk_put_prop_string(ctx, -2, Duktape_GetStashKey(duk_get_heapptr(ctx, -1))); //[retVal][stash]
|
|
duk_pop(ctx); //[retVal]
|
|
break;
|
|
case ILibDuktape_Timer_Type_INTERVAL:
|
|
ILibDuktape_WriteID(ctx, "Timers.interval");
|
|
break;
|
|
case ILibDuktape_Timer_Type_TIMEOUT:
|
|
ILibDuktape_WriteID(ctx, "Timers.timeout");
|
|
break;
|
|
}
|
|
ILibDuktape_CreateFinalizer(ctx, ILibDuktape_Polyfills_timer_finalizer);
|
|
|
|
ptrs = (ILibDuktape_Timer*)Duktape_PushBuffer(ctx, sizeof(ILibDuktape_Timer)); //[retVal][ptrs]
|
|
duk_put_prop_string(ctx, -2, ILibDuktape_Timer_Ptrs); //[retVal]
|
|
|
|
ptrs->ctx = ctx;
|
|
ptrs->object = duk_get_heapptr(ctx, -1);
|
|
ptrs->timerType = timerType;
|
|
ptrs->timeout = timerType == ILibDuktape_Timer_Type_IMMEDIATE ? 0 : (int)duk_require_int(ctx, 1);
|
|
ptrs->callback = duk_require_heapptr(ctx, 0);
|
|
|
|
duk_push_array(ctx); //[retVal][argArray]
|
|
for (argx = ILibDuktape_Timer_Type_IMMEDIATE == timerType ? 1 : 2; argx < nargs; ++argx)
|
|
{
|
|
duk_dup(ctx, argx); //[retVal][argArray][arg]
|
|
duk_put_prop_index(ctx, -2, argx - (ILibDuktape_Timer_Type_IMMEDIATE == timerType ? 1 : 2));//[retVal][argArray]
|
|
}
|
|
ptrs->args = duk_get_heapptr(ctx, -1); //[retVal]
|
|
duk_put_prop_string(ctx, -2, "\xFF_argArray");
|
|
|
|
duk_dup(ctx, 0); //[retVal][callback]
|
|
duk_put_prop_string(ctx, -2, "\xFF_callback"); //[retVal]
|
|
|
|
ILibLifeTime_AddEx(ILibGetBaseTimer(chain), ptrs, ptrs->timeout, ILibDuktape_Polyfills_timer_elapsed, NULL);
|
|
return 1;
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_timer_clear(duk_context *ctx)
|
|
{
|
|
ILibDuktape_Timer *ptrs;
|
|
ILibDuktape_Timer_Type timerType;
|
|
|
|
duk_push_current_function(ctx);
|
|
duk_get_prop_string(ctx, -1, "type");
|
|
timerType = (ILibDuktape_Timer_Type)duk_get_int(ctx, -1);
|
|
|
|
if(!duk_has_prop_string(ctx, 0, ILibDuktape_Timer_Ptrs))
|
|
{
|
|
switch (timerType)
|
|
{
|
|
case ILibDuktape_Timer_Type_TIMEOUT:
|
|
return(ILibDuktape_Error(ctx, "timers.clearTimeout(): Invalid Parameter"));
|
|
case ILibDuktape_Timer_Type_INTERVAL:
|
|
return(ILibDuktape_Error(ctx, "timers.clearInterval(): Invalid Parameter"));
|
|
case ILibDuktape_Timer_Type_IMMEDIATE:
|
|
return(ILibDuktape_Error(ctx, "timers.clearImmediate(): Invalid Parameter"));
|
|
}
|
|
}
|
|
|
|
duk_get_prop_string(ctx, 0, ILibDuktape_Timer_Ptrs);
|
|
ptrs = (ILibDuktape_Timer*)Duktape_GetBuffer(ctx, -1, NULL);
|
|
|
|
if (ptrs->timerType == ILibDuktape_Timer_Type_IMMEDIATE)
|
|
{
|
|
duk_push_heap_stash(ctx);
|
|
duk_del_prop_string(ctx, -1, Duktape_GetStashKey(ptrs->object));
|
|
duk_pop(ctx);
|
|
}
|
|
|
|
ILibLifeTime_Remove(ILibGetBaseTimer(Duktape_GetChain(ctx)), ptrs);
|
|
return 0;
|
|
}
|
|
void ILibDuktape_Polyfills_timer(duk_context *ctx)
|
|
{
|
|
ILibDuktape_CreateInstanceMethodWithIntProperty(ctx, "type", ILibDuktape_Timer_Type_TIMEOUT, "setTimeout", ILibDuktape_Polyfills_timer_set, DUK_VARARGS);
|
|
ILibDuktape_CreateInstanceMethodWithIntProperty(ctx, "type", ILibDuktape_Timer_Type_INTERVAL, "setInterval", ILibDuktape_Polyfills_timer_set, DUK_VARARGS);
|
|
ILibDuktape_CreateInstanceMethodWithIntProperty(ctx, "type", ILibDuktape_Timer_Type_IMMEDIATE, "setImmediate", ILibDuktape_Polyfills_timer_set, DUK_VARARGS);
|
|
|
|
ILibDuktape_CreateInstanceMethodWithIntProperty(ctx, "type", ILibDuktape_Timer_Type_TIMEOUT, "clearTimeout", ILibDuktape_Polyfills_timer_clear, 1);
|
|
ILibDuktape_CreateInstanceMethodWithIntProperty(ctx, "type", ILibDuktape_Timer_Type_INTERVAL, "clearInterval", ILibDuktape_Polyfills_timer_clear, 1);
|
|
ILibDuktape_CreateInstanceMethodWithIntProperty(ctx, "type", ILibDuktape_Timer_Type_IMMEDIATE, "clearImmediate", ILibDuktape_Polyfills_timer_clear, 1);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_getJSModule(duk_context *ctx)
|
|
{
|
|
if (ILibDuktape_ModSearch_GetJSModule(ctx, (char*)duk_require_string(ctx, 0)) == 0)
|
|
{
|
|
return(ILibDuktape_Error(ctx, "getJSModule(): (%s) not found", (char*)duk_require_string(ctx, 0)));
|
|
}
|
|
return(1);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_addModule(duk_context *ctx)
|
|
{
|
|
duk_size_t moduleLen;
|
|
char *module = (char*)Duktape_GetBuffer(ctx, 1, &moduleLen);
|
|
char *moduleName = (char*)duk_require_string(ctx, 0);
|
|
|
|
if (ILibDuktape_ModSearch_AddModule(ctx, moduleName, module, (int)moduleLen) != 0)
|
|
{
|
|
return(ILibDuktape_Error(ctx, "Cannot add module: %s", moduleName));
|
|
}
|
|
return(0);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_addModuleObject(duk_context *ctx)
|
|
{
|
|
void *module = duk_require_heapptr(ctx, 1);
|
|
char *moduleName = (char*)duk_require_string(ctx, 0);
|
|
|
|
ILibDuktape_ModSearch_AddModuleObject(ctx, moduleName, module);
|
|
return(0);
|
|
}
|
|
duk_ret_t ILibDuktape_Queue_Finalizer(duk_context *ctx)
|
|
{
|
|
duk_get_prop_string(ctx, 0, ILibDuktape_Queue_Ptr);
|
|
ILibQueue_Destroy((ILibQueue)duk_get_pointer(ctx, -1));
|
|
return(0);
|
|
}
|
|
duk_ret_t ILibDuktape_Queue_EnQueue(duk_context *ctx)
|
|
{
|
|
ILibQueue Q;
|
|
int i;
|
|
int nargs = duk_get_top(ctx);
|
|
duk_push_this(ctx); // [queue]
|
|
duk_get_prop_string(ctx, -1, ILibDuktape_Queue_Ptr); // [queue][ptr]
|
|
Q = (ILibQueue)duk_get_pointer(ctx, -1);
|
|
duk_pop(ctx); // [queue]
|
|
|
|
ILibDuktape_Push_ObjectStash(ctx); // [queue][stash]
|
|
duk_push_array(ctx); // [queue][stash][array]
|
|
for (i = 0; i < nargs; ++i)
|
|
{
|
|
duk_dup(ctx, i); // [queue][stash][array][arg]
|
|
duk_put_prop_index(ctx, -2, i); // [queue][stash][array]
|
|
}
|
|
ILibQueue_EnQueue(Q, duk_get_heapptr(ctx, -1));
|
|
duk_put_prop_string(ctx, -2, Duktape_GetStashKey(duk_get_heapptr(ctx, -1))); // [queue][stash]
|
|
return(0);
|
|
}
|
|
duk_ret_t ILibDuktape_Queue_DeQueue(duk_context *ctx)
|
|
{
|
|
duk_push_current_function(ctx);
|
|
duk_get_prop_string(ctx, -1, "peek");
|
|
int peek = duk_get_int(ctx, -1);
|
|
|
|
duk_push_this(ctx); // [Q]
|
|
duk_get_prop_string(ctx, -1, ILibDuktape_Queue_Ptr); // [Q][ptr]
|
|
ILibQueue Q = (ILibQueue)duk_get_pointer(ctx, -1);
|
|
void *h = peek == 0 ? ILibQueue_DeQueue(Q) : ILibQueue_PeekQueue(Q);
|
|
if (h == NULL) { return(ILibDuktape_Error(ctx, "Queue is empty")); }
|
|
duk_pop(ctx); // [Q]
|
|
ILibDuktape_Push_ObjectStash(ctx); // [Q][stash]
|
|
duk_push_heapptr(ctx, h); // [Q][stash][array]
|
|
int length = (int)duk_get_length(ctx, -1);
|
|
int i;
|
|
for (i = 0; i < length; ++i)
|
|
{
|
|
duk_get_prop_index(ctx, -i - 1, i); // [Q][stash][array][args]
|
|
}
|
|
if (peek == 0) { duk_del_prop_string(ctx, -length - 2, Duktape_GetStashKey(h)); }
|
|
return(length);
|
|
}
|
|
duk_ret_t ILibDuktape_Queue_isEmpty(duk_context *ctx)
|
|
{
|
|
duk_push_this(ctx);
|
|
duk_push_boolean(ctx, ILibQueue_IsEmpty((ILibQueue)Duktape_GetPointerProperty(ctx, -1, ILibDuktape_Queue_Ptr)));
|
|
return(1);
|
|
}
|
|
duk_ret_t ILibDuktape_Queue_new(duk_context *ctx)
|
|
{
|
|
duk_push_object(ctx); // [queue]
|
|
duk_push_pointer(ctx, ILibQueue_Create()); // [queue][ptr]
|
|
duk_put_prop_string(ctx, -2, ILibDuktape_Queue_Ptr); // [queue]
|
|
|
|
ILibDuktape_CreateFinalizer(ctx, ILibDuktape_Queue_Finalizer);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "enQueue", ILibDuktape_Queue_EnQueue, DUK_VARARGS);
|
|
ILibDuktape_CreateInstanceMethodWithIntProperty(ctx, "peek", 0, "deQueue", ILibDuktape_Queue_DeQueue, DUK_VARARGS);
|
|
ILibDuktape_CreateInstanceMethodWithIntProperty(ctx, "peek", 1, "peekQueue", ILibDuktape_Queue_DeQueue, DUK_VARARGS);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "isEmpty", ILibDuktape_Queue_isEmpty, 0);
|
|
|
|
return(1);
|
|
}
|
|
void ILibDuktape_Queue_Push(duk_context *ctx, void* chain)
|
|
{
|
|
duk_push_c_function(ctx, ILibDuktape_Queue_new, 0);
|
|
}
|
|
|
|
typedef struct ILibDuktape_DynamicBuffer_data
|
|
{
|
|
int start;
|
|
int end;
|
|
int unshiftBytes;
|
|
char *buffer;
|
|
int bufferLen;
|
|
}ILibDuktape_DynamicBuffer_data;
|
|
|
|
typedef struct ILibDuktape_DynamicBuffer_ContextSwitchData
|
|
{
|
|
void *chain;
|
|
void *heapptr;
|
|
ILibDuktape_DuplexStream *stream;
|
|
ILibDuktape_DynamicBuffer_data *data;
|
|
int bufferLen;
|
|
char buffer[];
|
|
}ILibDuktape_DynamicBuffer_ContextSwitchData;
|
|
|
|
ILibTransport_DoneState ILibDuktape_DynamicBuffer_WriteSink(ILibDuktape_DuplexStream *stream, char *buffer, int bufferLen, void *user);
|
|
void ILibDuktape_DynamicBuffer_WriteSink_ChainThread(void *chain, void *user)
|
|
{
|
|
ILibDuktape_DynamicBuffer_ContextSwitchData *data = (ILibDuktape_DynamicBuffer_ContextSwitchData*)user;
|
|
if(ILibMemory_CanaryOK(data->stream))
|
|
{
|
|
ILibDuktape_DynamicBuffer_WriteSink(data->stream, data->buffer, data->bufferLen, data->data);
|
|
ILibDuktape_DuplexStream_Ready(data->stream);
|
|
}
|
|
free(user);
|
|
}
|
|
ILibTransport_DoneState ILibDuktape_DynamicBuffer_WriteSink(ILibDuktape_DuplexStream *stream, char *buffer, int bufferLen, void *user)
|
|
{
|
|
ILibDuktape_DynamicBuffer_data *data = (ILibDuktape_DynamicBuffer_data*)user;
|
|
if (ILibIsRunningOnChainThread(stream->readableStream->chain) == 0)
|
|
{
|
|
ILibDuktape_DynamicBuffer_ContextSwitchData *tmp = (ILibDuktape_DynamicBuffer_ContextSwitchData*)ILibMemory_Allocate(sizeof(ILibDuktape_DynamicBuffer_ContextSwitchData) + bufferLen, 0, NULL, NULL);
|
|
tmp->chain = stream->readableStream->chain;
|
|
tmp->heapptr = stream->ParentObject;
|
|
tmp->stream = stream;
|
|
tmp->data = data;
|
|
tmp->bufferLen = bufferLen;
|
|
memcpy_s(tmp->buffer, bufferLen, buffer, bufferLen);
|
|
ILibChain_RunOnMicrostackThread(tmp->chain, ILibDuktape_DynamicBuffer_WriteSink_ChainThread, tmp);
|
|
return(ILibTransport_DoneState_INCOMPLETE);
|
|
}
|
|
|
|
|
|
if ((data->bufferLen - data->start - data->end) < bufferLen)
|
|
{
|
|
if (data->end > 0)
|
|
{
|
|
// Move the buffer first
|
|
memmove_s(data->buffer, data->bufferLen, data->buffer + data->start, data->end);
|
|
data->start = 0;
|
|
}
|
|
if ((data->bufferLen - data->end) < bufferLen)
|
|
{
|
|
// Need to resize buffer first
|
|
int tmpSize = data->bufferLen;
|
|
while ((tmpSize - data->end) < bufferLen)
|
|
{
|
|
tmpSize += 4096;
|
|
}
|
|
if ((data->buffer = (char*)realloc(data->buffer, tmpSize)) == NULL) { ILIBCRITICALEXIT(254); }
|
|
data->bufferLen = tmpSize;
|
|
}
|
|
}
|
|
|
|
|
|
memcpy_s(data->buffer + data->start + data->end, data->bufferLen - data->start - data->end, buffer, bufferLen);
|
|
data->end += bufferLen;
|
|
|
|
int unshifted = 0;
|
|
do
|
|
{
|
|
duk_push_heapptr(stream->readableStream->ctx, stream->ParentObject); // [ds]
|
|
duk_get_prop_string(stream->readableStream->ctx, -1, "emit"); // [ds][emit]
|
|
duk_swap_top(stream->readableStream->ctx, -2); // [emit][this]
|
|
duk_push_string(stream->readableStream->ctx, "readable"); // [emit][this][readable]
|
|
if (duk_pcall_method(stream->readableStream->ctx, 1) != 0) { ILibDuktape_Process_UncaughtExceptionEx(stream->readableStream->ctx, "DynamicBuffer.WriteSink => readable(): "); }
|
|
duk_pop(stream->readableStream->ctx); // ...
|
|
|
|
ILibDuktape_DuplexStream_WriteData(stream, data->buffer + data->start, data->end);
|
|
if (data->unshiftBytes == 0)
|
|
{
|
|
// All the data was consumed
|
|
data->start = data->end = 0;
|
|
}
|
|
else
|
|
{
|
|
unshifted = (data->end - data->unshiftBytes);
|
|
if (unshifted > 0)
|
|
{
|
|
data->start += unshifted;
|
|
data->end = data->unshiftBytes;
|
|
data->unshiftBytes = 0;
|
|
}
|
|
}
|
|
} while (unshifted != 0);
|
|
|
|
return(ILibTransport_DoneState_COMPLETE);
|
|
}
|
|
void ILibDuktape_DynamicBuffer_EndSink(ILibDuktape_DuplexStream *stream, void *user)
|
|
{
|
|
ILibDuktape_DuplexStream_WriteEnd(stream);
|
|
}
|
|
duk_ret_t ILibDuktape_DynamicBuffer_Finalizer(duk_context *ctx)
|
|
{
|
|
duk_get_prop_string(ctx, 0, "\xFF_buffer");
|
|
ILibDuktape_DynamicBuffer_data *data = (ILibDuktape_DynamicBuffer_data*)Duktape_GetBuffer(ctx, -1, NULL);
|
|
free(data->buffer);
|
|
return(0);
|
|
}
|
|
|
|
int ILibDuktape_DynamicBuffer_unshift(ILibDuktape_DuplexStream *sender, int unshiftBytes, void *user)
|
|
{
|
|
ILibDuktape_DynamicBuffer_data *data = (ILibDuktape_DynamicBuffer_data*)user;
|
|
data->unshiftBytes = unshiftBytes;
|
|
return(unshiftBytes);
|
|
}
|
|
duk_ret_t ILibDuktape_DynamicBuffer_read(duk_context *ctx)
|
|
{
|
|
ILibDuktape_DynamicBuffer_data *data;
|
|
duk_push_this(ctx); // [DynamicBuffer]
|
|
duk_get_prop_string(ctx, -1, "\xFF_buffer"); // [DynamicBuffer][buffer]
|
|
data = (ILibDuktape_DynamicBuffer_data*)Duktape_GetBuffer(ctx, -1, NULL);
|
|
duk_push_external_buffer(ctx); // [DynamicBuffer][buffer][extBuffer]
|
|
duk_config_buffer(ctx, -1, data->buffer + data->start, data->bufferLen - (data->start + data->end));
|
|
duk_push_buffer_object(ctx, -1, 0, data->bufferLen - (data->start + data->end), DUK_BUFOBJ_NODEJS_BUFFER);
|
|
return(1);
|
|
}
|
|
duk_ret_t ILibDuktape_DynamicBuffer_new(duk_context *ctx)
|
|
{
|
|
ILibDuktape_DynamicBuffer_data *data;
|
|
int initSize = 4096;
|
|
if (duk_get_top(ctx) != 0)
|
|
{
|
|
initSize = duk_require_int(ctx, 0);
|
|
}
|
|
|
|
duk_push_object(ctx); // [stream]
|
|
duk_push_fixed_buffer(ctx, sizeof(ILibDuktape_DynamicBuffer_data));
|
|
data = (ILibDuktape_DynamicBuffer_data*)Duktape_GetBuffer(ctx, -1, NULL);
|
|
memset(data, 0, sizeof(ILibDuktape_DynamicBuffer_data));
|
|
duk_put_prop_string(ctx, -2, "\xFF_buffer");
|
|
|
|
data->bufferLen = initSize;
|
|
data->buffer = (char*)malloc(initSize);
|
|
|
|
ILibDuktape_DuplexStream_InitEx(ctx, ILibDuktape_DynamicBuffer_WriteSink, ILibDuktape_DynamicBuffer_EndSink, NULL, NULL, ILibDuktape_DynamicBuffer_unshift, data);
|
|
ILibDuktape_EventEmitter_CreateEventEx(ILibDuktape_EventEmitter_GetEmitter(ctx, -1), "readable");
|
|
ILibDuktape_CreateInstanceMethod(ctx, "read", ILibDuktape_DynamicBuffer_read, DUK_VARARGS);
|
|
ILibDuktape_CreateFinalizer(ctx, ILibDuktape_DynamicBuffer_Finalizer);
|
|
|
|
return(1);
|
|
}
|
|
|
|
void ILibDuktape_DynamicBuffer_Push(duk_context *ctx, void *chain)
|
|
{
|
|
duk_push_c_function(ctx, ILibDuktape_DynamicBuffer_new, DUK_VARARGS);
|
|
}
|
|
|
|
duk_ret_t ILibDuktape_Polyfills_debugCrash(duk_context *ctx)
|
|
{
|
|
void *p = NULL;
|
|
((int*)p)[0] = 55;
|
|
return(0);
|
|
}
|
|
|
|
|
|
void ILibDuktape_Stream_PauseSink(struct ILibDuktape_readableStream *sender, void *user)
|
|
{
|
|
}
|
|
void ILibDuktape_Stream_ResumeSink(struct ILibDuktape_readableStream *sender, void *user)
|
|
{
|
|
int skip = 0;
|
|
duk_size_t bufferLen;
|
|
|
|
duk_push_heapptr(sender->ctx, sender->object); // [stream]
|
|
void *func = Duktape_GetHeapptrProperty(sender->ctx, -1, "_read");
|
|
duk_pop(sender->ctx); // ...
|
|
|
|
while (func != NULL && sender->paused == 0)
|
|
{
|
|
duk_push_heapptr(sender->ctx, sender->object); // [this]
|
|
if (!skip && duk_has_prop_string(sender->ctx, -1, ILibDuktape_Stream_Buffer))
|
|
{
|
|
duk_get_prop_string(sender->ctx, -1, ILibDuktape_Stream_Buffer); // [this][buffer]
|
|
if ((bufferLen = duk_get_length(sender->ctx, -1)) > 0)
|
|
{
|
|
// Buffer is not empty, so we need to 'PUSH' it
|
|
duk_get_prop_string(sender->ctx, -2, "push"); // [this][buffer][push]
|
|
duk_dup(sender->ctx, -3); // [this][buffer][push][this]
|
|
duk_dup(sender->ctx, -3); // [this][buffer][push][this][buffer]
|
|
duk_remove(sender->ctx, -4); // [this][push][this][buffer]
|
|
duk_call_method(sender->ctx, 1); // [this][boolean]
|
|
sender->paused = !duk_get_boolean(sender->ctx, -1);
|
|
duk_pop(sender->ctx); // [this]
|
|
|
|
if (duk_has_prop_string(sender->ctx, -1, ILibDuktape_Stream_Buffer))
|
|
{
|
|
duk_get_prop_string(sender->ctx, -1, ILibDuktape_Stream_Buffer); // [this][buffer]
|
|
if (duk_get_length(sender->ctx, -1) == bufferLen)
|
|
{
|
|
// All the data was unshifted
|
|
skip = !sender->paused;
|
|
}
|
|
duk_pop(sender->ctx); // [this]
|
|
}
|
|
duk_pop(sender->ctx); // ...
|
|
}
|
|
else
|
|
{
|
|
// Buffer is empty
|
|
duk_pop(sender->ctx); // [this]
|
|
duk_del_prop_string(sender->ctx, -1, ILibDuktape_Stream_Buffer);
|
|
duk_pop(sender->ctx); // ...
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// We need to 'read' more data
|
|
duk_push_heapptr(sender->ctx, func); // [this][read]
|
|
duk_swap_top(sender->ctx, -2); // [read][this]
|
|
if (duk_pcall_method(sender->ctx, 0) != 0) { ILibDuktape_Process_UncaughtException(sender->ctx); duk_pop(sender->ctx); break; }
|
|
// // [buffer]
|
|
duk_push_heapptr(sender->ctx, sender->object); // [buffer][this]
|
|
duk_swap_top(sender->ctx, -2); // [this][buffer]
|
|
if (duk_has_prop_string(sender->ctx, -2, ILibDuktape_Stream_Buffer))
|
|
{
|
|
duk_push_global_object(sender->ctx); // [this][buffer][g]
|
|
duk_get_prop_string(sender->ctx, -1, "Buffer"); // [this][buffer][g][Buffer]
|
|
duk_remove(sender->ctx, -2); // [this][buffer][Buffer]
|
|
duk_get_prop_string(sender->ctx, -1, "concat"); // [this][buffer][Buffer][concat]
|
|
duk_swap_top(sender->ctx, -2); // [this][buffer][concat][this]
|
|
duk_push_array(sender->ctx); // [this][buffer][concat][this][Array]
|
|
duk_get_prop_string(sender->ctx, -1, "push"); // [this][buffer][concat][this][Array][push]
|
|
duk_dup(sender->ctx, -2); // [this][buffer][concat][this][Array][push][this]
|
|
duk_get_prop_string(sender->ctx, -7, ILibDuktape_Stream_Buffer); // [this][buffer][concat][this][Array][push][this][buffer]
|
|
duk_call_method(sender->ctx, 1); duk_pop(sender->ctx); // [this][buffer][concat][this][Array]
|
|
duk_get_prop_string(sender->ctx, -1, "push"); // [this][buffer][concat][this][Array][push]
|
|
duk_dup(sender->ctx, -2); // [this][buffer][concat][this][Array][push][this]
|
|
duk_dup(sender->ctx, -6); // [this][buffer][concat][this][Array][push][this][buffer]
|
|
duk_remove(sender->ctx, -7); // [this][concat][this][Array][push][this][buffer]
|
|
duk_call_method(sender->ctx, 1); duk_pop(sender->ctx); // [this][concat][this][Array]
|
|
duk_call_method(sender->ctx, 1); // [this][buffer]
|
|
}
|
|
duk_put_prop_string(sender->ctx, -2, ILibDuktape_Stream_Buffer); // [this]
|
|
duk_pop(sender->ctx); // ...
|
|
skip = 0;
|
|
}
|
|
}
|
|
}
|
|
int ILibDuktape_Stream_UnshiftSink(struct ILibDuktape_readableStream *sender, int unshiftBytes, void *user)
|
|
{
|
|
duk_push_fixed_buffer(sender->ctx, unshiftBytes); // [buffer]
|
|
memcpy_s(Duktape_GetBuffer(sender->ctx, -1, NULL), unshiftBytes, sender->unshiftReserved, unshiftBytes);
|
|
duk_push_heapptr(sender->ctx, sender->object); // [buffer][stream]
|
|
duk_push_buffer_object(sender->ctx, -2, 0, unshiftBytes, DUK_BUFOBJ_NODEJS_BUFFER); // [buffer][stream][buffer]
|
|
duk_put_prop_string(sender->ctx, -2, ILibDuktape_Stream_Buffer); // [buffer][stream]
|
|
duk_pop_2(sender->ctx); // ...
|
|
|
|
return(unshiftBytes);
|
|
}
|
|
duk_ret_t ILibDuktape_Stream_Push(duk_context *ctx)
|
|
{
|
|
duk_push_this(ctx); // [stream]
|
|
ILibDuktape_readableStream *RS = (ILibDuktape_readableStream*)Duktape_GetPointerProperty(ctx, -1, ILibDuktape_Stream_ReadablePtr);
|
|
|
|
duk_size_t bufferLen;
|
|
char *buffer = (char*)Duktape_GetBuffer(ctx, 0, &bufferLen);
|
|
|
|
duk_push_boolean(ctx, !ILibDuktape_readableStream_WriteDataEx(RS, 0, buffer, (int)bufferLen)); // [stream][buffer][retVal]
|
|
return(1);
|
|
}
|
|
duk_ret_t ILibDuktape_Stream_EndSink(duk_context *ctx)
|
|
{
|
|
duk_push_this(ctx); // [stream]
|
|
ILibDuktape_readableStream *RS = (ILibDuktape_readableStream*)Duktape_GetPointerProperty(ctx, -1, ILibDuktape_Stream_ReadablePtr);
|
|
ILibDuktape_readableStream_WriteEnd(RS);
|
|
return(0);
|
|
}
|
|
duk_idx_t ILibDuktape_Stream_newReadable(duk_context *ctx)
|
|
{
|
|
ILibDuktape_readableStream *RS;
|
|
duk_push_object(ctx); // [Readable]
|
|
ILibDuktape_WriteID(ctx, "stream.readable");
|
|
RS = ILibDuktape_ReadableStream_InitEx(ctx, ILibDuktape_Stream_PauseSink, ILibDuktape_Stream_ResumeSink, ILibDuktape_Stream_UnshiftSink, NULL);
|
|
RS->paused = 1;
|
|
|
|
duk_push_pointer(ctx, RS);
|
|
duk_put_prop_string(ctx, -2, ILibDuktape_Stream_ReadablePtr);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "push", ILibDuktape_Stream_Push, DUK_VARARGS);
|
|
ILibDuktape_EventEmitter_AddOnceEx3(ctx, -1, "end", ILibDuktape_Stream_EndSink);
|
|
|
|
if (duk_is_object(ctx, 0))
|
|
{
|
|
void *h = Duktape_GetHeapptrProperty(ctx, 0, "read");
|
|
if (h != NULL) { duk_push_heapptr(ctx, h); duk_put_prop_string(ctx, -2, "_read"); }
|
|
}
|
|
return(1);
|
|
}
|
|
duk_ret_t ILibDuktape_Stream_Writable_WriteSink_Flush(duk_context *ctx)
|
|
{
|
|
duk_push_current_function(ctx);
|
|
ILibTransport_DoneState *retVal = (ILibTransport_DoneState*)Duktape_GetPointerProperty(ctx, -1, "retval");
|
|
if (retVal != NULL)
|
|
{
|
|
*retVal = ILibTransport_DoneState_COMPLETE;
|
|
}
|
|
else
|
|
{
|
|
ILibDuktape_WritableStream *WS = (ILibDuktape_WritableStream*)Duktape_GetPointerProperty(ctx, -1, ILibDuktape_Stream_WritablePtr);
|
|
ILibDuktape_WritableStream_Ready(WS);
|
|
}
|
|
return(0);
|
|
}
|
|
ILibTransport_DoneState ILibDuktape_Stream_Writable_WriteSink(struct ILibDuktape_WritableStream *stream, char *buffer, int bufferLen, void *user)
|
|
{
|
|
void *h;
|
|
ILibTransport_DoneState retVal = ILibTransport_DoneState_INCOMPLETE;
|
|
|
|
duk_push_this(stream->ctx); // [writable]
|
|
duk_get_prop_string(stream->ctx, -1, "_write"); // [writable][_write]
|
|
duk_swap_top(stream->ctx, -2); // [_write][this]
|
|
if (stream->Reserved == 0)
|
|
{
|
|
duk_push_external_buffer(stream->ctx); // [_write][this][extBuffer]
|
|
duk_insert(stream->ctx, -3); // [extBuffer][_write][this]
|
|
duk_config_buffer(stream->ctx, -3, buffer, (duk_size_t)bufferLen);
|
|
duk_push_buffer_object(stream->ctx, -3, 0, (duk_size_t)bufferLen, DUK_BUFOBJ_NODEJS_BUFFER);// [extBuffer][_write][this][buffer]
|
|
}
|
|
else
|
|
{
|
|
duk_push_lstring(stream->ctx, buffer, (duk_size_t)bufferLen); // [_write][this][string]
|
|
}
|
|
duk_push_c_function(stream->ctx, ILibDuktape_Stream_Writable_WriteSink_Flush, DUK_VARARGS); // [_write][this][string/buffer][callback]
|
|
h = duk_get_heapptr(stream->ctx, -1);
|
|
duk_push_heap_stash(stream->ctx); // [_write][this][string/buffer][callback][stash]
|
|
duk_dup(stream->ctx, -2); // [_write][this][string/buffer][callback][stash][callback]
|
|
duk_put_prop_string(stream->ctx, -2, Duktape_GetStashKey(h)); // [_write][this][string/buffer][callback][stash]
|
|
duk_pop(stream->ctx); // [_write][this][string/buffer][callback]
|
|
duk_push_pointer(stream->ctx, stream); duk_put_prop_string(stream->ctx, -2, ILibDuktape_Stream_WritablePtr);
|
|
|
|
duk_push_pointer(stream->ctx, &retVal); // [_write][this][string/buffer][callback][retval]
|
|
duk_put_prop_string(stream->ctx, -2, "retval"); // [_write][this][string/buffer][callback]
|
|
if (duk_pcall_method(stream->ctx, 2) != 0)
|
|
{
|
|
ILibDuktape_Process_UncaughtExceptionEx(stream->ctx, "stream.writable.write(): "); retVal = ILibTransport_DoneState_ERROR;
|
|
}
|
|
else
|
|
{
|
|
retVal = duk_to_boolean(stream->ctx, -1) ? ILibTransport_DoneState_COMPLETE : ILibTransport_DoneState_INCOMPLETE;
|
|
}
|
|
duk_pop(stream->ctx); // ...
|
|
|
|
duk_push_heapptr(stream->ctx, h); // [callback]
|
|
duk_del_prop_string(stream->ctx, -1, "retval");
|
|
duk_pop(stream->ctx); // ...
|
|
|
|
duk_push_heap_stash(stream->ctx);
|
|
duk_del_prop_string(stream->ctx, -1, Duktape_GetStashKey(h));
|
|
duk_pop(stream->ctx);
|
|
|
|
return(retVal);
|
|
}
|
|
duk_ret_t ILibDuktape_Stream_Writable_EndSink_finish(duk_context *ctx)
|
|
{
|
|
duk_push_current_function(ctx);
|
|
ILibDuktape_WritableStream *ws = (ILibDuktape_WritableStream*)Duktape_GetPointerProperty(ctx, -1, "ptr");
|
|
if (ILibMemory_CanaryOK(ws))
|
|
{
|
|
ILibDuktape_WritableStream_Finish(ws);
|
|
}
|
|
return(0);
|
|
}
|
|
void ILibDuktape_Stream_Writable_EndSink(struct ILibDuktape_WritableStream *stream, void *user)
|
|
{
|
|
duk_push_this(stream->ctx); // [writable]
|
|
duk_get_prop_string(stream->ctx, -1, "_final"); // [writable][_final]
|
|
duk_swap_top(stream->ctx, -2); // [_final][this]
|
|
duk_push_c_function(stream->ctx, ILibDuktape_Stream_Writable_EndSink_finish, 0); // [_final][this][callback]
|
|
duk_push_pointer(stream->ctx, stream); duk_put_prop_string(stream->ctx, -2, "ptr");
|
|
if (duk_pcall_method(stream->ctx, 1) != 0) { ILibDuktape_Process_UncaughtExceptionEx(stream->ctx, "stream.writable._final(): "); }
|
|
duk_pop(stream->ctx); // ...
|
|
}
|
|
duk_ret_t ILibDuktape_Stream_newWritable(duk_context *ctx)
|
|
{
|
|
ILibDuktape_WritableStream *WS;
|
|
duk_push_object(ctx); // [Writable]
|
|
ILibDuktape_WriteID(ctx, "stream.writable");
|
|
WS = ILibDuktape_WritableStream_Init(ctx, ILibDuktape_Stream_Writable_WriteSink, ILibDuktape_Stream_Writable_EndSink, NULL);
|
|
WS->JSCreated = 1;
|
|
|
|
duk_push_pointer(ctx, WS);
|
|
duk_put_prop_string(ctx, -2, ILibDuktape_Stream_WritablePtr);
|
|
|
|
if (duk_is_object(ctx, 0))
|
|
{
|
|
void *h = Duktape_GetHeapptrProperty(ctx, 0, "write");
|
|
if (h != NULL) { duk_push_heapptr(ctx, h); duk_put_prop_string(ctx, -2, "_write"); }
|
|
h = Duktape_GetHeapptrProperty(ctx, 0, "final");
|
|
if (h != NULL) { duk_push_heapptr(ctx, h); duk_put_prop_string(ctx, -2, "_final"); }
|
|
}
|
|
return(1);
|
|
}
|
|
void ILibDuktape_Stream_Duplex_PauseSink(ILibDuktape_DuplexStream *stream, void *user)
|
|
{
|
|
ILibDuktape_Stream_PauseSink(stream->readableStream, user);
|
|
}
|
|
void ILibDuktape_Stream_Duplex_ResumeSink(ILibDuktape_DuplexStream *stream, void *user)
|
|
{
|
|
ILibDuktape_Stream_ResumeSink(stream->readableStream, user);
|
|
}
|
|
int ILibDuktape_Stream_Duplex_UnshiftSink(ILibDuktape_DuplexStream *stream, int unshiftBytes, void *user)
|
|
{
|
|
return(ILibDuktape_Stream_UnshiftSink(stream->readableStream, unshiftBytes, user));
|
|
}
|
|
ILibTransport_DoneState ILibDuktape_Stream_Duplex_WriteSink(ILibDuktape_DuplexStream *stream, char *buffer, int bufferLen, void *user)
|
|
{
|
|
return(ILibDuktape_Stream_Writable_WriteSink(stream->writableStream, buffer, bufferLen, user));
|
|
}
|
|
void ILibDuktape_Stream_Duplex_EndSink(ILibDuktape_DuplexStream *stream, void *user)
|
|
{
|
|
ILibDuktape_Stream_Writable_EndSink(stream->writableStream, user);
|
|
}
|
|
duk_ret_t ILibDuktape_Stream_newDuplex(duk_context *ctx)
|
|
{
|
|
ILibDuktape_DuplexStream *DS;
|
|
duk_push_object(ctx); // [Duplex]
|
|
ILibDuktape_WriteID(ctx, "stream.Duplex");
|
|
DS = ILibDuktape_DuplexStream_InitEx(ctx, ILibDuktape_Stream_Duplex_WriteSink, ILibDuktape_Stream_Duplex_EndSink, ILibDuktape_Stream_Duplex_PauseSink, ILibDuktape_Stream_Duplex_ResumeSink, ILibDuktape_Stream_Duplex_UnshiftSink, NULL);
|
|
DS->writableStream->JSCreated = 1;
|
|
|
|
duk_push_pointer(ctx, DS->writableStream);
|
|
duk_put_prop_string(ctx, -2, ILibDuktape_Stream_WritablePtr);
|
|
|
|
duk_push_pointer(ctx, DS->readableStream);
|
|
duk_put_prop_string(ctx, -2, ILibDuktape_Stream_ReadablePtr);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "push", ILibDuktape_Stream_Push, DUK_VARARGS);
|
|
ILibDuktape_EventEmitter_AddOnceEx3(ctx, -1, "end", ILibDuktape_Stream_EndSink);
|
|
|
|
if (duk_is_object(ctx, 0))
|
|
{
|
|
void *h = Duktape_GetHeapptrProperty(ctx, 0, "write");
|
|
if (h != NULL) { duk_push_heapptr(ctx, h); duk_put_prop_string(ctx, -2, "_write"); }
|
|
h = Duktape_GetHeapptrProperty(ctx, 0, "final");
|
|
if (h != NULL) { duk_push_heapptr(ctx, h); duk_put_prop_string(ctx, -2, "_final"); }
|
|
h = Duktape_GetHeapptrProperty(ctx, 0, "read");
|
|
if (h != NULL) { duk_push_heapptr(ctx, h); duk_put_prop_string(ctx, -2, "_read"); }
|
|
}
|
|
return(1);
|
|
}
|
|
void ILibDuktape_Stream_Init(duk_context *ctx, void *chain)
|
|
{
|
|
duk_push_object(ctx); // [stream
|
|
ILibDuktape_WriteID(ctx, "stream");
|
|
ILibDuktape_CreateInstanceMethod(ctx, "Readable", ILibDuktape_Stream_newReadable, DUK_VARARGS);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "Writable", ILibDuktape_Stream_newWritable, DUK_VARARGS);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "Duplex", ILibDuktape_Stream_newDuplex, DUK_VARARGS);
|
|
}
|
|
void ILibDuktape_Polyfills_debugGC2(duk_context *ctx, void ** args, int argsLen)
|
|
{
|
|
if (g_displayFinalizerMessages) { printf("=> GC();\n"); }
|
|
duk_gc(ctx, 0);
|
|
duk_gc(ctx, 0);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_debugGC(duk_context *ctx)
|
|
{
|
|
ILibDuktape_Immediate(ctx, (void*[]) { NULL }, 0, ILibDuktape_Polyfills_debugGC2);
|
|
return(0);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_debug(duk_context *ctx)
|
|
{
|
|
#ifdef WIN32
|
|
if (IsDebuggerPresent()) { __debugbreak(); }
|
|
#elif defined(_POSIX)
|
|
raise(SIGTRAP);
|
|
#endif
|
|
return(0);
|
|
}
|
|
#ifndef MICROSTACK_NOTLS
|
|
duk_ret_t ILibDuktape_PKCS7_getSignedDataBlock(duk_context *ctx)
|
|
{
|
|
char *hash = ILibMemory_AllocateA(UTIL_SHA256_HASHSIZE);
|
|
char *pkeyHash = ILibMemory_AllocateA(UTIL_SHA256_HASHSIZE);
|
|
unsigned int size, r;
|
|
BIO *out = NULL;
|
|
PKCS7 *message = NULL;
|
|
char* data2 = NULL;
|
|
STACK_OF(X509) *st = NULL;
|
|
|
|
duk_size_t bufferLen;
|
|
char *buffer = Duktape_GetBuffer(ctx, 0, &bufferLen);
|
|
|
|
message = d2i_PKCS7(NULL, (const unsigned char**)&buffer, (long)bufferLen);
|
|
if (message == NULL) { return(ILibDuktape_Error(ctx, "PKCS7 Error")); }
|
|
|
|
// Lets rebuild the original message and check the size
|
|
size = i2d_PKCS7(message, NULL);
|
|
if (size < (unsigned int)bufferLen) { PKCS7_free(message); return(ILibDuktape_Error(ctx, "PKCS7 Error")); }
|
|
|
|
out = BIO_new(BIO_s_mem());
|
|
|
|
// Check the PKCS7 signature, but not the certificate chain.
|
|
r = PKCS7_verify(message, NULL, NULL, NULL, out, PKCS7_NOVERIFY);
|
|
if (r == 0) { PKCS7_free(message); BIO_free(out); return(ILibDuktape_Error(ctx, "PKCS7 Verify Error")); }
|
|
|
|
// If data block contains less than 32 bytes, fail.
|
|
size = (unsigned int)BIO_get_mem_data(out, &data2);
|
|
if (size <= ILibMemory_AllocateA_Size(hash)) { PKCS7_free(message); BIO_free(out); return(ILibDuktape_Error(ctx, "PKCS7 Size Mismatch Error")); }
|
|
|
|
|
|
duk_push_object(ctx); // [val]
|
|
duk_push_fixed_buffer(ctx, size); // [val][fbuffer]
|
|
duk_dup(ctx, -1); // [val][fbuffer][dup]
|
|
duk_put_prop_string(ctx, -3, "\xFF_fixedbuffer"); // [val][fbuffer]
|
|
duk_swap_top(ctx, -2); // [fbuffer][val]
|
|
duk_push_buffer_object(ctx, -2, 0, size, DUK_BUFOBJ_NODEJS_BUFFER); // [fbuffer][val][buffer]
|
|
ILibDuktape_CreateReadonlyProperty(ctx, "data"); // [fbuffer][val]
|
|
memcpy_s(Duktape_GetBuffer(ctx, -2, NULL), size, data2, size);
|
|
|
|
|
|
// Get the certificate signer
|
|
st = PKCS7_get0_signers(message, NULL, PKCS7_NOVERIFY);
|
|
|
|
// Get a full certificate hash of the signer
|
|
X509_digest(sk_X509_value(st, 0), EVP_sha256(), (unsigned char*)hash, NULL);
|
|
X509_pubkey_digest(sk_X509_value(st, 0), EVP_sha256(), (unsigned char*)pkeyHash, NULL);
|
|
|
|
sk_X509_free(st);
|
|
|
|
// Check certificate hash with first 32 bytes of data.
|
|
if (memcmp(hash, Duktape_GetBuffer(ctx, -2, NULL), ILibMemory_AllocateA_Size(hash)) != 0) { PKCS7_free(message); BIO_free(out); return(ILibDuktape_Error(ctx, "PKCS7 Certificate Hash Mismatch Error")); }
|
|
char *tmp = ILibMemory_AllocateA(1 + (ILibMemory_AllocateA_Size(hash) * 2));
|
|
util_tohex(hash, (int)ILibMemory_AllocateA_Size(hash), tmp);
|
|
duk_push_object(ctx); // [fbuffer][val][cert]
|
|
ILibDuktape_WriteID(ctx, "certificate");
|
|
duk_push_string(ctx, tmp); // [fbuffer][val][cert][fingerprint]
|
|
ILibDuktape_CreateReadonlyProperty(ctx, "fingerprint"); // [fbuffer][val][cert]
|
|
util_tohex(pkeyHash, (int)ILibMemory_AllocateA_Size(pkeyHash), tmp);
|
|
duk_push_string(ctx, tmp); // [fbuffer][val][cert][publickeyhash]
|
|
ILibDuktape_CreateReadonlyProperty(ctx, "publicKeyHash"); // [fbuffer][val][cert]
|
|
|
|
ILibDuktape_CreateReadonlyProperty(ctx, "signingCertificate"); // [fbuffer][val]
|
|
|
|
// Approved, cleanup and return.
|
|
BIO_free(out);
|
|
PKCS7_free(message);
|
|
|
|
return(1);
|
|
}
|
|
duk_ret_t ILibDuktape_PKCS7_signDataBlockFinalizer(duk_context *ctx)
|
|
{
|
|
char *buffer = Duktape_GetPointerProperty(ctx, 0, "\xFF_signature");
|
|
if (buffer != NULL) { free(buffer); }
|
|
return(0);
|
|
}
|
|
duk_ret_t ILibDuktape_PKCS7_signDataBlock(duk_context *ctx)
|
|
{
|
|
duk_get_prop_string(ctx, 1, "secureContext");
|
|
duk_get_prop_string(ctx, -1, "\xFF_SecureContext2CertBuffer");
|
|
struct util_cert *cert = (struct util_cert*)Duktape_GetBuffer(ctx, -1, NULL);
|
|
duk_size_t bufferLen;
|
|
char *buffer = (char*)Duktape_GetBuffer(ctx, 0, &bufferLen);
|
|
|
|
BIO *in = NULL;
|
|
PKCS7 *message = NULL;
|
|
char *signature = NULL;
|
|
int signatureLength = 0;
|
|
|
|
// Sign the block
|
|
in = BIO_new_mem_buf(buffer, (int)bufferLen);
|
|
message = PKCS7_sign(cert->x509, cert->pkey, NULL, in, PKCS7_BINARY);
|
|
if (message != NULL)
|
|
{
|
|
signatureLength = i2d_PKCS7(message, (unsigned char**)&signature);
|
|
PKCS7_free(message);
|
|
}
|
|
if (in != NULL) BIO_free(in);
|
|
if (signatureLength <= 0) { return(ILibDuktape_Error(ctx, "PKCS7_signDataBlockError: ")); }
|
|
|
|
duk_push_external_buffer(ctx);
|
|
duk_config_buffer(ctx, -1, signature, signatureLength);
|
|
duk_push_buffer_object(ctx, -1, 0, signatureLength, DUK_BUFOBJ_NODEJS_BUFFER);
|
|
duk_push_pointer(ctx, signature);
|
|
duk_put_prop_string(ctx, -2, "\xFF_signature");
|
|
ILibDuktape_CreateFinalizer(ctx, ILibDuktape_PKCS7_signDataBlockFinalizer);
|
|
|
|
return(1);
|
|
}
|
|
void ILibDuktape_PKCS7_Push(duk_context *ctx, void *chain)
|
|
{
|
|
duk_push_object(ctx);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "getSignedDataBlock", ILibDuktape_PKCS7_getSignedDataBlock, 1);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "signDataBlock", ILibDuktape_PKCS7_signDataBlock, DUK_VARARGS);
|
|
}
|
|
|
|
extern uint32_t crc32c(uint32_t crc, const unsigned char* buf, uint32_t len);
|
|
duk_ret_t ILibDuktape_Polyfills_crc32c(duk_context *ctx)
|
|
{
|
|
duk_size_t len;
|
|
char *buffer = Duktape_GetBuffer(ctx, 0, &len);
|
|
duk_push_int(ctx, crc32c(0, (unsigned char*)buffer, (uint32_t)len));
|
|
return(1);
|
|
}
|
|
#endif
|
|
duk_ret_t ILibDuktape_Polyfills_Object_hashCode(duk_context *ctx)
|
|
{
|
|
duk_push_this(ctx);
|
|
duk_push_string(ctx, Duktape_GetStashKey(duk_get_heapptr(ctx, -1)));
|
|
return(1);
|
|
}
|
|
duk_ret_t ILibDuktape_Polyfills_Array_peek(duk_context *ctx)
|
|
{
|
|
duk_push_this(ctx); // [Array]
|
|
duk_get_prop_index(ctx, -1, (duk_uarridx_t)duk_get_length(ctx, -1) - 1);
|
|
return(1);
|
|
}
|
|
void ILibDuktape_Polyfills_object(duk_context *ctx)
|
|
{
|
|
// Polyfill Object._hashCode()
|
|
duk_get_prop_string(ctx, -1, "Object"); // [g][Object]
|
|
duk_get_prop_string(ctx, -1, "prototype"); // [g][Object][prototype]
|
|
duk_push_c_function(ctx, ILibDuktape_Polyfills_Object_hashCode, 0); // [g][Object][prototype][func]
|
|
ILibDuktape_CreateReadonlyProperty(ctx, "_hashCode"); // [g][Object][prototype]
|
|
duk_pop_2(ctx); // [g]
|
|
|
|
duk_get_prop_string(ctx, -1, "Array"); // [g][Array]
|
|
duk_get_prop_string(ctx, -1, "prototype"); // [g][Array][prototype]
|
|
duk_push_c_function(ctx, ILibDuktape_Polyfills_Array_peek, 0); // [g][Array][prototype][peek]
|
|
ILibDuktape_CreateReadonlyProperty(ctx, "peek"); // [g][Array][prototype]
|
|
duk_pop_2(ctx); // [g]
|
|
}
|
|
|
|
|
|
#ifndef MICROSTACK_NOTLS
|
|
void ILibDuktape_bignum_addBigNumMethods(duk_context *ctx, BIGNUM *b);
|
|
duk_ret_t ILibDuktape_bignum_toString(duk_context *ctx)
|
|
{
|
|
duk_push_this(ctx);
|
|
BIGNUM *b = (BIGNUM*)Duktape_GetPointerProperty(ctx, -1, "\xFF_BIGNUM");
|
|
if (b != NULL)
|
|
{
|
|
char *numstr = BN_bn2dec(b);
|
|
duk_push_string(ctx, numstr);
|
|
OPENSSL_free(numstr);
|
|
return(1);
|
|
}
|
|
else
|
|
{
|
|
return(ILibDuktape_Error(ctx, "Invalid BIGNUM"));
|
|
}
|
|
}
|
|
duk_ret_t ILibDuktape_bignum_add(duk_context* ctx)
|
|
{
|
|
BIGNUM *ret = BN_new();
|
|
BIGNUM *r1, *r2;
|
|
|
|
duk_push_this(ctx);
|
|
r1 = (BIGNUM*)Duktape_GetPointerProperty(ctx, -1, "\xFF_BIGNUM");
|
|
r2 = (BIGNUM*)Duktape_GetPointerProperty(ctx, 0, "\xFF_BIGNUM");
|
|
|
|
BN_add(ret, r1, r2);
|
|
ILibDuktape_bignum_addBigNumMethods(ctx, ret);
|
|
return(1);
|
|
}
|
|
duk_ret_t ILibDuktape_bignum_sub(duk_context* ctx)
|
|
{
|
|
BIGNUM *ret = BN_new();
|
|
BIGNUM *r1, *r2;
|
|
|
|
duk_push_this(ctx);
|
|
r1 = (BIGNUM*)Duktape_GetPointerProperty(ctx, -1, "\xFF_BIGNUM");
|
|
r2 = (BIGNUM*)Duktape_GetPointerProperty(ctx, 0, "\xFF_BIGNUM");
|
|
|
|
BN_sub(ret, r1, r2);
|
|
ILibDuktape_bignum_addBigNumMethods(ctx, ret);
|
|
return(1);
|
|
}
|
|
duk_ret_t ILibDuktape_bignum_mul(duk_context* ctx)
|
|
{
|
|
BN_CTX *bx = BN_CTX_new();
|
|
BIGNUM *ret = BN_new();
|
|
BIGNUM *r1, *r2;
|
|
|
|
duk_push_this(ctx);
|
|
r1 = (BIGNUM*)Duktape_GetPointerProperty(ctx, -1, "\xFF_BIGNUM");
|
|
r2 = (BIGNUM*)Duktape_GetPointerProperty(ctx, 0, "\xFF_BIGNUM");
|
|
BN_mul(ret, r1, r2, bx);
|
|
BN_CTX_free(bx);
|
|
ILibDuktape_bignum_addBigNumMethods(ctx, ret);
|
|
return(1);
|
|
}
|
|
duk_ret_t ILibDuktape_bignum_div(duk_context* ctx)
|
|
{
|
|
BN_CTX *bx = BN_CTX_new();
|
|
BIGNUM *ret = BN_new();
|
|
BIGNUM *r1, *r2;
|
|
|
|
duk_push_this(ctx);
|
|
r1 = (BIGNUM*)Duktape_GetPointerProperty(ctx, -1, "\xFF_BIGNUM");
|
|
r2 = (BIGNUM*)Duktape_GetPointerProperty(ctx, 0, "\xFF_BIGNUM");
|
|
BN_div(ret, NULL, r1, r2, bx);
|
|
|
|
BN_CTX_free(bx);
|
|
ILibDuktape_bignum_addBigNumMethods(ctx, ret);
|
|
return(1);
|
|
}
|
|
duk_ret_t ILibDuktape_bignum_mod(duk_context* ctx)
|
|
{
|
|
BN_CTX *bx = BN_CTX_new();
|
|
BIGNUM *ret = BN_new();
|
|
BIGNUM *r1, *r2;
|
|
|
|
duk_push_this(ctx);
|
|
r1 = (BIGNUM*)Duktape_GetPointerProperty(ctx, -1, "\xFF_BIGNUM");
|
|
r2 = (BIGNUM*)Duktape_GetPointerProperty(ctx, 0, "\xFF_BIGNUM");
|
|
BN_div(NULL, ret, r1, r2, bx);
|
|
|
|
BN_CTX_free(bx);
|
|
ILibDuktape_bignum_addBigNumMethods(ctx, ret);
|
|
return(1);
|
|
}
|
|
|
|
duk_ret_t ILibDuktape_bignum_finalizer(duk_context *ctx)
|
|
{
|
|
BIGNUM *b = (BIGNUM*)Duktape_GetPointerProperty(ctx, 0, "\xFF_BIGNUM");
|
|
if (b != NULL)
|
|
{
|
|
BN_free(b);
|
|
}
|
|
return(0);
|
|
}
|
|
void ILibDuktape_bignum_addBigNumMethods(duk_context *ctx, BIGNUM *b)
|
|
{
|
|
duk_push_object(ctx);
|
|
duk_push_pointer(ctx, b); duk_put_prop_string(ctx, -2, "\xFF_BIGNUM");
|
|
ILibDuktape_CreateProperty_InstanceMethod(ctx, "toString", ILibDuktape_bignum_toString, DUK_VARARGS);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "add", ILibDuktape_bignum_add, 1);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "sub", ILibDuktape_bignum_sub, 1);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "mul", ILibDuktape_bignum_mul, 1);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "div", ILibDuktape_bignum_div, 1);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "mod", ILibDuktape_bignum_mod, 1);
|
|
duk_push_c_function(ctx, ILibDuktape_bignum_finalizer, 1); duk_set_finalizer(ctx, -2);
|
|
duk_eval_string(ctx, "(function toNumber(){return(parseInt(this.toString()));})"); duk_put_prop_string(ctx, -2, "toNumber");
|
|
}
|
|
duk_ret_t ILibDuktape_bignum_fromBuffer(duk_context *ctx)
|
|
{
|
|
char *endian = duk_get_top(ctx) > 1 ? Duktape_GetStringPropertyValue(ctx, 1, "endian", "big") : "big";
|
|
duk_size_t len;
|
|
char *buffer = Duktape_GetBuffer(ctx, 0, &len);
|
|
BIGNUM *b;
|
|
|
|
if (strcmp(endian, "big") == 0)
|
|
{
|
|
b = BN_bin2bn((unsigned char*)buffer, (int)len, NULL);
|
|
}
|
|
else if (strcmp(endian, "little") == 0)
|
|
{
|
|
b = BN_lebin2bn((unsigned char*)buffer, (int)len, NULL);
|
|
}
|
|
else
|
|
{
|
|
return(ILibDuktape_Error(ctx, "Invalid endian specified"));
|
|
}
|
|
|
|
ILibDuktape_bignum_addBigNumMethods(ctx, b);
|
|
return(1);
|
|
}
|
|
|
|
duk_ret_t ILibDuktape_bignum_func(duk_context *ctx)
|
|
{
|
|
BIGNUM *b = NULL;
|
|
BN_dec2bn(&b, duk_require_string(ctx, 0));
|
|
ILibDuktape_bignum_addBigNumMethods(ctx, b);
|
|
return(1);
|
|
}
|
|
void ILibDuktape_bignum_Push(duk_context *ctx, void *chain)
|
|
{
|
|
duk_push_c_function(ctx, ILibDuktape_bignum_func, DUK_VARARGS);
|
|
duk_push_c_function(ctx, ILibDuktape_bignum_fromBuffer, DUK_VARARGS); duk_put_prop_string(ctx, -2, "fromBuffer");
|
|
}
|
|
void ILibDuktape_dataGenerator_onPause(struct ILibDuktape_readableStream *sender, void *user)
|
|
{
|
|
|
|
}
|
|
void ILibDuktape_dataGenerator_onResume(struct ILibDuktape_readableStream *sender, void *user)
|
|
{
|
|
SHA256_CTX shctx;
|
|
|
|
char *buffer = (char*)user;
|
|
size_t bufferLen = ILibMemory_Size(buffer);
|
|
int val;
|
|
|
|
while (sender->paused == 0)
|
|
{
|
|
duk_push_heapptr(sender->ctx, sender->object);
|
|
val = Duktape_GetIntPropertyValue(sender->ctx, -1, "\xFF_counter", 0);
|
|
duk_push_int(sender->ctx, (val + 1) < 255 ? (val+1) : 0); duk_put_prop_string(sender->ctx, -2, "\xFF_counter");
|
|
duk_pop(sender->ctx);
|
|
|
|
//util_random((int)(bufferLen - UTIL_SHA256_HASHSIZE), buffer + UTIL_SHA256_HASHSIZE);
|
|
memset(buffer + UTIL_SHA256_HASHSIZE, val, bufferLen - UTIL_SHA256_HASHSIZE);
|
|
|
|
|
|
SHA256_Init(&shctx);
|
|
SHA256_Update(&shctx, buffer + UTIL_SHA256_HASHSIZE, bufferLen - UTIL_SHA256_HASHSIZE);
|
|
SHA256_Final((unsigned char*)buffer, &shctx);
|
|
ILibDuktape_readableStream_WriteData(sender, buffer, (int)bufferLen);
|
|
}
|
|
}
|
|
duk_ret_t ILibDuktape_dataGenerator_const(duk_context *ctx)
|
|
{
|
|
int bufSize = (int)duk_require_int(ctx, 0);
|
|
void *buffer;
|
|
|
|
if (bufSize <= UTIL_SHA256_HASHSIZE)
|
|
{
|
|
return(ILibDuktape_Error(ctx, "Value too small. Must be > %d", UTIL_SHA256_HASHSIZE));
|
|
}
|
|
|
|
duk_push_object(ctx);
|
|
duk_push_int(ctx, 0); duk_put_prop_string(ctx, -2, "\xFF_counter");
|
|
buffer = Duktape_PushBuffer(ctx, bufSize);
|
|
duk_put_prop_string(ctx, -2, "\xFF_buffer");
|
|
ILibDuktape_ReadableStream_Init(ctx, ILibDuktape_dataGenerator_onPause, ILibDuktape_dataGenerator_onResume, buffer)->paused = 1;
|
|
return(1);
|
|
}
|
|
void ILibDuktape_dataGenerator_Push(duk_context *ctx, void *chain)
|
|
{
|
|
duk_push_c_function(ctx, ILibDuktape_dataGenerator_const, DUK_VARARGS);
|
|
}
|
|
#endif
|
|
|
|
void ILibDuktape_Polyfills_JS_Init(duk_context *ctx)
|
|
{
|
|
// The following can be overriden by calling addModule() or by having a .js file in the module path
|
|
|
|
// http-digest. Refer to /modules folder for a human readable version
|
|
duk_peval_string_noresult(ctx, "addModule('http-digest', Buffer.from('LyoKQ29weXJpZ2h0IDIwMTkgSW50ZWwgQ29ycG9yYXRpb24KCkxpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwp5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCllvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAoKICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAoKVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQpkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLApXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZApsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KKi8NCg0KDQp2YXIgd3JpdGFibGUgPSByZXF1aXJlKCdzdHJlYW0nKS5Xcml0YWJsZTsNCnZhciBtZDUgPSByZXF1aXJlKCdNRDVTdHJlYW0nKS5jcmVhdGUoKTsNCg0KZnVuY3Rpb24gY2hlY2tFdmVudEZvcndhcmRpbmcoZGlnZXN0UmVxdWVzdCwgZXZlbnROYW1lKQ0Kew0KICAgIGlmIChkaWdlc3RSZXF1ZXN0Lmxpc3RlbmVyQ291bnQoZXZlbnROYW1lKSA+IDApDQogICAgew0KICAgICAgICB2YXIgZUZvcndhcmQgPSBmdW5jdGlvbiBfZUZvcndhcmQoKQ0KICAgICAgICB7DQogICAgICAgICAgICB2YXIgcCA9IFtlRm9yd2FyZC5fZXZlbnROYW1lXTsNCiAgICAgICAgICAgIGZvciAodmFyIGkgPSAwOyBpIDwgYXJndW1lbnRzLmxlbmd0aDsgKytpKSB7IHAucHVzaChhcmd1bWVudHNbaV0pOyB9DQogICAgICAgICAgICBfZUZvcndhcmQuX2RpZ2VzdFJlcXVlc3QuZW1pdC5hcHBseShfZUZvcndhcmQuX2RpZ2VzdFJlcXVlc3QsIHApOw0KICAgICAgICB9Ow0KICAgICAgICBlRm9yd2FyZC5fZXZlbnROYW1lID0gZXZlbnROYW1lOw0KICAgICAgICBlRm9yd2FyZC5fZGlnZXN0UmVxdWVzdCA9IGRpZ2VzdFJlcXVlc3Q7DQogICAgICAgIGRpZ2VzdFJlcXVlc3QuX3JlcXVlc3Qub24oZXZlbnROYW1lLCBlRm9yd2FyZCk7DQogICAgfQ0KfQ0KDQpmdW5jdGlvbiBnZW5lcmF0ZUF1dGhIZWFkZXJzKGltc2csIG9wdGlvbnMsIGRpZ2VzdCkNCnsNCiAgICB2YXIgYXV0aDsNCg0KICAgIGlmIChpbXNnICE9IG51bGwpDQogICAgew0KICAgICAgICBhdXRoID0geyByZWFsbTogbnVsbCwgbm9uY2U6IG51bGwsIG9wYXF1ZTogbnVsbCwgcW9wOiBudWxsIH07DQogICAgICAgIHZhciB3d3cgPSBpbXNnLmhlYWRlcnNbJ1dXVy1BdXRoZW50aWNhdGUnXTsNCiAgICAgICAgdmFyIHRva2VucyA9IHd3dy5zcGxpdCgnLCcpOw0KDQogICAgICAgIHZhciBwYWlyczsNCiAgICAgICAgZm9yICh2YXIgaSBpbiB0b2tlbnMpDQogICAgICAgIHsNCiAgICAgICAgICAgIHBhaXJzID0gdG9rZW5zW2ldLnNwbGl0KCc9Jyk7DQogICAgICAgICAgICBpZiAocGFpcnMubGVuZ3RoID09IDIpDQogICAgICAgICAgICB7DQogICAgICAgICAgICAgICAgc3dpdGNoIChwYWlyc1swXS50b0xvd2VyQ2FzZSgpLnRyaW0oKSkNCiAgICAgICAgICAgICAgICB7DQogICAgICAgICAgICAgICAgICAgIGNhc2UgJ2RpZ2VzdCByZWFsbSc6DQogICAgICAgICAgICAgICAgICAgICAgICBhdXRoLnJlYWxtID0gcGFpcnNbMV07DQogICAgICAgICAgICAgICAgICAgICAgICBpZiAoYXV0aC5yZWFsbVswXSA9PSAnIicpIHsgYXV0aC5yZWFsbSA9IGF1dGgucmVhbG0uc3Vic3RyaW5nKDEsIGF1dGgucmVhbG0ubGVuZ3RoIC0gMSk7IH0NCiAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOw0KICAgICAgICAgICAgICAgICAgICBjYXNlICdub25jZSc6DQogICAgICAgICAgICAgICAgICAgICAgICBhdXRoLm5vbmNlID0gcGFpcnNbMV07DQogICAgICAgICAgICAgICAgICAgICAgICBpZiAoYXV0aC5ub25jZVswXSA9PSAnIicpIHsgYXV0aC5ub25jZSA9IGF1dGgubm9uY2Uuc3Vic3RyaW5nKDEsIGF1dGgubm9uY2UubGVuZ3RoIC0gMSk7IH0NCiAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOw0KICAgICAgICAgICAgICAgICAgICBjYXNlICdvcGFxdWUnOg0KICAgICAgICAgICAgICAgICAgICAgICAgYXV0aC5vcGFxdWUgPSBwYWlyc1sxXTsNCiAgICAgICAgICAgICAgICAgICAgICAgIGlmIChhdXRoLm9wYXF1ZVswXSA9PSAnIicpIHsgYXV0aC5vcGFxdWUgPSBhdXRoLm9wYXF1ZS5zdWJzdHJpbmcoMSwgYXV0aC5vcGFxdWUubGVuZ3RoIC0gMSk7IH0NCiAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOw0KICAgICAgICAgICAgICAgICAgICBjYXNlICdxb3AnOg0KICAgICAgICAgICAgICAgICAgICAgICAgYXV0aC5xb3AgPSBwYWlyc1sxXTsNCiAgICAgICAgICAgICAgICAgICAgICAgIGlmIChhdXRoLnFvcFswXSA9PSAnIicpIHsgYXV0aC5xb3AgPSBhdXRoLnFvcC5zdWJzdHJpbmcoMSwgYXV0aC5xb3AubGVuZ3RoIC0gMSk7IH0NCiAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOw0KICAgICAgICAgICAgICAgIH0NCiAgICAgICAgICAgIH0NCiAgICAgICAgfQ0KICAgICAgICBkaWdlc3QuX2F1dGggPSBhdXRoOw0KICAgIH0NCiAgICBlbHNlDQogICAgew0KICAgICAgICBpZiAoIShhdXRoID0gZGlnZXN0Ll9hdXRoKSkgeyByZXR1cm47IH0NCiAgICB9DQoNCiAgICB2YXIgc3RlcDEgPSBkaWdlc3QuX29wdGlvbnMudXNlcm5hbWUgKyAnOicgKyBhdXRoLnJlYWxtICsgJzonICsgZGlnZXN0Ll9vcHRpb25zLnBhc3N3b3JkOw0KICAgIGF1dGguc3RlcDEgPSBtZDUuc3luY0hhc2goc3RlcDEpLnRvU3RyaW5nKCdoZXgnKS50b0xvd2VyQ2FzZSgpOw0KDQogICAgdmFyIHN0ZXAyID0gb3B0aW9ucy5tZXRob2QgKyAnOicgKyBvcHRpb25zLnBhdGg7DQogICAgYXV0aC5zdGVwMiA9IG1kNS5zeW5jSGFzaChzdGVwMikudG9TdHJpbmcoJ2hleCcpLnRvTG93ZXJDYXNlKCk7DQoNCg0KICAgIGlmIChhdXRoLnFvcCA9PSBudWxsKQ0KICAgIHsNCiAgICAgICAgdmFyIHN0ZXAzID0gYXV0aC5zdGVwMSArICc6JyArIGF1dGgubm9uY2UgKyAnOicgKyBhdXRoLnN0ZXAyOw0KICAgICAgICBhdXRoLnN0ZXAzID0gbWQ1LnN5bmNIYXNoKHN0ZXAzKS50b1N0cmluZygnaGV4JykudG9Mb3dlckNhc2UoKTsNCiAgICB9DQogICAgZWxzZQ0KICAgIHsNCiAgICAgICAgZGlnZXN0Ll9OQyArPSAxOwogICAgICAgIHZhciBzdGVwMyA9IGF1dGguc3RlcDEgKyAnOicgKyBhdXRoLm5vbmNlICsgJzonICsgZGlnZXN0Ll9OQy50b1N0cmluZygxNikudG9Mb3dlckNhc2UoKS5wYWRTdGFydCg4LCAnMCcpICsgJzonICsgZGlnZXN0Ll9DTk9OQ0UgKyAnOicgKyBhdXRoLnFvcCArICc6JyArIGF1dGguc3RlcDI7CiAgICAgICAgYXV0aC5zdGVwMyA9IG1kNS5zeW5jSGFzaChzdGVwMykudG9TdHJpbmcoJ2hleCcpLnRvTG93ZXJDYXNlKCk7DQogICAgfQ0KDQogICAgdmFyIHJldCA9ICdEaWdlc3QgdXNlcm5hbWU9IicgKyBkaWdlc3QuX29wdGlvbnMudXNlcm5hbWUgKyAnIixyZWFsbT0iJyArIGF1dGgucmVhbG0gKyAnIixub25jZT0iJyArIGF1dGgubm9uY2UgKyAnIix1cmk9IicgKyBvcHRpb25zLnBhdGggKyAnIic7DQogICAgaWYgKGF1dGgub3BhcXVlICE9IG51bGwpIHsgcmV0ICs9ICgnLG9wYXF1ZT0iJyArIGF1dGgub3BhcXVlICsgJyInKTsgfQ0KICAgIHJldCArPSAoJyxyZXNwb25zZT0iJyArIGF1dGguc3RlcDMgKyAnIicpOw0KDQogICAgaWYgKGF1dGgucW9wICE9IG51bGwpDQogICAgew0KICAgICAgICByZXQgKz0gKCcscW9wPSInICsgYXV0aC5xb3AgKyAnIixuYz0iJyArIGRpZ2VzdC5fTkMudG9TdHJpbmcoMTYpLnRvTG93ZXJDYXNlKCkucGFkU3RhcnQoOCwgJzAnKSArICciLGNub25jZT0iJyArIGRpZ2VzdC5fQ05PTkNFICsgJyInKTsNCiAgICB9DQoNCg0KICAgIGlmICghb3B0aW9ucy5oZWFkZXJzKSB7IG9wdGlvbnMuaGVhZGVycyA9IHt9OyB9DQogICAgb3B0aW9ucy5oZWFkZXJzWydBdXRob3JpemF0aW9uJ10gPSByZXQ7DQogICAgcmV0dXJuIChyZXQpOw0KfQ0KDQpmdW5jdGlvbiBodHRwX2RpZ2VzdCgpDQp7DQogICAgdGhpcy5fT2JqZWN0SUQgPSAiaHR0cC1kaWdlc3QiOw0KICAgIHRoaXMuY3JlYXRlID0gZnVuY3Rpb24oKQ0KICAgIHsNCiAgICAgICAgaWYoYXJndW1lbnRzLmxlbmd0aCA9PSAxICYmIHR5cGVvZihhcmd1bWVudHNbMF0gPT0gJ29iamVjdCcpKQ0KICAgICAgICB7DQogICAgICAgICAgICByZXR1cm4gKG5ldyBodHRwX2RpZ2VzdF9pbnN0YW5jZShhcmd1bWVudHNbMF0pKTsNCiAgICAgICAgfQ0KICAgICAgICBpZihhcmd1bWVudHMubGVuZ3RoID09IDIgJiYgdHlwZW9mKGFyZ3VtZW50c1swXSkgPT0gJ3N0cmluZycgJiYgdHlwZW9mKGFyZ3VtZW50c1sxXSkgPT0gJ3N0cmluZycpDQogICAgICAgIHsNCiAgICAgICAgICAgIHJldHVybiAobmV3IGh0dHBfZGlnZXN0X2luc3RhbmNlKHt1c2VybmFtZTogYXJndW1lbnRzWzBdLCBwYXNzd29yZDogYXJndW1lbnRzWzFdfSkpOw0KICAgICAgICB9DQogICAgICAgIHRocm93ICgnSW52YWxpZCBQYXJhbWV0ZXJzJyk7DQogICAgfQ0KfQ0KDQpmdW5jdGlvbiBodHRwX2RpZ2VzdF9pbnN0YW5jZShvcHRpb25zKQ0Kew0KICAgIHRoaXMuX09iamVjdElEID0gJ2h0dHAtZGlnZXN0Lmluc3RhbmNlJzsNCiAgICB0aGlzLl9vcHRpb25zID0gb3B0aW9uczsNCiAgICB0aGlzLmh0dHAgPSBudWxsOw0KICAgIHRoaXMuX05DID0gMDsNCiAgICB0aGlzLl9DTk9OQ0UgPSByZXF1aXJlKCdodHRwJykuZ2VuZXJhdGVOb25jZSgxNik7DQoNCiAgICB0aGlzLmdldCA9IGZ1bmN0aW9uKHVyaSkNCiAgICB7DQogICAgICAgIHJldHVybiAodGhpcy5yZXF1ZXN0KHVyaSkpOw0KICAgIH0NCiAgICB0aGlzLnJlcXVlc3QgPSBmdW5jdGlvbiAocGFyMSkNCiAgICB7DQogICAgICAgIHZhciBjYWxsZW5kID0gZmFsc2U7DQogICAgICAgIHZhciByZXQgPSBuZXcgd3JpdGFibGUoDQogICAgICAgICAgICB7DQogICAgICAgICAgICAgICAgd3JpdGU6IGZ1bmN0aW9uIChjaHVuaywgZmx1c2gpDQogICAgICAgICAgICAgICAgew0KICAgICAgICAgICAgICAgICAgICBpZiAodGhpcy5fZW5kZWQpIHsgdGhyb3cgKCdTdHJlYW0gYWxyZWFkeSBlbmRlZCcpOyB9DQogICAgICAgICAgICAgICAgICAgIGlmKCF0aGlzLl9idWZmZXJlZCkgDQogICAgICAgICAgICAgICAgICAgIHsNCiAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX2J1ZmZlcmVkID0gQnVmZmVyLmFsbG9jKGNodW5rLmxlbmd0aCk7DQogICAgICAgICAgICAgICAgICAgICAgICBjaHVuay5jb3B5KHRoaXMuX2J1ZmZlcmVkKTsNCiAgICAgICAgICAgICAgICAgICAgfQ0KICAgICAgICAgICAgICAgICAgICBlbHNlDQogICAgICAgICAgICAgICAgICAgIHsNCiAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX2J1ZmZlcmVkID0gQnVmZmVyLmNvbmNhdChbdGhpcy5fYnVmZmVyZWQsIGNodW5rXSwgdGhpcy5fYnVmZmVyZWQubGVuZ3RoICsgY2h1bmsubGVuZ3RoKTsNCiAgICAgICAgICAgICAgICAgICAgfQ0KDQogICAgICAgICAgICAgICAgICAgIGlmICh0aGlzLl9yZXF1ZXN0KSB7IHRoaXMuX3JlcXVlc3Qud3JpdGUoY2h1bmspOyB9DQogICAgICAgICAgICAgICAgICAgIGlmIChmbHVzaCAhPSBudWxsKSB7IGZsdXNoKCk7IH0NCiAgICAgICAgICAgICAgICAgICAgcmV0dXJuICh0cnVlKTsNCiAgICAgICAgICAgICAgICB9LA0KICAgICAgICAgICAgICAgIGZpbmFsOiBmdW5jdGlvbiAoZmx1c2gpDQogICAgICAgICAgICAgICAgew0KICAgICAgICAgICAgICAgICAgICBpZiAodGhpcy5fZW5kZWQpIHsgdGhyb3cgKCdTdHJlYW0gYWxyZWFkeSBlbmRlZCcpOyB9DQogICAgICAgICAgICAgICAgICAgIHRoaXMuX2VuZGVkID0gdHJ1ZTsNCiAgICAgICAgICAgICAgICAgICAgaWYgKHRoaXMuX3JlcXVlc3QpIHsgdGhpcy5fcmVxdWVzdC5lbmQoKTsgfQ0KICAgICAgICAgICAgICAgICAgICBpZiAoZmx1c2ggIT0gbnVsbCkgeyBmbHVzaCgpOyB9DQogICAgICAgICAgICAgICAgfQ0KICAgICAgICAgICAgfSk7DQogICAgICAgIHJldC5fYnVmZmVyZWQgPSBudWxsOw0KICAgICAgICByZXQuX2VuZGVkID0gZmFsc2U7DQogICAgICAgIHN3aXRjaCAodHlwZW9mIChwYXIxKSkNCiAgICAgICAgew0KICAgICAgICAgICAgZGVmYXVsdDoNCiAgICAgICAgICAgICAgICB0aHJvdyAoJ0ludmFsaWQgUGFyYW1ldGVyJyk7DQogICAgICAgICAgICAgICAgYnJlYWs7DQogICAgICAgICAgICBjYXNlICdzdHJpbmcnOg0KICAgICAgICAgICAgICAgIHJldC5vcHRpb25zID0gdGhpcy5odHRwLnBhcnNlVXJpKHBhcjEpOw0KICAgICAgICAgICAgICAgIGNhbGxlbmQgPSB0cnVlOw0KICAgICAgICAgICAgICAgIGJyZWFrOw0KICAgICAgICAgICAgY2FzZSAnb2JqZWN0JzoNCiAgICAgICAgICAgICAgICByZXQub3B0aW9ucyA9IHBhcjE7DQogICAgICAgICAgICAgICAgYnJlYWs7DQogICAgICAgIH0NCiAgICAgICAgcmVxdWlyZSgnZXZlbnRzJykuRXZlbnRFbWl0dGVyLmNhbGwocmV0LCB0cnVlKQ0KICAgICAgICAgICAgLmNyZWF0ZUV2ZW50KCdyZXNwb25zZScpDQogICAgICAgICAgICAuY3JlYXRlRXZlbnQoJ2Vycm9yJykNCiAgICAgICAgICAgIC5jcmVhdGVFdmVudCgndXBncmFkZScpDQogICAgICAgICAgICAuY3JlYXRlRXZlbnQoJ2NvbnRpbnVlJykNCiAgICAgICAgICAgIC5jcmVhdGVFdmVudCgndGltZW91dCcpOw0KICAgICAgICByZXQuX2RpZ2VzdCA9IHRoaXM7DQoNCiAgICAgICAgaWYgKGFyZ3VtZW50cy5sZW5ndGggPiAxICYmIHR5cGVvZiAoYXJndW1lbnRzWzFdKSA9PSAnZnVuY3Rpb24nKQ0KICAgICAgICB7DQogICAgICAgICAgICByZXQub25jZSgncmVzcG9uc2UnLCBhcmd1bWVudHNbMV0pOw0KICAgICAgICB9DQoNCiAgICAgICAgLy8NCiAgICAgICAgLy8gQ2hlY2sgaWYgd2UgY2FuIGFkZCBBdXRoSGVhZGVycyBub3cNCiAgICAgICAgLy8NCiAgICAgICAgZ2VuZXJhdGVBdXRoSGVhZGVycyhudWxsLCByZXQub3B0aW9ucywgdGhpcyk7DQoNCiAgICAgICAgLy8gV2hlbiBzb21lYm9keSBob29rcyB1cCBldmVudHMgdG8gZGlnZXN0LmNsaWVudFJlcXVlc3QsIHdlIG5lZWQgdG8gaG9vayB0aGUgcmVhbCBldmVudCBvbiBodHRwLmNsaWVudFJlcXVlc3QNCiAgICAgICAgcmV0Ll9yZXF1ZXN0ID0gdGhpcy5odHRwLnJlcXVlc3QocmV0Lm9wdGlvbnMpOw0KICAgICAgICByZXQuX3JlcXVlc3QuZGlnUmVxdWVzdCA9IHJldDsNCiAgICAgICAgcmV0Lm9uKCdfZXZlbnRIb29rJywgZnVuY3Rpb24gKGV2TmFtZSwgY2FsbGJhY2spDQogICAgICAgIHsNCiAgICAgICAgICAgIGlmIChldk5hbWUgIT0gJ3VwZ3JhZGUnICYmIGV2TmFtZSAhPSAnZXJyb3InICYmIGV2TmFtZSAhPSAnY29udGludWUnICYmIGV2TmFtZSAhPSAndGltZW91dCcgJiYgZXZOYW1lICE9ICdkcmFpbicpIHsgcmV0dXJuOyB9DQogICAgICAgICAgICBpZiAodGhpcy5fcmVxdWVzdC5saXN0ZW5lckNvdW50KGV2TmFtZSkgPT0gMCkNCiAgICAgICAgICAgIHsNCiAgICAgICAgICAgICAgICB2YXIgZXZTaW5rID0gZnVuY3Rpb24gX2V2U2luaygpDQogICAgICAgICAgICAgICAgew0KICAgICAgICAgICAgICAgICAgICB2YXIgcGFybXMgPSBbX2V2U2luay5ldmVudE5hbWVdOw0KICAgICAgICAgICAgICAgICAgICBmb3IodmFyIGk9MDtpPGFyZ3VtZW50cy5sZW5ndGg7KytpKSB7cGFybXMucHVzaChhcmd1bWVudHNbaV0pO30NCiAgICAgICAgICAgICAgICAgICAgdGhpcy5kaWdSZXF1ZXN0LmVtaXQuYXBwbHkodGhpcy5kaWdSZXF1ZXN0LCBwYXJtcyk7DQogICAgICAgICAgICAgICAgfTsNCiAgICAgICAgICAgICAgICBldlNpbmsuZXZlbnROYW1lID0gZXZOYW1lOw0KICAgICAgICAgICAgICAgIHRoaXMuX3JlcXVlc3Qub24oZXZOYW1lLCBldlNpbmspOw0KICAgICAgICAgICAgfQ0KICAgICAgICB9KTsNCg0KICAgICAgICByZXQuX3JlcXVlc3Qub25jZSgncmVzcG9uc2UnLCBmdW5jdGlvbiAoaW1zZykNCiAgICAgICAgew0KICAgICAgICAgICAgaWYgKGltc2cuc3RhdHVzQ29kZSA9PSA0MDEpDQogICAgICAgICAgICB7DQogICAgICAgICAgICAgICAgdmFyIGNhbGxlbmQgPSB0aGlzLmRpZ1JlcXVlc3QuX3JlcXVlc3QuX2NhbGxlbmQ7DQogICAgICAgICAgICAgICAgdmFyIGF1dGggPSBnZW5lcmF0ZUF1dGhIZWFkZXJzKGltc2csIHRoaXMuZGlnUmVxdWVzdC5vcHRpb25zLCB0aGlzLmRpZ1JlcXVlc3QuX2RpZ2VzdCk7DQoNCiAgICAgICAgICAgICAgICB0aGlzLmRpZ1JlcXVlc3QuX3JlcXVlc3QgPSB0aGlzLmRpZ1JlcXVlc3QuX2RpZ2VzdC5odHRwLnJlcXVlc3QodGhpcy5kaWdSZXF1ZXN0Lm9wdGlvbnMpOw0KICAgICAgICAgICAgICAgIHRoaXMuZGlnUmVxdWVzdC5fcmVxdWVzdC5kaWdSZXF1ZXN0ID0gdGhpcy5kaWdSZXF1ZXN0Ow0KICAgICAgICAgICAgICAgIHRoaXMuZGlnUmVxdWVzdC5fcmVxdWVzdC5vbmNlKCdyZXNwb25zZScsIGZ1bmN0aW9uIChpbXNnKQ0KICAgICAgICAgICAgICAgIHsNCiAgICAgICAgICAgICAgICAgICAgc3dpdGNoKGltc2cuc3RhdHVzQ29kZSkNCiAgICAgICAgICAgICAgICAgICAgew0KICAgICAgICAgICAgICAgICAgICAgICAgY2FzZSA0MDE6DQogICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5kaWdSZXF1ZXN0LmVtaXQoJ2Vycm9yJywgJ0RpZ2VzdCBmYWlsZWQgdG9vIG1hbnkgdGltZXMnKTsNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBicmVhazsNCiAgICAgICAgICAgICAgICAgICAgICAgIGRlZmF1bHQ6DQogICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5kaWdSZXF1ZXN0LmVtaXQoJ3Jlc3BvbnNlJywgaW1zZyk7DQogICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7DQogICAgICAgICAgICAgICAgICAgIH0NCiAgICAgICAgICAgICAgICB9KTsNCiAgICAgICAgICAgICAgICBjaGVja0V2ZW50Rm9yd2FyZGluZyh0aGlzLmRpZ1JlcXVlc3QsICd1cGdyYWRlJyk7DQogICAgICAgICAgICAgICAgY2hlY2tFdmVudEZvcndhcmRpbmcodGhpcy5kaWdSZXF1ZXN0LCAnZXJyb3InKTsNCiAgICAgICAgICAgICAgICBjaGVja0V2ZW50Rm9yd2FyZGluZyh0aGlzLmRpZ1JlcXVlc3QsICdjb250aW51ZScpOw0KICAgICAgICAgICAgICAgIGNoZWNrRXZlbnRGb3J3YXJkaW5nKHRoaXMuZGlnUmVxdWVzdCwgJ3RpbWVvdXQnKTsNCiAgICAgICAgICAgICAgICBjaGVja0V2ZW50Rm9yd2FyZGluZyh0aGlzLmRpZ1JlcXVlc3QsICdkcmFpbicpOw0KICAgICAgICAgICAgICAgIGlmIChjYWxsZW5kKQ0KICAgICAgICAgICAgICAgIHsNCiAgICAgICAgICAgICAgICAgICAgdGhpcy5kaWdSZXF1ZXN0Ll9yZXF1ZXN0LmVuZCgpOw0KICAgICAgICAgICAgICAgIH0NCiAgICAgICAgICAgICAgICBlbHNlDQogICAgICAgICAgICAgICAgew0KICAgICAgICAgICAgICAgICAgICBpZiAodGhpcy5kaWdSZXF1ZXN0Ll9idWZmZXJlZCkgeyB0aGlzLmRpZ1JlcXVlc3QuX3JlcXVlc3Qud3JpdGUodGhpcy5kaWdSZXF1ZXN0Ll9idWZmZXJlZCk7IH0NCiAgICAgICAgICAgICAgICAgICAgaWYgKHRoaXMuZGlnUmVxdWVzdC5fZW5kZWQpIHsgdGhpcy5kaWdSZXF1ZXN0Ll9yZXF1ZXN0LmVuZCgpOyB9DQogICAgICAgICAgICAgICAgfQ0KICAgICAgICAgICAgfQ0KICAgICAgICAgICAgZWxzZQ0KICAgICAgICAgICAgew0KICAgICAgICAgICAgICAgIHRoaXMuZGlnUmVxdWVzdC5lbWl0KCdyZXNwb25zZScsIGltc2cpOw0KICAgICAgICAgICAgfQ0KICAgICAgICB9KTsNCiAgICAgICAgaWYgKGNhbGxlbmQpDQogICAgICAgIHsNCiAgICAgICAgICAgIHJldC5fcmVxdWVzdC5fY2FsbGVuZCA9IHRydWU7IHJldC5fcmVxdWVzdC5lbmQoKTsNCiAgICAgICAgfQ0KICAgICAgICBlbHNlDQogICAgICAgIHsNCiAgICAgICAgICAgIGlmIChyZXQuX2J1ZmZlcmVkKSB7IHJldC5fcmVxdWVzdC53cml0ZShyZXQuX2J1ZmZlcmVkKTsgfQ0KICAgICAgICAgICAgaWYgKHJldC5fZW5kZWQpIHsgcmV0Ll9yZXF1ZXN0LmVuZCgpOyB9DQogICAgICAgIH0NCiAgICAgICAgcmV0dXJuIChyZXQpOw0KICAgIH07DQp9DQoNCg0KbW9kdWxlLmV4cG9ydHMgPSBuZXcgaHR0cF9kaWdlc3QoKTsNCg0K', 'base64').toString());");
|
|
|
|
// Clipboard. Refer to /modules folder for a human readable version
|
|
duk_peval_string_noresult(ctx, "addModule('clipboard', Buffer.from('LyoKQ29weXJpZ2h0IDIwMTkgSW50ZWwgQ29ycG9yYXRpb24KCkxpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwp5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCllvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAoKICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAoKVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQpkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLApXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZApsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KKi8KCnZhciBwcm9taXNlID0gcmVxdWlyZSgncHJvbWlzZScpOwoKZnVuY3Rpb24gbmF0aXZlQWRkTW9kdWxlKG5hbWUpCnsKICAgIHZhciB2YWx1ZSA9IGdldEpTTW9kdWxlKG5hbWUpOwogICAgdmFyIHJldCA9ICJkdWtfcGV2YWxfc3RyaW5nX25vcmVzdWx0KGN0eCwgXCJhZGRNb2R1bGUoJyIgKyBuYW1lICsgIicsIEJ1ZmZlci5mcm9tKCciICsgQnVmZmVyLmZyb20odmFsdWUpLnRvU3RyaW5nKCdiYXNlNjQnKSArICInLCAnYmFzZTY0JykudG9TdHJpbmcoKSk7XCIpOyI7CiAgICBtb2R1bGUuZXhwb3J0cyhyZXQpOwp9CgpmdW5jdGlvbiBsaW5fcmVhZHRleHQoKQp7CiAgICB2YXIgcmV0ID0gbmV3IHByb21pc2UoZnVuY3Rpb24gKHJlcywgcmVqKSB7IHRoaXMuX3JlcyA9IHJlczsgdGhpcy5fcmVqID0gcmVqOyB9KTsKICAgIHRyeQogICAgewogICAgICAgIHJlcXVpcmUoJ21vbml0b3ItaW5mbycpDQogICAgfQogICAgY2F0Y2goZXhjKQogICAgew0KICAgICAgICByZXQuX3JlaihleGMpOw0KICAgICAgICByZXR1cm4gKHJldCk7DQogICAgfQoKICAgIHZhciBYMTEgPSByZXF1aXJlKCdtb25pdG9yLWluZm8nKS5fWDExOwogICAgaWYgKCFYMTEpCiAgICB7DQogICAgICAgIHJldC5fcmVqKCdYMTEgcmVxdWlyZWQgZm9yIENsaXBib2FyZCBNYW5pcHVsYXRpb24nKTsNCiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgICAgdmFyIFNlbGVjdGlvbk5vdGlmeSA9IDMxOwogICAgICAgIHZhciBBbnlQcm9wZXJ0eVR5cGUgPSAwOwogICAgICAgIHZhciBHTSA9IHJlcXVpcmUoJ21vbml0b3ItaW5mbycpLl9nbTsKCiAgICAgICAgcmV0Ll9nZXRJbmZvUHJvbWlzZSA9IHJlcXVpcmUoJ21vbml0b3ItaW5mbycpLmdldEluZm8oKTsKICAgICAgICByZXQuX2dldEluZm9Qcm9taXNlLl9tYXN0ZXJQcm9taXNlID0gcmV0OwogICAgICAgIHJldC5fZ2V0SW5mb1Byb21pc2UudGhlbihmdW5jdGlvbiAobW9uKQogICAgICAgIHsNCiAgICAgICAgICAgIGlmIChtb24ubGVuZ3RoID4gMCkNCiAgICAgICAgICAgIHsNCiAgICAgICAgICAgICAgICB2YXIgd2hpdGUgPSBYMTEuWFdoaXRlUGl4ZWwobW9uWzBdLmRpc3BsYXksIG1vblswXS5zY3JlZW5JZCkuVmFsOw0KDQogICAgICAgICAgICAgICAgdGhpcy5fbWFzdGVyUHJvbWlzZS5DTElQSUQgPSBYMTEuWEludGVybkF0b20obW9uWzBdLmRpc3BsYXksIEdNLkNyZWF0ZVZhcmlhYmxlKCdDTElQQk9BUkQnKSwgMCk7DQogICAgICAgICAgICAgICAgdGhpcy5fbWFzdGVyUHJvbWlzZS5GTVRJRCA9IFgxMS5YSW50ZXJuQXRvbShtb25bMF0uZGlzcGxheSwgR00uQ3JlYXRlVmFyaWFibGUoJ1VURjhfU1RSSU5HJyksIDApOw0KICAgICAgICAgICAgICAgIHRoaXMuX21hc3RlclByb21pc2UuUFJPUElEID0gWDExLlhJbnRlcm5BdG9tKG1vblswXS5kaXNwbGF5LCBHTS5DcmVhdGVWYXJpYWJsZSgnWFNFTF9EQVRBJyksIDApOw0KICAgICAgICAgICAgICAgIHRoaXMuX21hc3RlclByb21pc2UuSU5DUklEID0gWDExLlhJbnRlcm5BdG9tKG1vblswXS5kaXNwbGF5LCBHTS5DcmVhdGVWYXJpYWJsZSgnSU5DUicpLCAwKTsNCiAgICAgICAgICAgICAgICB0aGlzLl9tYXN0ZXJQcm9taXNlLlJPT1RXSU4gPSBYMTEuWFJvb3RXaW5kb3cobW9uWzBdLmRpc3BsYXksIG1vblswXS5zY3JlZW5JZCk7DQogICAgICAgICAgICAgICAgdGhpcy5fbWFzdGVyUHJvbWlzZS5GQUtFV0lOID0gWDExLlhDcmVhdGVTaW1wbGVXaW5kb3cobW9uWzBdLmRpc3BsYXksIHRoaXMuX21hc3RlclByb21pc2UuUk9PVFdJTiwgMCwgMCwgbW9uWzBdLnJpZ2h0LCA1LCAwLCB3aGl0ZSwgd2hpdGUpOw0KDQogICAgICAgICAgICAgICAgWDExLlhTeW5jKG1vblswXS5kaXNwbGF5LCAwKTsNCiAgICAgICAgICAgICAgICBYMTEuWENvbnZlcnRTZWxlY3Rpb24obW9uWzBdLmRpc3BsYXksIHRoaXMuX21hc3RlclByb21pc2UuQ0xJUElELCB0aGlzLl9tYXN0ZXJQcm9taXNlLkZNVElELCB0aGlzLl9tYXN0ZXJQcm9taXNlLlBST1BJRCwgdGhpcy5fbWFzdGVyUHJvbWlzZS5GQUtFV0lOLCAwKTsNCiAgICAgICAgICAgICAgICBYMTEuWFN5bmMobW9uWzBdLmRpc3BsYXksIDApOw0KDQogICAgICAgICAgICAgICAgdGhpcy5fbWFzdGVyUHJvbWlzZS5EZXNjcmlwdG9yRXZlbnQgPSByZXF1aXJlKCdEZXNjcmlwdG9yRXZlbnRzJykuYWRkRGVzY3JpcHRvcihYMTEuWENvbm5lY3Rpb25OdW1iZXIobW9uWzBdLmRpc3BsYXkpLlZhbCwgeyByZWFkc2V0OiB0cnVlIH0pOw0KICAgICAgICAgICAgICAgIHRoaXMuX21hc3RlclByb21pc2UuRGVzY3JpcHRvckV2ZW50Ll9tYXN0ZXJQcm9taXNlID0gdGhpcy5fbWFzdGVyUHJvbWlzZTsNCiAgICAgICAgICAgICAgICB0aGlzLl9tYXN0ZXJQcm9taXNlLkRlc2NyaXB0b3JFdmVudC5fZGlzcGxheSA9IG1vblswXS5kaXNwbGF5Ow0KICAgICAgICAgICAgICAgIHRoaXMuX21hc3RlclByb21pc2UuRGVzY3JpcHRvckV2ZW50Lm9uKCdyZWFkc2V0JywgZnVuY3Rpb24gKGZkKQ0KICAgICAgICAgICAgICAgIHsNCiAgICAgICAgICAgICAgICAgICAgdmFyIFhFID0gR00uQ3JlYXRlVmFyaWFibGUoMTAyNCk7DQogICAgICAgICAgICAgICAgICAgIHdoaWxlIChYMTEuWFBlbmRpbmcodGhpcy5fZGlzcGxheSkuVmFsKQ0KICAgICAgICAgICAgICAgICAgICB7DQogICAgICAgICAgICAgICAgICAgICAgICBYMTEuWE5leHRFdmVudFN5bmModGhpcy5fZGlzcGxheSwgWEUpOw0KICAgICAgICAgICAgICAgICAgICAgICAgaWYoWEUuRGVyZWYoMCwgNCkudG9CdWZmZXIoKS5yZWFkVUludDMyTEUoKSA9PSBTZWxlY3Rpb25Ob3RpZnkpDQogICAgICAgICAgICAgICAgICAgICAgICB7DQogICAgICAgICAgICAgICAgICAgICAgICAgICAgdmFyIGlkID0gR00uQ3JlYXRlUG9pbnRlcigpOwogICAgICAgICAgICAgICAgICAgICAgICAgICAgdmFyIGJpdHMgPSBHTS5DcmVhdGVQb2ludGVyKCk7CiAgICAgICAgICAgICAgICAgICAgICAgICAgICB2YXIgc3ogPSBHTS5DcmVhdGVQb2ludGVyKCk7CiAgICAgICAgICAgICAgICAgICAgICAgICAgICB2YXIgdGFpbCA9IEdNLkNyZWF0ZVBvaW50ZXIoKTsKICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZhciByZXN1bHQgPSBHTS5DcmVhdGVQb2ludGVyKCk7DQoNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBYMTEuWEdldFdpbmRvd1Byb3BlcnR5KHRoaXMuX2Rpc3BsYXksIHRoaXMuX21hc3RlclByb21pc2UuRkFLRVdJTiwgdGhpcy5fbWFzdGVyUHJvbWlzZS5QUk9QSUQsIDAsIDY1NTM1LCAwLCBBbnlQcm9wZXJ0eVR5cGUsIGlkLCBiaXRzLCBzeiwgdGFpbCwgcmVzdWx0KTsNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aGlzLl9tYXN0ZXJQcm9taXNlLl9yZXMocmVzdWx0LkRlcmVmKCkuU3RyaW5nKTsNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBYMTEuWEZyZWUocmVzdWx0LkRlcmVmKCkpOw0KICAgICAgICAgICAgICAgICAgICAgICAgICAgIFgxMS5YRGVzdHJveVdpbmRvdyh0aGlzLl9kaXNwbGF5LCB0aGlzLl9tYXN0ZXJQcm9taXNlLkZBS0VXSU4pOw0KDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5yZW1vdmVEZXNjcmlwdG9yKGZkKTsNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBicmVhazsNCiAgICAgICAgICAgICAgICAgICAgICAgIH0NCiAgICAgICAgICAgICAgICAgICAgfQ0KICAgICAgICAgICAgICAgIH0pOw0KICAgICAgICAgICAgfQ0KICAgICAgICB9KTsKICAgIH0KICAgIHJldHVybiAocmV0KTsKfQpmdW5jdGlvbiBsaW5fY29weXRleHQoKQp7Cn0KCmZ1bmN0aW9uIHdpbl9yZWFkdGV4dCgpCnsKICAgIHZhciByZXQgPSAnJzsKICAgIHZhciBDRl9URVhUID0gMTsKICAgIHZhciBHTSA9IHJlcXVpcmUoJ19HZW5lcmljTWFyc2hhbCcpOwogICAgdmFyIHVzZXIzMiA9IEdNLkNyZWF0ZU5hdGl2ZVByb3h5KCd1c2VyMzIuZGxsJyk7CiAgICB2YXIga2VybmVsMzIgPSBHTS5DcmVhdGVOYXRpdmVQcm94eSgna2VybmVsMzIuZGxsJyk7CiAgICBrZXJuZWwzMi5DcmVhdGVNZXRob2QoJ0dsb2JhbEFsbG9jJyk7CiAgICBrZXJuZWwzMi5DcmVhdGVNZXRob2QoJ0dsb2JhbExvY2snKTsKICAgIGtlcm5lbDMyLkNyZWF0ZU1ldGhvZCgnR2xvYmFsVW5sb2NrJyk7CiAgICB1c2VyMzIuQ3JlYXRlTWV0aG9kKCdPcGVuQ2xpcGJvYXJkJyk7CiAgICB1c2VyMzIuQ3JlYXRlTWV0aG9kKCdDbG9zZUNsaXBib2FyZCcpOwogICAgdXNlcjMyLkNyZWF0ZU1ldGhvZCgnR2V0Q2xpcGJvYXJkRGF0YScpOwoKICAgIHVzZXIzMi5PcGVuQ2xpcGJvYXJkKDApOwogICAgdmFyIGggPSB1c2VyMzIuR2V0Q2xpcGJvYXJkRGF0YShDRl9URVhUKTsKICAgIGlmKGguVmFsIT0wKQogICAgewogICAgICAgIHZhciBoYnVmZmVyID0ga2VybmVsMzIuR2xvYmFsTG9jayhoKTsKICAgICAgICByZXQgPSBoYnVmZmVyLlN0cmluZzsKICAgICAgICBrZXJuZWwzMi5HbG9iYWxVbmxvY2soaCk7CiAgICB9CiAgICB1c2VyMzIuQ2xvc2VDbGlwYm9hcmQoKTsKCiAgICB2YXIgcCA9IG5ldyBwcm9taXNlKGZ1bmN0aW9uIChyZXMsIHJlaikgeyB0aGlzLl9yZXMgPSByZXM7IHRoaXMuX3JlaiA9IHJlajsgfSk7CiAgICBwLl9yZXMocmV0KTsKICAgIHJldHVybiAocCk7Cn0KCmZ1bmN0aW9uIHdpbl9jb3B5dGV4dCh0eHQpCnsKICAgIHZhciBHTUVNX01PVkVBQkxFID0gMHgwMDAyOwogICAgdmFyIENGX1RFWFQgPSAxOwoKICAgIHZhciBHTSA9IHJlcXVpcmUoJ19HZW5lcmljTWFyc2hhbCcpOwogICAgdmFyIHVzZXIzMiA9IEdNLkNyZWF0ZU5hdGl2ZVByb3h5KCd1c2VyMzIuZGxsJyk7CiAgICB2YXIga2VybmVsMzIgPSBHTS5DcmVhdGVOYXRpdmVQcm94eSgna2VybmVsMzIuZGxsJyk7CiAgICBrZXJuZWwzMi5DcmVhdGVNZXRob2QoJ0dsb2JhbEFsbG9jJyk7CiAgICBrZXJuZWwzMi5DcmVhdGVNZXRob2QoJ0dsb2JhbExvY2snKTsKICAgIGtlcm5lbDMyLkNyZWF0ZU1ldGhvZCgnR2xvYmFsVW5sb2NrJyk7CiAgICB1c2VyMzIuQ3JlYXRlTWV0aG9kKCdPcGVuQ2xpcGJvYXJkJyk7CiAgICB1c2VyMzIuQ3JlYXRlTWV0aG9kKCdFbXB0eUNsaXBib2FyZCcpOwogICAgdXNlcjMyLkNyZWF0ZU1ldGhvZCgnQ2xvc2VDbGlwYm9hcmQnKTsKICAgIHVzZXIzMi5DcmVhdGVNZXRob2QoJ1NldENsaXBib2FyZERhdGEnKTsKCiAgICB2YXIgaCA9IGtlcm5lbDMyLkdsb2JhbEFsbG9jKEdNRU1fTU9WRUFCTEUsIHR4dC5sZW5ndGggKyAyKTsKICAgIGguYXV0b0ZyZWUoZmFsc2UpOwogICAgdmFyIGhidWZmZXIgPSBrZXJuZWwzMi5HbG9iYWxMb2NrKGgpOwogICAgaGJ1ZmZlci5hdXRvRnJlZShmYWxzZSk7CiAgICB2YXIgdG1wID0gQnVmZmVyLmFsbG9jKHR4dC5sZW5ndGggKyAxKTsKICAgIEJ1ZmZlci5mcm9tKHR4dCkuY29weSh0bXApOwogICAgdG1wLmNvcHkoaGJ1ZmZlci5EZXJlZigwLCB0eHQubGVuZ3RoICsgMSkudG9CdWZmZXIoKSk7CiAgICBrZXJuZWwzMi5HbG9iYWxVbmxvY2soaCk7CgogICAgdXNlcjMyLk9wZW5DbGlwYm9hcmQoMCk7CiAgICB1c2VyMzIuRW1wdHlDbGlwYm9hcmQoKTsKICAgIHVzZXIzMi5TZXRDbGlwYm9hcmREYXRhKENGX1RFWFQsIGgpOwogICAgdXNlcjMyLkNsb3NlQ2xpcGJvYXJkKCk7Cn0KCnN3aXRjaChwcm9jZXNzLnBsYXRmb3JtKQp7CiAgICBjYXNlICd3aW4zMic6CiAgICAgICAgbW9kdWxlLmV4cG9ydHMgPSB3aW5fY29weXRleHQ7CiAgICAgICAgbW9kdWxlLmV4cG9ydHMucmVhZCA9IHdpbl9yZWFkdGV4dDsKICAgICAgICBicmVhazsKICAgIGNhc2UgJ2xpbnV4JzoKICAgICAgICBtb2R1bGUuZXhwb3J0cyA9IGxpbl9jb3B5dGV4dDsKICAgICAgICBtb2R1bGUuZXhwb3J0cy5yZWFkID0gbGluX3JlYWR0ZXh0OwogICAgICAgIGJyZWFrOwogICAgY2FzZSAnZGFyd2luJzoKICAgICAgICBicmVhazsKfQptb2R1bGUuZXhwb3J0cy5uYXRpdmVBZGRNb2R1bGUgPSBuYXRpdmVBZGRNb2R1bGU7', 'base64').toString());");
|
|
|
|
// Promise: This is very important, as it is used everywhere. Refer to /modules folder to see a human readable version of promise.js
|
|
duk_peval_string_noresult(ctx, "addModule('promise', Buffer.from('LyoKQ29weXJpZ2h0IDIwMTggSW50ZWwgQ29ycG9yYXRpb24KCkxpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwp5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCllvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAoKICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAoKVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQpkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLApXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZApsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KKi8KCnZhciByZWZUYWJsZSA9IHt9OwoKZnVuY3Rpb24gZXZlbnRfc3dpdGNoZXJfaGVscGVyKGRlc2lyZWRfY2FsbGVlLCB0YXJnZXQpCnsKICAgIHRoaXMuX09iamVjdElEID0gJ2V2ZW50X3N3aXRjaGVyJzsKICAgIHRoaXMuZnVuYyA9IGZ1bmN0aW9uIGZ1bmMoKQogICAgewogICAgICAgIHZhciBhcmdzID0gW107CiAgICAgICAgZm9yKHZhciBpIGluIGFyZ3VtZW50cykKICAgICAgICB7CiAgICAgICAgICAgIGFyZ3MucHVzaChhcmd1bWVudHNbaV0pOwogICAgICAgIH0KICAgICAgICByZXR1cm4gKGZ1bmMudGFyZ2V0LmFwcGx5KGZ1bmMuZGVzaXJlZCwgYXJncykpOwogICAgfTsKICAgIHRoaXMuZnVuYy5kZXNpcmVkID0gZGVzaXJlZF9jYWxsZWU7CiAgICB0aGlzLmZ1bmMudGFyZ2V0ID0gdGFyZ2V0OwogICAgdGhpcy5mdW5jLnNlbGYgPSB0aGlzOwp9CmZ1bmN0aW9uIGV2ZW50X3N3aXRjaGVyKGRlc2lyZWRfY2FsbGVlLCB0YXJnZXQpCnsKICAgIHJldHVybiAobmV3IGV2ZW50X3N3aXRjaGVyX2hlbHBlcihkZXNpcmVkX2NhbGxlZSwgdGFyZ2V0KSk7Cn0KCmZ1bmN0aW9uIFByb21pc2UocHJvbWlzZUZ1bmMpCnsKICAgIHRoaXMuX09iamVjdElEID0gJ3Byb21pc2UnOwogICAgdGhpcy5wcm9taXNlID0gdGhpczsKICAgIHRoaXMuX2ludGVybmFsID0geyBfT2JqZWN0SUQ6ICdwcm9taXNlLmludGVybmFsJywgcHJvbWlzZTogdGhpcywgZnVuYzogcHJvbWlzZUZ1bmMsIGNvbXBsZXRlZDogZmFsc2UsIGVycm9yczogZmFsc2UsIGNvbXBsZXRlZEFyZ3M6IFtdIH07CiAgICByZXF1aXJlKCdldmVudHMnKS5FdmVudEVtaXR0ZXIuY2FsbCh0aGlzLl9pbnRlcm5hbCk7CiAgICB0aGlzLl9pbnRlcm5hbC5vbignX2V2ZW50SG9vaycsIGZ1bmN0aW9uIChldmVudE5hbWUsIGV2ZW50Q2FsbGJhY2spCiAgICB7CiAgICAgICAgLy9jb25zb2xlLmxvZygnaG9vaycsIGV2ZW50TmFtZSwgJ2Vycm9ycy8nICsgdGhpcy5lcnJvcnMgKyAnIGNvbXBsZXRlZC8nICsgdGhpcy5jb21wbGV0ZWQpOwogICAgICAgIHZhciByID0gbnVsbDsKCiAgICAgICAgaWYgKGV2ZW50TmFtZSA9PSAncmVzb2x2ZWQnICYmICF0aGlzLmVycm9ycyAmJiB0aGlzLmNvbXBsZXRlZCkKICAgICAgICB7CiAgICAgICAgICAgIHIgPSBldmVudENhbGxiYWNrLmFwcGx5KHRoaXMsIHRoaXMuY29tcGxldGVkQXJncyk7CiAgICAgICAgICAgIGlmKHIhPW51bGwpCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIHRoaXMuZW1pdF9yZXR1cm5WYWx1ZSgncmVzb2x2ZWQnLCByKTsKICAgICAgICAgICAgfQogICAgICAgIH0KICAgICAgICBpZiAoZXZlbnROYW1lID09ICdyZWplY3RlZCcgJiYgdGhpcy5lcnJvcnMgJiYgdGhpcy5jb21wbGV0ZWQpCiAgICAgICAgewogICAgICAgICAgICBldmVudENhbGxiYWNrLmFwcGx5KHRoaXMsIHRoaXMuY29tcGxldGVkQXJncyk7CiAgICAgICAgfQogICAgICAgIGlmIChldmVudE5hbWUgPT0gJ3NldHRsZWQnICYmIHRoaXMuY29tcGxldGVkKQogICAgICAgIHsKICAgICAgICAgICAgZXZlbnRDYWxsYmFjay5hcHBseSh0aGlzLCBbXSk7CiAgICAgICAgfQogICAgfSk7CiAgICB0aGlzLl9pbnRlcm5hbC5yZXNvbHZlciA9IGZ1bmN0aW9uIF9yZXNvbHZlcigpCiAgICB7CiAgICAgICAgX3Jlc29sdmVyLl9zZWxmLmVycm9ycyA9IGZhbHNlOwogICAgICAgIF9yZXNvbHZlci5fc2VsZi5jb21wbGV0ZWQgPSB0cnVlOwogICAgICAgIF9yZXNvbHZlci5fc2VsZi5jb21wbGV0ZWRBcmdzID0gW107CiAgICAgICAgdmFyIGFyZ3MgPSBbJ3Jlc29sdmVkJ107CiAgICAgICAgaWYgKHRoaXMuZW1pdF9yZXR1cm5WYWx1ZSAmJiB0aGlzLmVtaXRfcmV0dXJuVmFsdWUoJ3Jlc29sdmVkJykgIT0gbnVsbCkKICAgICAgICB7CiAgICAgICAgICAgIF9yZXNvbHZlci5fc2VsZi5jb21wbGV0ZWRBcmdzLnB1c2godGhpcy5lbWl0X3JldHVyblZhbHVlKCdyZXNvbHZlZCcpKTsKICAgICAgICAgICAgYXJncy5wdXNoKHRoaXMuZW1pdF9yZXR1cm5WYWx1ZSgncmVzb2x2ZWQnKSk7CiAgICAgICAgfQogICAgICAgIGVsc2UKICAgICAgICB7CiAgICAgICAgICAgIGZvciAodmFyIGEgaW4gYXJndW1lbnRzKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBfcmVzb2x2ZXIuX3NlbGYuY29tcGxldGVkQXJncy5wdXNoKGFyZ3VtZW50c1thXSk7CiAgICAgICAgICAgICAgICBhcmdzLnB1c2goYXJndW1lbnRzW2FdKTsKICAgICAgICAgICAgfQogICAgICAgIH0KICAgICAgICBfcmVzb2x2ZXIuX3NlbGYuZW1pdC5hcHBseShfcmVzb2x2ZXIuX3NlbGYsIGFyZ3MpOwogICAgICAgIF9yZXNvbHZlci5fc2VsZi5lbWl0KCdzZXR0bGVkJyk7CiAgICB9OwogICAgdGhpcy5faW50ZXJuYWwucmVqZWN0b3IgPSBmdW5jdGlvbiBfcmVqZWN0b3IoKQogICAgewogICAgICAgIF9yZWplY3Rvci5fc2VsZi5lcnJvcnMgPSB0cnVlOwogICAgICAgIF9yZWplY3Rvci5fc2VsZi5jb21wbGV0ZWQgPSB0cnVlOwogICAgICAgIF9yZWplY3Rvci5fc2VsZi5jb21wbGV0ZWRBcmdzID0gW107CiAgICAgICAgdmFyIGFyZ3MgPSBbJ3JlamVjdGVkJ107CiAgICAgICAgZm9yICh2YXIgYSBpbiBhcmd1bWVudHMpCiAgICAgICAgewogICAgICAgICAgICBfcmVqZWN0b3IuX3NlbGYuY29tcGxldGVkQXJncy5wdXNoKGFyZ3VtZW50c1thXSk7CiAgICAgICAgICAgIGFyZ3MucHVzaChhcmd1bWVudHNbYV0pOwogICAgICAgIH0KCiAgICAgICAgX3JlamVjdG9yLl9zZWxmLmVtaXQuYXBwbHkoX3JlamVjdG9yLl9zZWxmLCBhcmdzKTsKICAgICAgICBfcmVqZWN0b3IuX3NlbGYuZW1pdCgnc2V0dGxlZCcpOwogICAgfTsKICAgIHRoaXMuY2F0Y2ggPSBmdW5jdGlvbihmdW5jKQogICAgewogICAgICAgIHRoaXMuX2ludGVybmFsLm9uY2UoJ3JlamVjdGVkJywgZXZlbnRfc3dpdGNoZXIodGhpcywgZnVuYykuZnVuYyk7CiAgICB9CiAgICB0aGlzLmZpbmFsbHkgPSBmdW5jdGlvbiAoZnVuYykKICAgIHsKICAgICAgICB0aGlzLl9pbnRlcm5hbC5vbmNlKCdzZXR0bGVkJywgZXZlbnRfc3dpdGNoZXIodGhpcywgZnVuYykuZnVuYyk7CiAgICB9OwogICAgdGhpcy50aGVuID0gZnVuY3Rpb24gKHJlc29sdmVkLCByZWplY3RlZCkKICAgIHsKICAgICAgICBpZiAocmVzb2x2ZWQpIHsgdGhpcy5faW50ZXJuYWwub25jZSgncmVzb2x2ZWQnLCBldmVudF9zd2l0Y2hlcih0aGlzLCByZXNvbHZlZCkuZnVuYyk7IH0KICAgICAgICBpZiAocmVqZWN0ZWQpIHsgdGhpcy5faW50ZXJuYWwub25jZSgncmVqZWN0ZWQnLCBldmVudF9zd2l0Y2hlcih0aGlzLCByZWplY3RlZCkuZnVuYyk7IH0KCiAgICAgICAgdmFyIHJldFZhbCA9IG5ldyBQcm9taXNlKGZ1bmN0aW9uIChyLCBqKSB7IH0pOwogICAgICAgIHRoaXMuX2ludGVybmFsLm9uY2UoJ3Jlc29sdmVkJywgcmV0VmFsLl9pbnRlcm5hbC5yZXNvbHZlcik7CiAgICAgICAgdGhpcy5faW50ZXJuYWwub25jZSgncmVqZWN0ZWQnLCByZXRWYWwuX2ludGVybmFsLnJlamVjdG9yKTsKICAgICAgICByZXRWYWwucGFyZW50UHJvbWlzZSA9IHRoaXM7CiAgICAgICAgcmV0dXJuIChyZXRWYWwpOwogICAgfTsKCiAgICB0aGlzLl9pbnRlcm5hbC5yZXNvbHZlci5fc2VsZiA9IHRoaXMuX2ludGVybmFsOwogICAgdGhpcy5faW50ZXJuYWwucmVqZWN0b3IuX3NlbGYgPSB0aGlzLl9pbnRlcm5hbDs7CgogICAgdHJ5CiAgICB7CiAgICAgICAgcHJvbWlzZUZ1bmMuY2FsbCh0aGlzLCB0aGlzLl9pbnRlcm5hbC5yZXNvbHZlciwgdGhpcy5faW50ZXJuYWwucmVqZWN0b3IpOwogICAgfQogICAgY2F0Y2goZSkKICAgIHsKICAgICAgICB0aGlzLl9pbnRlcm5hbC5lcnJvcnMgPSB0cnVlOwogICAgICAgIHRoaXMuX2ludGVybmFsLmNvbXBsZXRlZCA9IHRydWU7CiAgICAgICAgdGhpcy5faW50ZXJuYWwuY29tcGxldGVkQXJncyA9IFtlXTsKICAgICAgICB0aGlzLl9pbnRlcm5hbC5lbWl0KCdyZWplY3RlZCcsIGUpOwogICAgICAgIHRoaXMuX2ludGVybmFsLmVtaXQoJ3NldHRsZWQnKTsKICAgIH0KCiAgICBpZighdGhpcy5faW50ZXJuYWwuY29tcGxldGVkKQogICAgewogICAgICAgIC8vIFNhdmUgcmVmZXJlbmNlIG9mIHRoaXMgb2JqZWN0CiAgICAgICAgcmVmVGFibGVbdGhpcy5faW50ZXJuYWwuX2hhc2hDb2RlKCldID0gdGhpcy5faW50ZXJuYWw7CiAgICAgICAgdGhpcy5faW50ZXJuYWwub25jZSgnc2V0dGxlZCcsIGZ1bmN0aW9uICgpIHsgcmVmVGFibGVbdGhpcy5faGFzaENvZGUoKV0gPSBudWxsOyB9KTsKICAgIH0KfQoKUHJvbWlzZS5yZXNvbHZlID0gZnVuY3Rpb24gcmVzb2x2ZSgpCnsKICAgIHZhciByZXRWYWwgPSBuZXcgUHJvbWlzZShmdW5jdGlvbiAociwgaikgeyB9KTsKICAgIHZhciBhcmdzID0gW107CiAgICBmb3IgKHZhciBpIGluIGFyZ3VtZW50cykKICAgIHsKICAgICAgICBhcmdzLnB1c2goYXJndW1lbnRzW2ldKTsKICAgIH0KICAgIHJldFZhbC5faW50ZXJuYWwucmVzb2x2ZXIuYXBwbHkocmV0VmFsLl9pbnRlcm5hbCwgYXJncyk7CiAgICByZXR1cm4gKHJldFZhbCk7Cn07ClByb21pc2UucmVqZWN0ID0gZnVuY3Rpb24gcmVqZWN0KCkgewogICAgdmFyIHJldFZhbCA9IG5ldyBQcm9taXNlKGZ1bmN0aW9uIChyLCBqKSB7IH0pOwogICAgdmFyIGFyZ3MgPSBbXTsKICAgIGZvciAodmFyIGkgaW4gYXJndW1lbnRzKSB7CiAgICAgICAgYXJncy5wdXNoKGFyZ3VtZW50c1tpXSk7CiAgICB9CiAgICByZXRWYWwuX2ludGVybmFsLnJlamVjdG9yLmFwcGx5KHJldFZhbC5faW50ZXJuYWwsIGFyZ3MpOwogICAgcmV0dXJuIChyZXRWYWwpOwp9OwpQcm9taXNlLmFsbCA9IGZ1bmN0aW9uIGFsbChwcm9taXNlTGlzdCkKewogICAgdmFyIHJldCA9IG5ldyBQcm9taXNlKGZ1bmN0aW9uIChyZXMsIHJlaikKICAgIHsKICAgICAgICB0aGlzLl9fcmVqZWN0b3IgPSByZWo7CiAgICAgICAgdGhpcy5fX3Jlc29sdmVyID0gcmVzOwogICAgICAgIHRoaXMuX19wcm9taXNlTGlzdCA9IHByb21pc2VMaXN0OwogICAgICAgIHRoaXMuX19kb25lID0gZmFsc2U7CiAgICAgICAgdGhpcy5fX2NvdW50ID0gMDsKICAgIH0pOwoKICAgIGZvciAodmFyIGkgaW4gcHJvbWlzZUxpc3QpCiAgICB7CiAgICAgICAgcHJvbWlzZUxpc3RbaV0udGhlbihmdW5jdGlvbiAoKQogICAgICAgIHsKICAgICAgICAgICAgLy8gU3VjY2VzcwogICAgICAgICAgICBpZigrK3JldC5fX2NvdW50ID09IHJldC5fX3Byb21pc2VMaXN0Lmxlbmd0aCkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgcmV0Ll9fZG9uZSA9IHRydWU7CiAgICAgICAgICAgICAgICByZXQuX19yZXNvbHZlcihyZXQuX19wcm9taXNlTGlzdCk7CiAgICAgICAgICAgIH0KICAgICAgICB9LCBmdW5jdGlvbiAoYXJnKQogICAgICAgIHsKICAgICAgICAgICAgLy8gRmFpbHVyZQogICAgICAgICAgICBpZighcmV0Ll9fZG9uZSkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgcmV0Ll9fZG9uZSA9IHRydWU7CiAgICAgICAgICAgICAgICByZXQuX19yZWplY3RvcihhcmcpOwogICAgICAgICAgICB9CiAgICAgICAgfSk7CiAgICB9CiAgICBpZiAocHJvbWlzZUxpc3QubGVuZ3RoID09IDApCiAgICB7CiAgICAgICAgcmV0Ll9fcmVzb2x2ZXIocHJvbWlzZUxpc3QpOwogICAgfQogICAgcmV0dXJuIChyZXQpOwp9OwoKbW9kdWxlLmV4cG9ydHMgPSBQcm9taXNlOw==', 'base64').toString());");
|
|
|
|
#ifdef WIN32
|
|
// Adding win-registry, since it is very useful for windows... Refer to /modules folder to see a human readable version of win-registry.js
|
|
duk_peval_string_noresult(ctx, "addModule('win-registry', Buffer.from('LyoKQ29weXJpZ2h0IDIwMTggSW50ZWwgQ29ycG9yYXRpb24KCkxpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwp5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCllvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAoKICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAoKVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQpkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLApXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZApsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KKi8KCnZhciBLRVlfUVVFUllfVkFMVUUgPSAweDAwMDE7CnZhciBLRVlfRU5VTUVSQVRFX1NVQl9LRVlTID0gMHgwMDA4Owp2YXIgS0VZX1dSSVRFID0gMHgyMDAwNjsKCnZhciBLRVlfREFUQV9UWVBFUyA9CiAgICB7CiAgICAgICAgUkVHX05PTkU6IDAsCiAgICAgICAgUkVHX1NaOiAxLAogICAgICAgIFJFR19FWFBBTkRfU1o6IDIsCiAgICAgICAgUkVHX0JJTkFSWTogMywKICAgICAgICBSRUdfRFdPUkQ6IDQsCiAgICAgICAgUkVHX0RXT1JEX0JJR19FTkRJQU46IDUsCiAgICAgICAgUkVHX0xJTks6IDYsCiAgICAgICAgUkVHX01VTFRJX1NaOiA3LAogICAgICAgIFJFR19SRVNPVVJDRV9MSVNUOiA4LAogICAgICAgIFJFR19GVUxMX1JFU09VUkNFX0RFU0NSSVBUT1I6IDksCiAgICAgICAgUkVHX1JFU09VUkNFX1JFUVVJUkVNRU5UU19MSVNUOiAxMCwKICAgICAgICBSRUdfUVdPUkQ6IDExCiAgICB9OwoKZnVuY3Rpb24gd2luZG93c19yZWdpc3RyeSgpCnsKICAgIHRoaXMuX09iamVjdElkID0gJ3dpbi1yZWdpc3RyeSc7CiAgICB0aGlzLl9tYXJzaGFsID0gcmVxdWlyZSgnX0dlbmVyaWNNYXJzaGFsJyk7CiAgICB0aGlzLl9BZHZBcGkgPSB0aGlzLl9tYXJzaGFsLkNyZWF0ZU5hdGl2ZVByb3h5KCdBZHZhcGkzMi5kbGwnKTsKICAgIHRoaXMuX0FkdkFwaS5DcmVhdGVNZXRob2QoJ1JlZ0NyZWF0ZUtleUV4QScpOwogICAgdGhpcy5fQWR2QXBpLkNyZWF0ZU1ldGhvZCgnUmVnRW51bUtleUV4QScpOwogICAgdGhpcy5fQWR2QXBpLkNyZWF0ZU1ldGhvZCgnUmVnRW51bVZhbHVlQScpOwogICAgdGhpcy5fQWR2QXBpLkNyZWF0ZU1ldGhvZCgnUmVnT3BlbktleUV4QScpOwogICAgdGhpcy5fQWR2QXBpLkNyZWF0ZU1ldGhvZCgnUmVnUXVlcnlJbmZvS2V5QScpOwogICAgdGhpcy5fQWR2QXBpLkNyZWF0ZU1ldGhvZCgnUmVnUXVlcnlWYWx1ZUV4QScpOwogICAgdGhpcy5fQWR2QXBpLkNyZWF0ZU1ldGhvZCgnUmVnQ2xvc2VLZXknKTsKICAgIHRoaXMuX0FkdkFwaS5DcmVhdGVNZXRob2QoJ1JlZ0RlbGV0ZUtleUEnKTsKICAgIHRoaXMuX0FkdkFwaS5DcmVhdGVNZXRob2QoJ1JlZ0RlbGV0ZVZhbHVlQScpOwogICAgdGhpcy5fQWR2QXBpLkNyZWF0ZU1ldGhvZCgnUmVnU2V0VmFsdWVFeEEnKTsKICAgIHRoaXMuSEtFWSA9IHsgUm9vdDogQnVmZmVyLmZyb20oJzgwMDAwMDAwJywgJ2hleCcpLnN3YXAzMigpLCBDdXJyZW50VXNlcjogQnVmZmVyLmZyb20oJzgwMDAwMDAxJywgJ2hleCcpLnN3YXAzMigpLCBMb2NhbE1hY2hpbmU6IEJ1ZmZlci5mcm9tKCc4MDAwMDAwMicsICdoZXgnKS5zd2FwMzIoKSwgVXNlcnM6IEJ1ZmZlci5mcm9tKCc4MDAwMDAwMycsICdoZXgnKS5zd2FwMzIoKSB9OwoKICAgIHRoaXMuUXVlcnlLZXkgPSBmdW5jdGlvbiBRdWVyeUtleShoa2V5LCBwYXRoLCBrZXkpCiAgICB7CiAgICAgICAgdmFyIGVycjsKICAgICAgICB2YXIgaCA9IHRoaXMuX21hcnNoYWwuQ3JlYXRlUG9pbnRlcigpOwogICAgICAgIHZhciBsZW4gPSB0aGlzLl9tYXJzaGFsLkNyZWF0ZVZhcmlhYmxlKDQpOwogICAgICAgIHZhciB2YWxUeXBlID0gdGhpcy5fbWFyc2hhbC5DcmVhdGVWYXJpYWJsZSg0KTsKICAgICAgICB2YXIgSEsgPSB0aGlzLl9tYXJzaGFsLkNyZWF0ZVBvaW50ZXIoaGtleSk7CiAgICAgICAgdmFyIHJldFZhbCA9IG51bGw7CiAgICAgICAgaWYgKGtleSkgeyBrZXkgPSB0aGlzLl9tYXJzaGFsLkNyZWF0ZVZhcmlhYmxlKGtleSk7IH0KICAgICAgICBpZiAoIXBhdGgpIHsgcGF0aCA9ICcnOyB9CgoKICAgICAgICBpZiAoKGVyciA9IHRoaXMuX0FkdkFwaS5SZWdPcGVuS2V5RXhBKEhLLCB0aGlzLl9tYXJzaGFsLkNyZWF0ZVZhcmlhYmxlKHBhdGgpLCAwLCBLRVlfUVVFUllfVkFMVUUgfCBLRVlfRU5VTUVSQVRFX1NVQl9LRVlTLCBoKS5WYWwpICE9IDApCiAgICAgICAgewogICAgICAgICAgICB0aHJvdyAoJ09wZW5pbmcgUmVnaXN0cnkgS2V5OiAnICsgcGF0aCArICcgPT4gUmV0dXJuZWQgRXJyb3I6ICcgKyBlcnIpOwogICAgICAgIH0KICAKICAgICAgICBpZiAoKHBhdGggPT0gJycgJiYgIWtleSkgfHwgIWtleSkKICAgICAgICB7DQogICAgICAgICAgICB2YXIgcmVzdWx0ID0geyBzdWJrZXlzOiBbXSwgdmFsdWVzOiBbXSB9Ow0KDQogICAgICAgICAgICAvLyBFbnVtZXJhdGUgIGtleXMNCiAgICAgICAgICAgIHZhciBhY2hDbGFzcyA9IHRoaXMuX21hcnNoYWwuQ3JlYXRlVmFyaWFibGUoMTAyNCk7DQogICAgICAgICAgICB2YXIgYWNoS2V5ID0gdGhpcy5fbWFyc2hhbC5DcmVhdGVWYXJpYWJsZSgxMDI0KTsNCiAgICAgICAgICAgIHZhciBhY2hWYWx1ZSA9IHRoaXMuX21hcnNoYWwuQ3JlYXRlVmFyaWFibGUoMzI3NjgpOw0KICAgICAgICAgICAgdmFyIGFjaFZhbHVlU2l6ZSA9IHRoaXMuX21hcnNoYWwuQ3JlYXRlVmFyaWFibGUoNCk7DQogICAgICAgICAgICB2YXIgbmFtZVNpemUgPSB0aGlzLl9tYXJzaGFsLkNyZWF0ZVZhcmlhYmxlKDQpOyANCiAgICAgICAgICAgIHZhciBhY2hDbGFzc1NpemUgPSB0aGlzLl9tYXJzaGFsLkNyZWF0ZVZhcmlhYmxlKDQpOyBhY2hDbGFzc1NpemUudG9CdWZmZXIoKS53cml0ZVVJbnQzMkxFKDEwMjQpOw0KICAgICAgICAgICAgdmFyIG51bVN1YktleXMgPSB0aGlzLl9tYXJzaGFsLkNyZWF0ZVZhcmlhYmxlKDQpOw0KICAgICAgICAgICAgdmFyIG51bVZhbHVlcyA9IHRoaXMuX21hcnNoYWwuQ3JlYXRlVmFyaWFibGUoNCk7DQogICAgICAgICAgICB2YXIgbG9uZ2VzdFN1YmtleVNpemUgPSB0aGlzLl9tYXJzaGFsLkNyZWF0ZVZhcmlhYmxlKDQpOw0KICAgICAgICAgICAgdmFyIGxvbmdlc3RDbGFzc1N0cmluZyA9IHRoaXMuX21hcnNoYWwuQ3JlYXRlVmFyaWFibGUoNCk7DQogICAgICAgICAgICB2YXIgbG9uZ2VzdFZhbHVlTmFtZSA9IHRoaXMuX21hcnNoYWwuQ3JlYXRlVmFyaWFibGUoNCk7DQogICAgICAgICAgICB2YXIgbG9uZ2VzdFZhbHVlRGF0YSA9IHRoaXMuX21hcnNoYWwuQ3JlYXRlVmFyaWFibGUoNCk7DQogICAgICAgICAgICB2YXIgc2VjdXJpdHlEZXNjcmlwdG9yID0gdGhpcy5fbWFyc2hhbC5DcmVhdGVWYXJpYWJsZSg0KTsNCiAgICAgICAgICAgIHZhciBsYXN0V3JpdGVUaW1lID0gdGhpcy5fbWFyc2hhbC5DcmVhdGVWYXJpYWJsZSg4KTsNCg0KICAgICAgICAgICAgcmV0VmFsID0gdGhpcy5fQWR2QXBpLlJlZ1F1ZXJ5SW5mb0tleUEoaC5EZXJlZigpLCBhY2hDbGFzcywgYWNoQ2xhc3NTaXplLCAwLA0KICAgICAgICAgICAgICAgIG51bVN1YktleXMsIGxvbmdlc3RTdWJrZXlTaXplLCBsb25nZXN0Q2xhc3NTdHJpbmcsIG51bVZhbHVlcywNCiAgICAgICAgICAgICAgICBsb25nZXN0VmFsdWVOYW1lLCBsb25nZXN0VmFsdWVEYXRhLCBzZWN1cml0eURlc2NyaXB0b3IsIGxhc3RXcml0ZVRpbWUpOw0KICAgICAgICAgICAgaWYgKHJldFZhbC5WYWwgIT0gMCkgeyB0aHJvdyAoJ1JlZ1F1ZXJ5SW5mb0tleUEoKSByZXR1cm5lZCBlcnJvcjogJyArIHJldFZhbC5WYWwpOyB9DQogICAgICAgICAgICBmb3IodmFyIGkgPSAwOyBpIDwgbnVtU3ViS2V5cy50b0J1ZmZlcigpLnJlYWRVSW50MzJMRSgpOyArK2kpDQogICAgICAgICAgICB7DQogICAgICAgICAgICAgICAgbmFtZVNpemUudG9CdWZmZXIoKS53cml0ZVVJbnQzMkxFKDEwMjQpOw0KICAgICAgICAgICAgICAgIHJldFZhbCA9IHRoaXMuX0FkdkFwaS5SZWdFbnVtS2V5RXhBKGguRGVyZWYoKSwgaSwgYWNoS2V5LCBuYW1lU2l6ZSwgMCwgMCwgMCwgbGFzdFdyaXRlVGltZSk7DQogICAgICAgICAgICAgICAgaWYocmV0VmFsLlZhbCA9PSAwKQ0KICAgICAgICAgICAgICAgIHsNCiAgICAgICAgICAgICAgICAgICAgcmVzdWx0LnN1YmtleXMucHVzaChhY2hLZXkuU3RyaW5nKTsNCiAgICAgICAgICAgICAgICB9DQogICAgICAgICAgICB9DQogICAgICAgICAgICBmb3IgKHZhciBpID0gMDsgaSA8IG51bVZhbHVlcy50b0J1ZmZlcigpLnJlYWRVSW50MzJMRSgpIDsgKytpKQ0KICAgICAgICAgICAgew0KICAgICAgICAgICAgICAgIGFjaFZhbHVlU2l6ZS50b0J1ZmZlcigpLndyaXRlVUludDMyTEUoMzI3NjgpOw0KICAgICAgICAgICAgICAgIGlmKHRoaXMuX0FkdkFwaS5SZWdFbnVtVmFsdWVBKGguRGVyZWYoKSwgaSwgYWNoVmFsdWUsIGFjaFZhbHVlU2l6ZSwgMCwgMCwgMCwgMCkuVmFsID09IDApDQogICAgICAgICAgICAgICAgew0KICAgICAgICAgICAgICAgICAgICByZXN1bHQudmFsdWVzLnB1c2goYWNoVmFsdWUuU3RyaW5nKTsNCiAgICAgICAgICAgICAgICB9DQogICAgICAgICAgICB9DQogICAgICAgICAgICByZXR1cm4gKHJlc3VsdCk7DQogICAgICAgIH0KCiAgICAgICAgaWYodGhpcy5fQWR2QXBpLlJlZ1F1ZXJ5VmFsdWVFeEEoaC5EZXJlZigpLCBrZXksIDAsIDAsIDAsIGxlbikuVmFsID09IDApCiAgICAgICAgewogICAgICAgICAgICB2YXIgZGF0YSA9IHRoaXMuX21hcnNoYWwuQ3JlYXRlVmFyaWFibGUobGVuLnRvQnVmZmVyKCkucmVhZFVJbnQzMkxFKCkpOwogICAgICAgICAgICBpZiAodGhpcy5fQWR2QXBpLlJlZ1F1ZXJ5VmFsdWVFeEEoaC5EZXJlZigpLCBrZXksIDAsIHZhbFR5cGUsIGRhdGEsIGxlbikuVmFsID09IDApCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIHN3aXRjaCh2YWxUeXBlLnRvQnVmZmVyKCkucmVhZFVJbnQzMkxFKCkpCiAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgY2FzZSBLRVlfREFUQV9UWVBFUy5SRUdfRFdPUkQ6CiAgICAgICAgICAgICAgICAgICAgICAgIHJldFZhbCA9IGRhdGEudG9CdWZmZXIoKS5yZWFkVUludDMyTEUoKTsKICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgICAgICAgICAgICAgY2FzZSBLRVlfREFUQV9UWVBFUy5SRUdfRFdPUkRfQklHX0VORElBTjoKICAgICAgICAgICAgICAgICAgICAgICAgcmV0VmFsID0gZGF0YS50b0J1ZmZlcigpLnJlYWRVSW50MzJCRSgpOwogICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKICAgICAgICAgICAgICAgICAgICBjYXNlIEtFWV9EQVRBX1RZUEVTLlJFR19TWjoKICAgICAgICAgICAgICAgICAgICAgICAgcmV0VmFsID0gZGF0YS5TdHJpbmc7CiAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgICAgICAgICAgICAgIGNhc2UgS0VZX0RBVEFfVFlQRVMuUkVHX0JJTkFSWToKICAgICAgICAgICAgICAgICAgICBkZWZhdWx0OgogICAgICAgICAgICAgICAgICAgICAgICByZXRWYWwgPSBkYXRhLnRvQnVmZmVyKCk7CiAgICAgICAgICAgICAgICAgICAgICAgIHJldFZhbC5fZGF0YSA9IGRhdGE7CiAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICB9CiAgICAgICAgfQogICAgICAgIGVsc2UKICAgICAgICB7CiAgICAgICAgICAgIHRoaXMuX0FkdkFwaS5SZWdDbG9zZUtleShoLkRlcmVmKCkpOwogICAgICAgICAgICB0aHJvdyAoJ05vdCBGb3VuZCcpOwogICAgICAgIH0KICAgICAgICB0aGlzLl9BZHZBcGkuUmVnQ2xvc2VLZXkoaC5EZXJlZigpKTsKICAgICAgICByZXR1cm4gKHJldFZhbCk7CiAgICB9OwogICAgdGhpcy5Xcml0ZUtleSA9IGZ1bmN0aW9uIFdyaXRlS2V5KGhrZXksIHBhdGgsIGtleSwgdmFsdWUpCiAgICB7CiAgICAgICAgdmFyIHJlc3VsdDsKICAgICAgICB2YXIgaCA9IHRoaXMuX21hcnNoYWwuQ3JlYXRlUG9pbnRlcigpOwoKICAgICAgICBpZiAodGhpcy5fQWR2QXBpLlJlZ0NyZWF0ZUtleUV4QSh0aGlzLl9tYXJzaGFsLkNyZWF0ZVBvaW50ZXIoaGtleSksIHRoaXMuX21hcnNoYWwuQ3JlYXRlVmFyaWFibGUocGF0aCksIDAsIDAsIDAsIEtFWV9XUklURSwgMCwgaCwgMCkuVmFsICE9IDApCiAgICAgICAgewogICAgICAgICAgICB0aHJvdyAoJ0Vycm9yIE9wZW5pbmcgUmVnaXN0cnkgS2V5OiAnICsgcGF0aCk7CiAgICAgICAgfQoKICAgICAgICB2YXIgZGF0YTsKICAgICAgICB2YXIgZGF0YVR5cGU7CgogICAgICAgIHN3aXRjaCh0eXBlb2YodmFsdWUpKQogICAgICAgIHsKICAgICAgICAgICAgY2FzZSAnYm9vbGVhbic6CiAgICAgICAgICAgICAgICBkYXRhVHlwZSA9IEtFWV9EQVRBX1RZUEVTLlJFR19EV09SRDsKICAgICAgICAgICAgICAgIGRhdGEgPSB0aGlzLl9tYXJzaGFsLkNyZWF0ZVZhcmlhYmxlKDQpOwogICAgICAgICAgICAgICAgZGF0YS50b0J1ZmZlcigpLndyaXRlVUludDMyTEUodmFsdWUgPyAxIDogMCk7CiAgICAgICAgICAgICAgICBicmVhazsKICAgICAgICAgICAgY2FzZSAnbnVtYmVyJzoKICAgICAgICAgICAgICAgIGRhdGFUeXBlID0gS0VZX0RBVEFfVFlQRVMuUkVHX0RXT1JEOwogICAgICAgICAgICAgICAgZGF0YSA9IHRoaXMuX21hcnNoYWwuQ3JlYXRlVmFyaWFibGUoNCk7CiAgICAgICAgICAgICAgICBkYXRhLnRvQnVmZmVyKCkud3JpdGVVSW50MzJMRSh2YWx1ZSk7CiAgICAgICAgICAgICAgICBicmVhazsKICAgICAgICAgICAgY2FzZSAnc3RyaW5nJzoKICAgICAgICAgICAgICAgIGRhdGFUeXBlID0gS0VZX0RBVEFfVFlQRVMuUkVHX1NaOwogICAgICAgICAgICAgICAgZGF0YSA9IHRoaXMuX21hcnNoYWwuQ3JlYXRlVmFyaWFibGUodmFsdWUpOwogICAgICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgICAgIGRlZmF1bHQ6CiAgICAgICAgICAgICAgICBkYXRhVHlwZSA9IEtFWV9EQVRBX1RZUEVTLlJFR19CSU5BUlk7CiAgICAgICAgICAgICAgICBkYXRhID0gdGhpcy5fbWFyc2hhbC5DcmVhdGVWYXJpYWJsZSh2YWx1ZS5sZW5ndGgpOwogICAgICAgICAgICAgICAgdmFsdWUuY29weShkYXRhLnRvQnVmZmVyKCkpOwogICAgICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgfQoKICAgICAgICBpZih0aGlzLl9BZHZBcGkuUmVnU2V0VmFsdWVFeEEoaC5EZXJlZigpLCB0aGlzLl9tYXJzaGFsLkNyZWF0ZVZhcmlhYmxlKGtleSksIDAsIGRhdGFUeXBlLCBkYXRhLCBkYXRhLl9zaXplKS5WYWwgIT0gMCkKICAgICAgICB7ICAgICAgICAgICAKICAgICAgICAgICAgdGhpcy5fQWR2QXBpLlJlZ0Nsb3NlS2V5KGguRGVyZWYoKSk7CiAgICAgICAgICAgIHRocm93ICgnRXJyb3Igd3JpdGluZyByZWcga2V5OiAnICsga2V5KTsKICAgICAgICB9CiAgICAgICAgdGhpcy5fQWR2QXBpLlJlZ0Nsb3NlS2V5KGguRGVyZWYoKSk7CiAgICB9OwogICAgdGhpcy5EZWxldGVLZXkgPSBmdW5jdGlvbiBEZWxldGVLZXkoaGtleSwgcGF0aCwga2V5KQogICAgewogICAgICAgIGlmKCFrZXkpCiAgICAgICAgewogICAgICAgICAgICBpZih0aGlzLl9BZHZBcGkuUmVnRGVsZXRlS2V5QSh0aGlzLl9tYXJzaGFsLkNyZWF0ZVBvaW50ZXIoaGtleSksIHRoaXMuX21hcnNoYWwuQ3JlYXRlVmFyaWFibGUocGF0aCkpLlZhbCAhPSAwKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICB0aHJvdyAoJ0Vycm9yIERlbGV0aW5nIEtleTogJyArIHBhdGgpOwogICAgICAgICAgICB9CiAgICAgICAgfQogICAgICAgIGVsc2UKICAgICAgICB7CiAgICAgICAgICAgIHZhciBoID0gdGhpcy5fbWFyc2hhbC5DcmVhdGVQb2ludGVyKCk7CiAgICAgICAgICAgIHZhciByZXN1bHQ7CiAgICAgICAgICAgIGlmICh0aGlzLl9BZHZBcGkuUmVnT3BlbktleUV4QSh0aGlzLl9tYXJzaGFsLkNyZWF0ZVBvaW50ZXIoaGtleSksIHRoaXMuX21hcnNoYWwuQ3JlYXRlVmFyaWFibGUocGF0aCksIDAsIEtFWV9RVUVSWV9WQUxVRSB8IEtFWV9XUklURSwgaCkuVmFsICE9IDApCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIHRocm93ICgnRXJyb3IgT3BlbmluZyBSZWdpc3RyeSBLZXk6ICcgKyBwYXRoKTsKICAgICAgICAgICAgfQogICAgICAgICAgICBpZiAoKHJlc3VsdCA9IHRoaXMuX0FkdkFwaS5SZWdEZWxldGVWYWx1ZUEoaC5EZXJlZigpLCB0aGlzLl9tYXJzaGFsLkNyZWF0ZVZhcmlhYmxlKGtleSkpLlZhbCkgIT0gMCkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgdGhpcy5fQWR2QXBpLlJlZ0Nsb3NlS2V5KGguRGVyZWYoKSk7CiAgICAgICAgICAgICAgICB0aHJvdyAoJ0Vycm9yWycgKyByZXN1bHQgKyAnXSBEZWxldGluZyBLZXk6ICcgKyBwYXRoICsgJy4nICsga2V5KTsKICAgICAgICAgICAgfQogICAgICAgICAgICB0aGlzLl9BZHZBcGkuUmVnQ2xvc2VLZXkoaC5EZXJlZigpKTsKICAgICAgICB9CiAgICB9Owp9Cgptb2R1bGUuZXhwb3J0cyA9IG5ldyB3aW5kb3dzX3JlZ2lzdHJ5KCk7Cgo=', 'base64').toString());");
|
|
#endif
|
|
|
|
}
|
|
|
|
void ILibDuktape_ChainViewer_PostSelect(void* object, int slct, fd_set *readset, fd_set *writeset, fd_set *errorset)
|
|
{
|
|
duk_context *ctx = (duk_context*)((void**)((ILibTransport*)object)->ChainLink.ExtraMemoryPtr)[0];
|
|
void *hptr = ((void**)((ILibTransport*)object)->ChainLink.ExtraMemoryPtr)[1];
|
|
|
|
ILibDuktape_EventEmitter_SetupEmit(ctx, hptr, "PostSelect"); // [emit][this][name]
|
|
duk_push_int(ctx, slct); // [emit][this][name][select]
|
|
|
|
char *m = ILibChain_GetMetaDataFromDescriptorSet(Duktape_GetChain(ctx), readset, writeset, errorset);
|
|
duk_push_string(ctx, m);
|
|
if (duk_pcall_method(ctx, 3) != 0) { ILibDuktape_Process_UncaughtExceptionEx(ctx, "ChainViewer.emit('PostSelect'): Error "); }
|
|
duk_pop(ctx);
|
|
}
|
|
|
|
extern void ILibPrependToChain(void *Chain, void *object);
|
|
void ILibDuktape_ChainViewer_Push(duk_context *ctx, void *chain)
|
|
{
|
|
duk_push_object(ctx); // [viewer]
|
|
|
|
ILibTransport *t = (ILibTransport*)ILibChain_Link_Allocate(sizeof(ILibTransport), 2*sizeof(void*));
|
|
t->ChainLink.MetaData = "ILibDuktape_ChainViewer";
|
|
t->ChainLink.PostSelectHandler = ILibDuktape_ChainViewer_PostSelect;
|
|
((void**)t->ChainLink.ExtraMemoryPtr)[0] = ctx;
|
|
((void**)t->ChainLink.ExtraMemoryPtr)[1] = duk_get_heapptr(ctx, -1);
|
|
ILibDuktape_EventEmitter *emitter = ILibDuktape_EventEmitter_Create(ctx);
|
|
ILibDuktape_EventEmitter_CreateEventEx(emitter, "PostSelect");
|
|
ILibPrependToChain(chain, (void*)t);
|
|
}
|
|
|
|
duk_ret_t ILibDuktape_httpHeaders(duk_context *ctx)
|
|
{
|
|
ILibHTTPPacket *packet = NULL;
|
|
packetheader_field_node *node;
|
|
int headersOnly = duk_get_top(ctx) > 1 ? (duk_require_boolean(ctx, 1) ? 1 : 0) : 0;
|
|
|
|
duk_size_t bufferLen;
|
|
char *buffer = (char*)Duktape_GetBuffer(ctx, 0, &bufferLen);
|
|
|
|
packet = ILibParsePacketHeader(buffer, 0, (int)bufferLen);
|
|
if (packet == NULL) { return(ILibDuktape_Error(ctx, "http-headers(): Error parsing data")); }
|
|
|
|
if (headersOnly == 0)
|
|
{
|
|
duk_push_object(ctx);
|
|
if (packet->Directive != NULL)
|
|
{
|
|
duk_push_lstring(ctx, packet->Directive, packet->DirectiveLength);
|
|
duk_put_prop_string(ctx, -2, "method");
|
|
duk_push_lstring(ctx, packet->DirectiveObj, packet->DirectiveObjLength);
|
|
duk_put_prop_string(ctx, -2, "url");
|
|
}
|
|
else
|
|
{
|
|
duk_push_int(ctx, packet->StatusCode);
|
|
duk_put_prop_string(ctx, -2, "statusCode");
|
|
duk_push_lstring(ctx, packet->StatusData, packet->StatusDataLength);
|
|
duk_put_prop_string(ctx, -2, "statusMessage");
|
|
}
|
|
if (packet->VersionLength == 3)
|
|
{
|
|
duk_push_object(ctx);
|
|
duk_push_lstring(ctx, packet->Version, 1);
|
|
duk_put_prop_string(ctx, -2, "major");
|
|
duk_push_lstring(ctx, packet->Version + 2, 1);
|
|
duk_put_prop_string(ctx, -2, "minor");
|
|
duk_put_prop_string(ctx, -2, "version");
|
|
}
|
|
}
|
|
|
|
duk_push_object(ctx); // headers
|
|
node = packet->FirstField;
|
|
while (node != NULL)
|
|
{
|
|
duk_push_lstring(ctx, node->Field, node->FieldLength); // [str]
|
|
duk_get_prop_string(ctx, -1, "toLowerCase"); // [str][toLower]
|
|
duk_swap_top(ctx, -2); // [toLower][this]
|
|
duk_call_method(ctx, 0); // [result]
|
|
duk_push_lstring(ctx, node->FieldData, node->FieldDataLength);
|
|
duk_put_prop(ctx, -3);
|
|
node = node->NextField;
|
|
}
|
|
if (headersOnly == 0)
|
|
{
|
|
duk_put_prop_string(ctx, -2, "headers");
|
|
}
|
|
ILibDestructPacket(packet);
|
|
return(1);
|
|
}
|
|
void ILibDuktape_httpHeaders_PUSH(duk_context *ctx, void *chain)
|
|
{
|
|
duk_push_c_function(ctx, ILibDuktape_httpHeaders, DUK_VARARGS);
|
|
}
|
|
void ILibDuktape_DescriptorEvents_PreSelect(void* object, fd_set *readset, fd_set *writeset, fd_set *errorset, int* blocktime)
|
|
{
|
|
duk_context *ctx = (duk_context*)((void**)((ILibChain_Link*)object)->ExtraMemoryPtr)[0];
|
|
void *h = ((void**)((ILibChain_Link*)object)->ExtraMemoryPtr)[1];
|
|
int i = duk_get_top(ctx);
|
|
int fd;
|
|
|
|
duk_push_heapptr(ctx, h); // [obj]
|
|
duk_get_prop_string(ctx, -1, ILibDuktape_DescriptorEvents_Table); // [obj][table]
|
|
duk_enum(ctx, -1, DUK_ENUM_OWN_PROPERTIES_ONLY); // [obj][table][enum]
|
|
while (duk_next(ctx, -1, 1)) // [obj][table][enum][FD][emitter]
|
|
{
|
|
fd = (int)duk_to_int(ctx, -2);
|
|
duk_get_prop_string(ctx, -1, ILibDuktape_DescriptorEvents_Options); // [obj][table][enum][FD][emitter][options]
|
|
if (Duktape_GetBooleanProperty(ctx, -1, "readset", 0)) { FD_SET(fd, readset); }
|
|
if (Duktape_GetBooleanProperty(ctx, -1, "writeset", 0)) { FD_SET(fd, writeset); }
|
|
if (Duktape_GetBooleanProperty(ctx, -1, "errorset", 0)) { FD_SET(fd, errorset); }
|
|
duk_pop_3(ctx); // [obj][table][enum]
|
|
}
|
|
|
|
duk_set_top(ctx, i);
|
|
}
|
|
void ILibDuktape_DescriptorEvents_PostSelect(void* object, int slct, fd_set *readset, fd_set *writeset, fd_set *errorset)
|
|
{
|
|
duk_context *ctx = (duk_context*)((void**)((ILibChain_Link*)object)->ExtraMemoryPtr)[0];
|
|
void *h = ((void**)((ILibChain_Link*)object)->ExtraMemoryPtr)[1];
|
|
int i = duk_get_top(ctx);
|
|
int fd;
|
|
|
|
duk_push_array(ctx); // [array]
|
|
duk_push_heapptr(ctx, h); // [array][obj]
|
|
duk_get_prop_string(ctx, -1, ILibDuktape_DescriptorEvents_Table); // [array][obj][table]
|
|
duk_enum(ctx, -1, DUK_ENUM_OWN_PROPERTIES_ONLY); // [array][obj][table][enum]
|
|
while (duk_next(ctx, -1, 1)) // [array][obj][table][enum][FD][emitter]
|
|
{
|
|
fd = (int)duk_to_int(ctx, -2);
|
|
if (FD_ISSET(fd, readset) || FD_ISSET(fd, writeset) || FD_ISSET(fd, errorset))
|
|
{
|
|
duk_put_prop_index(ctx, -6, duk_get_length(ctx, -6)); // [array][obj][table][enum][FD]
|
|
duk_pop(ctx); // [array][obj][table][enum]
|
|
}
|
|
else
|
|
{
|
|
duk_pop_2(ctx); // [array][obj][table][enum]
|
|
|
|
}
|
|
}
|
|
duk_pop_3(ctx); // [array]
|
|
|
|
while (duk_get_length(ctx, -1) > 0)
|
|
{
|
|
duk_get_prop_string(ctx, -1, "pop"); // [array][pop]
|
|
duk_dup(ctx, -2); // [array][pop][this]
|
|
if (duk_pcall_method(ctx, 0) == 0) // [array][emitter]
|
|
{
|
|
if ((fd = Duktape_GetIntPropertyValue(ctx, -1, ILibDuktape_DescriptorEvents_FD, -1)) != -1)
|
|
{
|
|
if (FD_ISSET(fd, readset))
|
|
{
|
|
ILibDuktape_EventEmitter_SetupEmit(ctx, duk_get_heapptr(ctx, -1), "readset"); // [array][emitter][emit][this][readset]
|
|
duk_push_int(ctx, fd); // [array][emitter][emit][this][readset][fd]
|
|
duk_pcall_method(ctx, 2); duk_pop(ctx); // [array][emitter]
|
|
}
|
|
if (FD_ISSET(fd, writeset))
|
|
{
|
|
ILibDuktape_EventEmitter_SetupEmit(ctx, duk_get_heapptr(ctx, -1), "writeset"); // [array][emitter][emit][this][writeset]
|
|
duk_push_int(ctx, fd); // [array][emitter][emit][this][writeset][fd]
|
|
duk_pcall_method(ctx, 2); duk_pop(ctx); // [array][emitter]
|
|
}
|
|
if (FD_ISSET(fd, errorset))
|
|
{
|
|
ILibDuktape_EventEmitter_SetupEmit(ctx, duk_get_heapptr(ctx, -1), "errorset"); // [array][emitter][emit][this][errorset]
|
|
duk_push_int(ctx, fd); // [array][emitter][emit][this][errorset][fd]
|
|
duk_pcall_method(ctx, 2); duk_pop(ctx); // [array][emitter]
|
|
}
|
|
}
|
|
}
|
|
duk_pop(ctx); // [array]
|
|
}
|
|
duk_set_top(ctx, i);
|
|
}
|
|
duk_ret_t ILibDuktape_DescriptorEvents_Remove(duk_context *ctx)
|
|
{
|
|
if (!duk_is_number(ctx, 0)) { return(ILibDuktape_Error(ctx, "Invalid Descriptor")); }
|
|
ILibForceUnBlockChain(Duktape_GetChain(ctx));
|
|
|
|
duk_push_this(ctx); // [obj]
|
|
duk_get_prop_string(ctx, -1, ILibDuktape_DescriptorEvents_Table); // [obj][table]
|
|
duk_dup(ctx, 0); // [obj][table][key]
|
|
duk_del_prop(ctx, -2); // [obj][table]
|
|
return(0);
|
|
}
|
|
duk_ret_t ILibDuktape_DescriptorEvents_Add(duk_context *ctx)
|
|
{
|
|
if (!duk_is_number(ctx, 0)) { return(ILibDuktape_Error(ctx, "Invalid Descriptor")); }
|
|
ILibForceUnBlockChain(Duktape_GetChain(ctx));
|
|
|
|
duk_push_this(ctx); // [obj]
|
|
duk_get_prop_string(ctx, -1, ILibDuktape_DescriptorEvents_Table); // [obj][table]
|
|
duk_push_object(ctx); // [obj][table][value]
|
|
duk_dup(ctx, 0); // [obj][table][value][key]
|
|
duk_dup(ctx, -2); // [obj][table][value][key][value]
|
|
ILibDuktape_EventEmitter *e = ILibDuktape_EventEmitter_Create(ctx);
|
|
ILibDuktape_EventEmitter_CreateEventEx(e, "readset");
|
|
ILibDuktape_EventEmitter_CreateEventEx(e, "writeset");
|
|
ILibDuktape_EventEmitter_CreateEventEx(e, "errorset");
|
|
duk_dup(ctx, 0); // [obj][table][value][key][value][FD]
|
|
duk_put_prop_string(ctx, -2, ILibDuktape_DescriptorEvents_FD); // [obj][table][value][key][value]
|
|
duk_dup(ctx, 1); duk_put_prop_string(ctx, -2, ILibDuktape_DescriptorEvents_Options);
|
|
|
|
duk_put_prop(ctx, -4); // [obj][table][value]
|
|
|
|
return(1);
|
|
}
|
|
void ILibDuktape_DescriptorEvents_Push(duk_context *ctx, void *chain)
|
|
{
|
|
ILibChain_Link *link = (ILibChain_Link*)ILibChain_Link_Allocate(sizeof(ILibChain_Link), 2 * sizeof(void*));
|
|
link->MetaData = "ILibDuktape_DescriptorEvents";
|
|
link->PreSelectHandler = ILibDuktape_DescriptorEvents_PreSelect;
|
|
link->PostSelectHandler = ILibDuktape_DescriptorEvents_PostSelect;
|
|
|
|
duk_push_object(ctx);
|
|
duk_push_object(ctx); duk_put_prop_string(ctx, -2, ILibDuktape_DescriptorEvents_Table);
|
|
|
|
((void**)link->ExtraMemoryPtr)[0] = ctx;
|
|
((void**)link->ExtraMemoryPtr)[1] = duk_get_heapptr(ctx, -1);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "addDescriptor", ILibDuktape_DescriptorEvents_Add, 2);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "removeDescriptor", ILibDuktape_DescriptorEvents_Remove, 1);
|
|
|
|
|
|
ILibAddToChain(chain, link);
|
|
}
|
|
void ILibDuktape_Polyfills_Init(duk_context *ctx)
|
|
{
|
|
ILibDuktape_ModSearch_AddHandler(ctx, "queue", ILibDuktape_Queue_Push);
|
|
ILibDuktape_ModSearch_AddHandler(ctx, "DynamicBuffer", ILibDuktape_DynamicBuffer_Push);
|
|
ILibDuktape_ModSearch_AddHandler(ctx, "stream", ILibDuktape_Stream_Init);
|
|
ILibDuktape_ModSearch_AddHandler(ctx, "http-headers", ILibDuktape_httpHeaders_PUSH);
|
|
|
|
#ifndef MICROSTACK_NOTLS
|
|
ILibDuktape_ModSearch_AddHandler(ctx, "pkcs7", ILibDuktape_PKCS7_Push);
|
|
#endif
|
|
|
|
#ifndef MICROSTACK_NOTLS
|
|
ILibDuktape_ModSearch_AddHandler(ctx, "bignum", ILibDuktape_bignum_Push);
|
|
ILibDuktape_ModSearch_AddHandler(ctx, "dataGenerator", ILibDuktape_dataGenerator_Push);
|
|
#endif
|
|
ILibDuktape_ModSearch_AddHandler(ctx, "ChainViewer", ILibDuktape_ChainViewer_Push);
|
|
ILibDuktape_ModSearch_AddHandler(ctx, "DescriptorEvents", ILibDuktape_DescriptorEvents_Push);
|
|
|
|
// Global Polyfills
|
|
duk_push_global_object(ctx); // [g]
|
|
|
|
ILibDuktape_Polyfills_String(ctx);
|
|
ILibDuktape_Polyfills_Buffer(ctx);
|
|
ILibDuktape_Polyfills_Console(ctx);
|
|
ILibDuktape_Polyfills_byte_ordering(ctx);
|
|
ILibDuktape_Polyfills_timer(ctx);
|
|
ILibDuktape_Polyfills_object(ctx);
|
|
|
|
ILibDuktape_CreateInstanceMethod(ctx, "addModuleObject", ILibDuktape_Polyfills_addModuleObject, 2);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "addModule", ILibDuktape_Polyfills_addModule, 2);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "getJSModule", ILibDuktape_Polyfills_getJSModule, 1);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "_debugCrash", ILibDuktape_Polyfills_debugCrash, 0);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "_debugGC", ILibDuktape_Polyfills_debugGC, 0);
|
|
ILibDuktape_CreateInstanceMethod(ctx, "_debug", ILibDuktape_Polyfills_debug, 0);
|
|
#ifndef MICROSTACK_NOTLS
|
|
ILibDuktape_CreateInstanceMethod(ctx, "crc32c", ILibDuktape_Polyfills_crc32c, DUK_VARARGS);
|
|
#endif
|
|
duk_pop(ctx); // ...
|
|
|
|
ILibDuktape_Debugger_Init(ctx, 9091);
|
|
}
|
|
|
|
#ifdef __DOXY__
|
|
/*!
|
|
\brief String
|
|
*/
|
|
class String
|
|
{
|
|
public:
|
|
/*!
|
|
\brief Finds a String within another String
|
|
\param str \<String\> Substring to search for
|
|
\return <Integer> Index of where the string was found. -1 if not found
|
|
*/
|
|
Integer indexOf(str);
|
|
/*!
|
|
\brief Extracts a String from a String.
|
|
\param startIndex <Integer> Starting index to extract
|
|
\param length <Integer> Number of characters to extract
|
|
\return \<String\> extracted String
|
|
*/
|
|
String substr(startIndex, length);
|
|
/*!
|
|
\brief Extracts a String from a String.
|
|
\param startIndex <Integer> Starting index to extract
|
|
\param endIndex <Integer> Ending index to extract
|
|
\return \<String\> extracted String
|
|
*/
|
|
String splice(startIndex, endIndex);
|
|
/*!
|
|
\brief Split String into substrings
|
|
\param str \<String\> Delimiter to split on
|
|
\return Array of Tokens
|
|
*/
|
|
Array<String> split(str);
|
|
/*!
|
|
\brief Determines if a String starts with the given substring
|
|
\param str \<String\> substring
|
|
\return <boolean> True, if this String starts with the given substring
|
|
*/
|
|
boolean startsWith(str);
|
|
};
|
|
/*!
|
|
\brief Instances of the Buffer class are similar to arrays of integers but correspond to fixed-sized, raw memory allocations.
|
|
*/
|
|
class Buffer
|
|
{
|
|
public:
|
|
/*!
|
|
\brief Create a new Buffer instance of the specified number of bytes
|
|
\param size <integer>
|
|
\return \<Buffer\> new Buffer instance
|
|
*/
|
|
Buffer(size);
|
|
|
|
/*!
|
|
\brief Returns the amount of memory allocated in bytes
|
|
*/
|
|
integer length;
|
|
/*!
|
|
\brief Creates a new Buffer instance from an encoded String
|
|
\param str \<String\> encoded String
|
|
\param encoding \<String\> Encoding. Can be either 'base64' or 'hex'
|
|
\return \<Buffer\> new Buffer instance
|
|
*/
|
|
static Buffer from(str, encoding);
|
|
/*!
|
|
\brief Decodes Buffer to a String
|
|
\param encoding \<String\> Optional. Can be either 'base64' or 'hex'. If not specified, will just encode as an ANSI string
|
|
\param start <integer> Optional. Starting offset. <b>Default:</b> 0
|
|
\param end <integer> Optional. Ending offset (not inclusive) <b>Default:</b> buffer length
|
|
\return \<String\> Encoded String
|
|
*/
|
|
String toString([encoding[, start[, end]]]);
|
|
/*!
|
|
\brief Returns a new Buffer that references the same memory as the original, but offset and cropped by the start and end indices.
|
|
\param start <integer> Where the new Buffer will start. <b>Default:</b> 0
|
|
\param end <integer> Where the new Buffer will end. (Not inclusive) <b>Default:</b> buffer length
|
|
\return \<Buffer\>
|
|
*/
|
|
Buffer slice([start[, end]]);
|
|
};
|
|
/*!
|
|
\brief Console
|
|
*/
|
|
class Console
|
|
{
|
|
public:
|
|
/*!
|
|
\brief Serializes the input parameters to the Console Display
|
|
\param args <any>
|
|
*/
|
|
void log(...args);
|
|
};
|
|
/*!
|
|
\brief Global Timer Methods
|
|
*/
|
|
class Timers
|
|
{
|
|
public:
|
|
/*!
|
|
\brief Schedules the "immediate" execution of the callback after I/O events' callbacks.
|
|
\param callback <func> Function to call at the end of the event loop
|
|
\param args <any> Optional arguments to pass when the callback is called
|
|
\return Immediate for use with clearImmediate().
|
|
*/
|
|
Immediate setImmediate(callback[, ...args]);
|
|
/*!
|
|
\brief Schedules execution of a one-time callback after delay milliseconds.
|
|
\param callback <func> Function to call when the timeout elapses
|
|
\param args <any> Optional arguments to pass when the callback is called
|
|
\return Timeout for use with clearTimeout().
|
|
*/
|
|
Timeout setTimeout(callback, delay[, ...args]);
|
|
/*!
|
|
\brief Schedules repeated execution of callback every delay milliseconds.
|
|
\param callback <func> Function to call when the timer elapses
|
|
\param args <any> Optional arguments to pass when the callback is called
|
|
\return Timeout for use with clearInterval().
|
|
*/
|
|
Timeout setInterval(callback, delay[, ...args]);
|
|
|
|
/*!
|
|
\brief Cancels a Timeout returned by setTimeout()
|
|
\param timeout Timeout
|
|
*/
|
|
void clearTimeout(timeout);
|
|
/*!
|
|
\brief Cancels a Timeout returned by setInterval()
|
|
\param interval Timeout
|
|
*/
|
|
void clearInterval(interval);
|
|
/*!
|
|
\brief Cancels an Immediate returned by setImmediate()
|
|
\param immediate Immediate
|
|
*/
|
|
void clearImmediate(immediate);
|
|
|
|
/*!
|
|
\brief Scheduled Timer
|
|
*/
|
|
class Timeout
|
|
{
|
|
public:
|
|
};
|
|
/*!
|
|
\implements Timeout
|
|
\brief Scheduled Immediate
|
|
*/
|
|
class Immediate
|
|
{
|
|
public:
|
|
};
|
|
};
|
|
|
|
/*!
|
|
\brief Global methods for byte ordering manipulation
|
|
*/
|
|
class BytesOrdering
|
|
{
|
|
public:
|
|
/*!
|
|
\brief Converts 2 bytes from network order to host order
|
|
\param buffer \<Buffer\> bytes to convert
|
|
\param offset <integer> offset to start
|
|
\return <integer> host order value
|
|
*/
|
|
static integer ntohs(buffer, offset);
|
|
/*!
|
|
\brief Converts 4 bytes from network order to host order
|
|
\param buffer \<Buffer\> bytes to convert
|
|
\param offset <integer> offset to start
|
|
\return <integer> host order value
|
|
*/
|
|
static integer ntohl(buffer, offset);
|
|
/*!
|
|
\brief Writes 2 bytes in network order
|
|
\param buffer \<Buffer\> Buffer to write to
|
|
\param offset <integer> offset to start writing
|
|
\param val <integer> host order value to write
|
|
*/
|
|
static void htons(buffer, offset, val);
|
|
/*!
|
|
\brief Writes 4 bytes in network order
|
|
\param buffer \<Buffer\> Buffer to write to
|
|
\param offset <integer> offset to start writing
|
|
\param val <integer> host order value to write
|
|
*/
|
|
static void htonl(buffer, offset, val);
|
|
};
|
|
#endif
|