295 lines
9.2 KiB
JavaScript
295 lines
9.2 KiB
JavaScript
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Copyright 2016 Realm Inc.
|
|
//
|
|
// 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.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
'use strict';
|
|
|
|
import * as base64 from './base64';
|
|
import { keys, objectTypes } from './constants';
|
|
|
|
const { id: idKey, realm: _realmKey } = keys;
|
|
let registeredCallbacks = [];
|
|
const typeConverters = {};
|
|
|
|
// Callbacks that are registered initially (currently only refreshAccessToken) will
|
|
// carry this symbol so they are not wiped in clearTestState.
|
|
const persistentCallback = Symbol("persistentCallback");
|
|
|
|
let XMLHttpRequest = global.originalXMLHttpRequest || global.XMLHttpRequest;
|
|
let sessionHost;
|
|
let sessionId;
|
|
|
|
// Check if XMLHttpRequest has been overridden, and get the native one if that's the case.
|
|
if (XMLHttpRequest.__proto__ != global.XMLHttpRequestEventTarget) {
|
|
let fakeXMLHttpRequest = XMLHttpRequest;
|
|
delete global.XMLHttpRequest;
|
|
XMLHttpRequest = global.XMLHttpRequest;
|
|
global.XMLHttpRequest = fakeXMLHttpRequest;
|
|
}
|
|
|
|
registerTypeConverter(objectTypes.DATA, (_, { value }) => base64.decode(value));
|
|
registerTypeConverter(objectTypes.DATE, (_, { value }) => new Date(value));
|
|
registerTypeConverter(objectTypes.DICT, deserializeDict);
|
|
registerTypeConverter(objectTypes.FUNCTION, deserializeFunction);
|
|
|
|
export function registerTypeConverter(type, handler) {
|
|
typeConverters[type] = handler;
|
|
}
|
|
|
|
export function createSession(refreshAccessToken, host) {
|
|
refreshAccessToken[persistentCallback] = true;
|
|
sessionId = sendRequest('create_session', { refreshAccessToken: serialize(undefined, refreshAccessToken) }, host);
|
|
sessionHost = host;
|
|
return sessionId;
|
|
}
|
|
|
|
export function createRealm(args) {
|
|
if (args) {
|
|
args = args.map((arg) => serialize(null, arg));
|
|
}
|
|
|
|
return sendRequest('create_realm', { arguments: args });
|
|
}
|
|
|
|
export function createUser(args) {
|
|
args = args.map((arg) => serialize(null, arg));
|
|
const result = sendRequest('create_user', { arguments: args });
|
|
return deserialize(undefined, result);
|
|
}
|
|
|
|
export function _adminUser(args) {
|
|
args = args.map((arg) => serialize(null, arg));
|
|
const result = sendRequest('_adminUser', { arguments: args });
|
|
return deserialize(undefined, result);
|
|
}
|
|
|
|
export function callMethod(realmId, id, name, args) {
|
|
if (args) {
|
|
args = args.map((arg) => serialize(realmId, arg));
|
|
}
|
|
|
|
let result = sendRequest('call_method', { realmId, id, name, arguments: args });
|
|
return deserialize(realmId, result);
|
|
}
|
|
|
|
export function getProperty(realmId, id, name) {
|
|
let result = sendRequest('get_property', { realmId, id, name });
|
|
return deserialize(realmId, result);
|
|
}
|
|
|
|
export function setProperty(realmId, id, name, value) {
|
|
value = serialize(realmId, value);
|
|
sendRequest('set_property', { realmId, id, name, value });
|
|
}
|
|
|
|
export function getAllUsers() {
|
|
let result = sendRequest('get_all_users');
|
|
return deserialize(undefined, result);
|
|
}
|
|
|
|
export function clearTestState() {
|
|
sendRequest('clear_test_state');
|
|
|
|
// Clear all registered callbacks that are specific to this session.
|
|
registeredCallbacks = registeredCallbacks.filter(cb => Reflect.has(cb, persistentCallback));
|
|
}
|
|
|
|
function registerCallback(callback) {
|
|
let key = registeredCallbacks.indexOf(callback);
|
|
return key >= 0 ? key : (registeredCallbacks.push(callback) - 1);
|
|
}
|
|
|
|
function serialize(realmId, value) {
|
|
if (typeof value == 'undefined') {
|
|
return { type: objectTypes.UNDEFINED };
|
|
}
|
|
if (typeof value == 'function') {
|
|
return { type: objectTypes.FUNCTION, value: registerCallback(value) };
|
|
}
|
|
if (!value || typeof value != 'object') {
|
|
return { value: value };
|
|
}
|
|
|
|
let id = value[idKey];
|
|
if (id) {
|
|
return { id };
|
|
}
|
|
|
|
if (value instanceof Date) {
|
|
return { type: objectTypes.DATE, value: value.getTime() };
|
|
}
|
|
|
|
if (Array.isArray(value)) {
|
|
let array = value.map((item) => serialize(realmId, item));
|
|
return { value: array };
|
|
}
|
|
|
|
if (value instanceof ArrayBuffer || ArrayBuffer.isView(value)) {
|
|
return { type: objectTypes.DATA, value: base64.encode(value) };
|
|
}
|
|
|
|
let keys = Object.keys(value);
|
|
let values = keys.map((key) => serialize(realmId, value[key]));
|
|
return { type: objectTypes.DICT, keys, values };
|
|
}
|
|
|
|
export function deserialize(realmId, info) {
|
|
let type = info.type;
|
|
let handler = type && typeConverters[type];
|
|
if (handler) {
|
|
return handler(realmId, info);
|
|
}
|
|
|
|
let value = info.value;
|
|
if (value && Array.isArray(value)) {
|
|
return value.map((item) => deserialize(realmId, item));
|
|
}
|
|
|
|
return value;
|
|
}
|
|
|
|
function deserializeDict(realmId, info) {
|
|
let { keys, values } = info;
|
|
let object = {};
|
|
|
|
for (let i = 0, len = keys.length; i < len; i++) {
|
|
object[keys[i]] = deserialize(realmId, values[i]);
|
|
}
|
|
|
|
return object;
|
|
}
|
|
|
|
function deserializeFunction(realmId, info) {
|
|
return registeredCallbacks[info.value];
|
|
}
|
|
|
|
function makeRequest(url, data) {
|
|
let statusCode;
|
|
let responseText;
|
|
|
|
// The global __debug__ object is provided by Visual Studio Code.
|
|
if (global.__debug__) {
|
|
let request = global.__debug__.require('sync-request');
|
|
let response = request('POST', url, {
|
|
body: JSON.stringify(data),
|
|
headers: {
|
|
"Content-Type": "text/plain;charset=UTF-8"
|
|
}
|
|
});
|
|
|
|
statusCode = response.statusCode;
|
|
responseText = response.body.toString('utf-8');
|
|
} else {
|
|
let body = JSON.stringify(data);
|
|
let request = new XMLHttpRequest();
|
|
|
|
request.open('POST', url, false);
|
|
request.send(body);
|
|
|
|
statusCode = request.status;
|
|
responseText = request.responseText;
|
|
}
|
|
|
|
if (statusCode != 200) {
|
|
throw new Error(responseText);
|
|
}
|
|
|
|
return JSON.parse(responseText);
|
|
}
|
|
|
|
let pollTimeoutId;
|
|
|
|
//returns an object from rpc serialized json value
|
|
function deserialize_json_value(value) {
|
|
let result = {};
|
|
for (let index = 0; index < value.keys.length; index++) {
|
|
var propName = value.keys[index];
|
|
var propValue = value.values[index];
|
|
if (propValue.type && propValue.type == 'dict') {
|
|
result[propName] = deserialize_json_value(propValue);
|
|
}
|
|
else {
|
|
result[propName] = propValue.value;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
function sendRequest(command, data, host = sessionHost) {
|
|
clearTimeout(pollTimeoutId);
|
|
try {
|
|
if (!host) {
|
|
throw new Error('Must first create RPC session with a valid host');
|
|
}
|
|
|
|
data = Object.assign({}, data, sessionId ? { sessionId } : null);
|
|
|
|
let url = 'http://' + host + '/' + command;
|
|
let response = makeRequest(url, data);
|
|
|
|
if (!response || response.error) {
|
|
let error = response && response.error;
|
|
|
|
// Remove the type prefix from the error message (e.g. "Error: ").
|
|
if (error && error.replace) {
|
|
error = error.replace(/^[a-z]+: /i, '');
|
|
}
|
|
else if (error.type && error.type === 'dict') {
|
|
const responseError = deserialize_json_value(error);
|
|
let responeMessage;
|
|
if (response.message && response.message !== '') {
|
|
// Remove the type prefix from the error message (e.g. "Error: ").
|
|
responeMessage = response.message.replace(/^[a-z]+: /i, '');
|
|
}
|
|
|
|
const exceptionToReport = new Error(responeMessage);
|
|
Object.assign(exceptionToReport, responseError);
|
|
throw exceptionToReport;
|
|
}
|
|
|
|
throw new Error(error || `Invalid response for "${command}"`);
|
|
}
|
|
let callback = response.callback;
|
|
if (callback != null) {
|
|
let result;
|
|
let error;
|
|
try {
|
|
let realmId = data.realmId;
|
|
let thisObject = deserialize(realmId, response.this);
|
|
let args = deserialize(realmId, response.arguments);
|
|
result = registeredCallbacks[callback].apply(thisObject, args);
|
|
result = serialize(realmId, result);
|
|
} catch (e) {
|
|
error = e.message || ('' + e);
|
|
}
|
|
|
|
let callbackCommand = "callback_result";
|
|
if (command == 'callbacks_poll') {
|
|
callbackCommand = "callback_poll_result";
|
|
}
|
|
|
|
return sendRequest(callbackCommand, { callback, result, error, "callback_call_counter": response.callback_call_counter });
|
|
}
|
|
|
|
return response.result;
|
|
}
|
|
finally {
|
|
pollTimeoutId = setTimeout(() => sendRequest('callbacks_poll'), 100);
|
|
}
|
|
}
|