@haventech/semotus
Advanced tools
Comparing version 4.0.0 to 4.1.0
@@ -108,4 +108,5 @@ "use strict"; | ||
var remoteValidator = defineProperty.serverValidation; | ||
var isPublic = !!defineProperty.public; | ||
return function (target, propertyName, descriptor) { | ||
descriptor.value = objectTemplate._setupFunction(propertyName, descriptor.value, defineProperty.on, defineProperty.validate, remoteValidator, defineProperty.target); | ||
descriptor.value = objectTemplate._setupFunction(propertyName, descriptor.value, defineProperty.on, defineProperty.validate, remoteValidator, isPublic, defineProperty.target); | ||
/* | ||
@@ -139,2 +140,2 @@ this function been marked as a server API either explicitly or by default. | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorators.js","sourceRoot":"","sources":["../src/decorators.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Attaches syncStates props to the template\n * If a class is decorated like @supertypeClass({syncStates: ['internal', 'home']}), any instances of that class\n * including those that exist at the beginning of the application, should only be sent over when the controller state on\n * the server indicates that the correct syncState (if and only if controller.syncState = 'home' or 'internal', then an\n * object of this class will be sent, otherwise semotus will refrain from sending it over). The caveat, however, is if\n * semotus' controller's syncState is defined as '*' then we will by default send all templates unless specified other-\n * wise (the same as treating syncState: ['*'])\n *\n */\n\n\n/**\n * @TODO: This should be consolidated with Supertype's version when we finish converting to typescript\n */\n\n// Context Switch Chain:\n// 1st pass: Semotus (sClass) -> Supertype (sClass) -> Semotus (return Ret)\n// 2nd pass: Semotus (sClass) -> Semotus (decorator) -> Supertype (eval ret) -> Semotus (return)\nexport function supertypeClass(objectTemplate, SupertypeModule, target): any {\n    let ret;\n    let ObjectTemplate = SupertypeModule.default;\n    let syncStates: String[] = [];  // Default syncStates to empty array\n\n    // Decorator workerbee\n    const decorator = function decorator(target) {\n        // second time we must call the function returned the first time because it has the\n        // properties as a closure\n        if (ret) {\n            ret = ret(target, objectTemplate)\n        } else {\n            ret = SupertypeModule.supertypeClass(target, objectTemplate);\n        }\n\n        // Mainly for persistor properties to make sure they get transported\n        target.createProperty = function (propertyName, defineProperty) {\n            if (defineProperty.body) {\n                target.prototype[propertyName] = objectTemplate._setupFunction(\n                    propertyName,\n                    defineProperty.body,\n                    defineProperty.on,\n                    defineProperty.validate\n                );\n            } else {\n                target.prototype.__amorphicprops__[propertyName] = defineProperty;\n                const value = defineProperty.value;\n\n                // The getter actually initializes the property\n                defineProperty.get = function () {\n                    let nameSpacedProperty = `__${propertyName}`;\n\n                    if (!this[nameSpacedProperty]) {\n                        let cloneTemplate = defineProperty.of || defineProperty.type || null;\n                        this[nameSpacedProperty] = ObjectTemplate.clone(value, cloneTemplate);\n                    }\n                    return this[nameSpacedProperty];\n                };\n\n                const defineProperties = {};\n                objectTemplate._setupProperty(propertyName, defineProperty, undefined, defineProperties);\n                Object.defineProperties(target.prototype, defineProperties);\n            }\n        };\n\n        // Assigning sync states from prop definition onto the template\n        if (syncStates) {\n            target.syncStates = syncStates;\n        }\n\n        return ret;\n    };\n\n    // Calling the decorator processor with the actual Template (2nd time)\n    if (target.prototype) {\n        return decorator(target);\n    } else if (target && target.syncStates && target.syncStates instanceof Array) {\n        syncStates = target.syncStates;\n    } else if (target && target.syncStates ) {\n        console.warn(`Semotus warning: ${target.syncStates} is not an array value, so not setting up SyncStates for this target: ${target}`);\n    }\n\n    // Called first time with parameters (prop definition) rather than target (template)\n    // call supertypes supertypeClass function which will return a function that must be\n    // called on the 2nd pass when we have a target. This function's closure will also have a ref to the original properties\n    ret = SupertypeModule.supertypeClass(target, objectTemplate);\n    return decorator; // decorator will be called 2nd time with ret as a closure\n}\n\nexport function Supertype(template, objectTemplate, Supertype) {\n    return Supertype.call(template, objectTemplate);\n}\n\nexport function property(objectTemplate, SupertypeModule, props, toClientRuleSet, toServerRuleSet) {\n    props = props || {};\n    props.toClient = applyRuleSet(props.toClient, toClientRuleSet);\n    props.toServer = applyRuleSet(props.toServer, toServerRuleSet);\n    const baseDecorator = SupertypeModule.property(props, objectTemplate);\n    return function (target, targetKey) {\n        baseDecorator(target, targetKey);\n        const defineProperties = {};\n        props.enumerable = true;\n        props.writable = true;\n        objectTemplate._setupProperty(targetKey, props, undefined, defineProperties);\n        Object.defineProperties(target, defineProperties);\n    };\n}\n\n\nexport function remote(objectTemplate, defineProperty) {\n    defineProperty = defineProperty || {};\n\n    /*\n        if we haven't supplied a configuration object into the decorator,\n         default the role of this function to a server API function\n     */\n    if (!defineProperty.on) {\n        defineProperty.on = 'server';\n    }\n\n    // function that we call to validate any changes for remote calls\n    let remoteValidator = defineProperty.serverValidation;\n\n    return function (target, propertyName, descriptor) {\n        descriptor.value = objectTemplate._setupFunction(\n            propertyName,\n            descriptor.value,\n            defineProperty.on,\n            defineProperty.validate,\n            remoteValidator,\n            defineProperty.target\n        );\n\n        /*\n            this function been marked as a server API either explicitly or by default.\n            set the appropriate metadata.\n         */\n        if (defineProperty.on === 'server') {\n            descriptor.value.__on__ = 'server';\n        }\n\n        if (defineProperty.type) {\n            descriptor.value.__returns__ = defineProperty.type;\n        }\n        if (defineProperty.of) {\n            descriptor.value.__returns__ = defineProperty.of;\n            descriptor.value.__returnsarray__ = true;\n        }\n    };\n};\n\n\nfunction applyRuleSet(prop, ruleSet) {\n    if (prop instanceof Array && ruleSet instanceof Array && ruleSet.length > 0) {\n        return prop.some(function (r) {\n            return ruleSet.indexOf(r) >= 0;\n        });\n    } else {\n        return prop;\n    }\n}"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorators.js","sourceRoot":"","sources":["../src/decorators.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Attaches syncStates props to the template\n * If a class is decorated like @supertypeClass({syncStates: ['internal', 'home']}), any instances of that class\n * including those that exist at the beginning of the application, should only be sent over when the controller state on\n * the server indicates that the correct syncState (if and only if controller.syncState = 'home' or 'internal', then an\n * object of this class will be sent, otherwise semotus will refrain from sending it over). The caveat, however, is if\n * semotus' controller's syncState is defined as '*' then we will by default send all templates unless specified other-\n * wise (the same as treating syncState: ['*'])\n *\n */\n\n\n/**\n * @TODO: This should be consolidated with Supertype's version when we finish converting to typescript\n */\n\n// Context Switch Chain:\n// 1st pass: Semotus (sClass) -> Supertype (sClass) -> Semotus (return Ret)\n// 2nd pass: Semotus (sClass) -> Semotus (decorator) -> Supertype (eval ret) -> Semotus (return)\nexport function supertypeClass(objectTemplate, SupertypeModule, target): any {\n    let ret;\n    let ObjectTemplate = SupertypeModule.default;\n    let syncStates: String[] = [];  // Default syncStates to empty array\n\n    // Decorator workerbee\n    const decorator = function decorator(target) {\n        // second time we must call the function returned the first time because it has the\n        // properties as a closure\n        if (ret) {\n            ret = ret(target, objectTemplate)\n        } else {\n            ret = SupertypeModule.supertypeClass(target, objectTemplate);\n        }\n\n        // Mainly for persistor properties to make sure they get transported\n        target.createProperty = function (propertyName, defineProperty) {\n            if (defineProperty.body) {\n                target.prototype[propertyName] = objectTemplate._setupFunction(\n                    propertyName,\n                    defineProperty.body,\n                    defineProperty.on,\n                    defineProperty.validate\n                );\n            } else {\n                target.prototype.__amorphicprops__[propertyName] = defineProperty;\n                const value = defineProperty.value;\n\n                // The getter actually initializes the property\n                defineProperty.get = function () {\n                    let nameSpacedProperty = `__${propertyName}`;\n\n                    if (!this[nameSpacedProperty]) {\n                        let cloneTemplate = defineProperty.of || defineProperty.type || null;\n                        this[nameSpacedProperty] = ObjectTemplate.clone(value, cloneTemplate);\n                    }\n                    return this[nameSpacedProperty];\n                };\n\n                const defineProperties = {};\n                objectTemplate._setupProperty(propertyName, defineProperty, undefined, defineProperties);\n                Object.defineProperties(target.prototype, defineProperties);\n            }\n        };\n\n        // Assigning sync states from prop definition onto the template\n        if (syncStates) {\n            target.syncStates = syncStates;\n        }\n\n        return ret;\n    };\n\n    // Calling the decorator processor with the actual Template (2nd time)\n    if (target.prototype) {\n        return decorator(target);\n    } else if (target && target.syncStates && target.syncStates instanceof Array) {\n        syncStates = target.syncStates;\n    } else if (target && target.syncStates ) {\n        console.warn(`Semotus warning: ${target.syncStates} is not an array value, so not setting up SyncStates for this target: ${target}`);\n    }\n\n    // Called first time with parameters (prop definition) rather than target (template)\n    // call supertypes supertypeClass function which will return a function that must be\n    // called on the 2nd pass when we have a target. This function's closure will also have a ref to the original properties\n    ret = SupertypeModule.supertypeClass(target, objectTemplate);\n    return decorator; // decorator will be called 2nd time with ret as a closure\n}\n\nexport function Supertype(template, objectTemplate, Supertype) {\n    return Supertype.call(template, objectTemplate);\n}\n\nexport function property(objectTemplate, SupertypeModule, props, toClientRuleSet, toServerRuleSet) {\n    props = props || {};\n    props.toClient = applyRuleSet(props.toClient, toClientRuleSet);\n    props.toServer = applyRuleSet(props.toServer, toServerRuleSet);\n    const baseDecorator = SupertypeModule.property(props, objectTemplate);\n    return function (target, targetKey) {\n        baseDecorator(target, targetKey);\n        const defineProperties = {};\n        props.enumerable = true;\n        props.writable = true;\n        objectTemplate._setupProperty(targetKey, props, undefined, defineProperties);\n        Object.defineProperties(target, defineProperties);\n    };\n}\n\n\nexport function remote(objectTemplate, defineProperty) {\n    defineProperty = defineProperty || {};\n\n    /*\n        if we haven't supplied a configuration object into the decorator,\n         default the role of this function to a server API function\n     */\n    if (!defineProperty.on) {\n        defineProperty.on = 'server';\n    }\n\n    // function that we call to validate any changes for remote calls\n    let remoteValidator = defineProperty.serverValidation;\n    let isPublic = !!defineProperty.public;\n\n    return function (target, propertyName, descriptor) {\n        descriptor.value = objectTemplate._setupFunction(\n            propertyName,\n            descriptor.value,\n            defineProperty.on,\n            defineProperty.validate,\n            remoteValidator,\n            isPublic,\n            defineProperty.target\n        );\n\n        /*\n            this function been marked as a server API either explicitly or by default.\n            set the appropriate metadata.\n         */\n        if (defineProperty.on === 'server') {\n            descriptor.value.__on__ = 'server';\n        }\n\n        if (defineProperty.type) {\n            descriptor.value.__returns__ = defineProperty.type;\n        }\n        if (defineProperty.of) {\n            descriptor.value.__returns__ = defineProperty.of;\n            descriptor.value.__returnsarray__ = true;\n        }\n    };\n};\n\n\nfunction applyRuleSet(prop, ruleSet) {\n    if (prop instanceof Array && ruleSet instanceof Array && ruleSet.length > 0) {\n        return prop.some(function (r) {\n            return ruleSet.indexOf(r) >= 0;\n        });\n    } else {\n        return prop;\n    }\n}"]} |
@@ -118,3 +118,3 @@ "use strict"; | ||
function preCallHook(payload, forceupdate) { | ||
var semotus = payload.semotus, remoteCall = payload.remoteCall, callContext = payload.callContext, session = payload.session, subscriptionId = payload.subscriptionId, remoteCallId = payload.remoteCallId, restoreSessionCallback = payload.restoreSessionCallback; | ||
var semotus = payload.semotus, remoteCall = payload.remoteCall, session = payload.session, callContext = payload.callContext, HTTPObjs = payload.HTTPObjs; | ||
semotus.logger.info({ | ||
@@ -134,3 +134,5 @@ component: 'semotus', | ||
} | ||
return semotus.controller.preServerCall.call(semotus.controller, remoteCall.changes.length > 2, changes, callContext, forceupdate); | ||
var remoteObject = session.objects[remoteCall.id]; | ||
var isPublic = semotus.role === 'server' && remoteObject[remoteCall.name].remotePublic; | ||
return semotus.controller.preServerCall.call(semotus.controller, remoteCall.changes.length > 2, changes, callContext, forceupdate, remoteCall.name, remoteCall, isPublic, HTTPObjs); | ||
} | ||
@@ -262,10 +264,10 @@ else { | ||
return __awaiter(this, void 0, void 0, function () { | ||
var semotus, remoteCall, callContext, hasChanges; | ||
var semotus, remoteCall, callContext, HTTPObjs, hasChanges; | ||
return __generator(this, function (_a) { | ||
switch (_a.label) { | ||
case 0: | ||
semotus = payload.semotus, remoteCall = payload.remoteCall, callContext = payload.callContext; | ||
semotus = payload.semotus, remoteCall = payload.remoteCall, callContext = payload.callContext, HTTPObjs = payload.HTTPObjs; | ||
if (!(semotus.controller && semotus.controller.postServerCall)) return [3 /*break*/, 2]; | ||
hasChanges = remoteCall.changes.length > 2; | ||
return [4 /*yield*/, semotus.controller.postServerCall.call(semotus.controller, hasChanges, callContext, semotus.changeString)]; | ||
return [4 /*yield*/, semotus.controller.postServerCall.call(semotus.controller, hasChanges, callContext, semotus.changeString, HTTPObjs)]; | ||
case 1: | ||
@@ -499,2 +501,2 @@ _a.sent(); | ||
exports.postCallErrorLog = postCallErrorLog; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ProcessCall.js","sourceRoot":"","sources":["../../src/helpers/ProcessCall.ts"],"names":[],"mappings":"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","sourcesContent":["import {delay, getError, logTime} from './Utilities';\nimport {CallContext, ProcessCallPayload, RemoteCall, Semotus, Session} from './Types';\n\n/**\n * We process the call the remote method in stages starting by letting the controller examine the\n * changes (preCallHook) and giving it a chance to refresh data if it needs to.  Then we apply any\n * changes in the messages and give the object owning the method a chance to validate that the\n * call is valid and take care of any authorization concerns.  Finally we let the controller perform\n * any post-call processing such as commiting data and then we deal with a failure or success.\n *\n * @param payload\n * @param {unknown} forceupdate unknown\n *\n * @returns {unknown} unknown\n */\nexport async function processCall(payload: ProcessCallPayload, forceupdate?: boolean) {\n\n    try {\n        forceupdate = await Promise.resolve(forceupdate);\n        await preCallHook(payload, forceupdate);\n        let validation = await applyChangesAndValidateCall(payload);\n        validation = await customValidation(payload, validation);\n        const result = await callIfValid(payload, validation);\n        await postCallHook(payload, result);\n        await postCallSuccess(payload, result);\n    } catch (err) {\n        await postCallFailure(payload, err);\n    }\n}\n\n\n/**\n * If there is an update conflict we want to retry after restoring the session\n *\n * @returns {*} unknown\n */\nasync function retryCall(payload: ProcessCallPayload) {\n    const {semotus, remoteCall, callContext, session, subscriptionId, remoteCallId, restoreSessionCallback} = payload;\n\n    if (restoreSessionCallback) {\n        restoreSessionCallback();\n    }\n\n    return processCall(payload, true);\n}\n\n/**\n * Determine what objects changed and pass this to the preServerCall method on the controller\n *\n * @param payload\n * @param  forceupdate unknown\n *\n * @returns  unknown\n */\nfunction preCallHook(payload: ProcessCallPayload, forceupdate?: boolean): boolean {\n    const {semotus, remoteCall, callContext, session, subscriptionId, remoteCallId, restoreSessionCallback} = payload;\n    semotus.logger.info(\n        {\n            component: 'semotus',\n            module: 'processCall',\n            activity: 'preServerCall',\n            data: {\n                call: remoteCall.name,\n                sequence: remoteCall.sequence\n            }\n        },\n        remoteCall.name\n    );\n\n    if (semotus.controller && semotus.controller.preServerCall) {\n        let changes = {};\n\n        for (var objId in JSON.parse(remoteCall.changes)) {\n            changes[semotus.__dictionary__[objId.replace(/[^-]*-/, '').replace(/-.*/, '')].__name__] = true;\n        }\n\n        return semotus.controller.preServerCall.call(\n            semotus.controller,\n            remoteCall.changes.length > 2,\n            changes,\n            callContext,\n            forceupdate\n        );\n    } else {\n        return true;\n    }\n}\n\n/**\n * Apply changes in the message and then validate the call.  Throw \"Sync Error\" if changes can't be applied\n *\n * @returns {unknown} unknown\n */\nfunction applyChangesAndValidateCall(payload: ProcessCallPayload): boolean {\n    const {semotus, remoteCall, callContext, session, subscriptionId, remoteCallId, restoreSessionCallback} = payload;\n\n    semotus.logger.info(\n        {\n            component: 'semotus',\n            module: 'processCall',\n            activity: 'applyChangesAndValidateCall',\n            data: {\n                call: remoteCall.name,\n                sequence: remoteCall.sequence,\n                remoteCallId: remoteCall.id\n            }\n        },\n        remoteCall.name\n    );\n\n    let changes = JSON.parse(remoteCall.changes);\n\n    const res = semotus._applyChanges(changes, semotus.role === 'client', subscriptionId, callContext);\n    if (res) {\n        const obj = session.objects[remoteCall.id];\n\n        if (!obj) {\n            throw new Error(`Cannot find object for remote call ${remoteCall.id}`);\n        }\n\n        // check to see if this function is supposed to be called directly from client\n        if (obj.__proto__[remoteCall.name].__on__ !== 'server') {\n            throw 'Invalid Function Call; not an API function';\n        }\n\n        if (semotus.role === 'server' && obj.validateServerCall) {\n            return obj.validateServerCall.call(obj, remoteCall.name, callContext);\n        }\n\n        return true;\n    } else {\n        throw 'Sync Error';\n    }\n}\n\n/**\n * Apply function specific custom serverSide validation functions\n *\n * @param semotus\n * @param {boolean} isValid - Result of previous validation step (applyChangesAndValidateCall)\n * @param session\n * @param remoteCall\n * @returns {boolean} True if passed function\n */\nfunction customValidation(payload: ProcessCallPayload, isValid: boolean): boolean {\n    const {semotus, remoteCall, callContext, session, subscriptionId, remoteCallId, restoreSessionCallback} = payload;\n\n    let loggerObject = {\n        component: 'semotus',\n        module: 'processCall',\n        activity: 'customValidation',\n        data: {\n            call: remoteCall.name,\n            sequence: remoteCall.sequence,\n            remoteCallId: remoteCall.id\n        }\n    };\n\n    let remoteObject = session.objects[remoteCall.id];\n\n    semotus.logger.info(loggerObject, remoteCall.name);\n\n    if (!isValid) {\n        return false;\n    } else if (semotus.role === 'server' && remoteObject[remoteCall.name].serverValidation) {\n        let args = semotus._extractArguments(remoteCall);\n        args.unshift(remoteObject);\n\n        return remoteObject[remoteCall.name].serverValidation.apply(null, args);\n    } else {\n        return true;\n    }\n}\n\n/**\n * If the changes could be applied and the validation was successful call the method\n *\n * @param semotus\n * @param {boolean} isValid - takes a flag if the call is valid or not, if it is then we proceed normally,\n * otherwise, we throw an error and stop execution\n *\n * @param session\n * @param remoteCall\n * @returns {unknown} unknown\n */\nasync function callIfValid(payload: ProcessCallPayload, isValid: boolean) {\n    const {semotus, remoteCall, callContext, session, subscriptionId, remoteCallId, restoreSessionCallback} = payload;\n\n    let loggerObject = {\n        component: 'semotus',\n        module: 'processCall',\n        activity: 'callIfValid',\n        data: {\n            call: remoteCall.name,\n            sequence: remoteCall.sequence,\n            remoteCallId: remoteCall.id\n        }\n    };\n\n    semotus.logger.info(loggerObject, remoteCall.name);\n\n    let obj = session.objects[remoteCall.id];\n\n    if (!obj[remoteCall.name]) {\n        throw new Error(remoteCall.name + ' function does not exist.');\n    }\n\n    if (!isValid && remoteCall && remoteCall.name) {\n        throw new Error(remoteCall.name + ' refused');\n    }\n\n    let args = semotus._extractArguments(remoteCall);\n\n    return obj[remoteCall.name].apply(obj, args);\n}\n\n/**\n * Let the controller know that the method was completed and give it a chance to commit changes\n *\n * @param payload\n * @param  returnValue unknown\n *\n * @returns\n */\nasync function postCallHook(payload: ProcessCallPayload, returnValue) {\n    const {semotus, remoteCall, callContext} = payload;\n\n    if (semotus.controller && semotus.controller.postServerCall) {\n        const hasChanges: boolean = remoteCall.changes.length > 2;\n        await semotus.controller.postServerCall.call(semotus.controller, hasChanges, callContext, semotus.changeString);\n    }\n    return returnValue;\n}\n\n/**\n * Package up any changes resulting from the execution and send them back in the message, clearing\n * our change queue to accumulate more changes for the next call\n *\n * @param semotus\n * @param remoteCall\n * @param remoteCallId\n * @param {unknown} ret unknown\n */\nfunction postCallSuccess(payload: ProcessCallPayload, ret): void {\n    const {semotus, remoteCall, callContext, session, remoteCallId} = payload;\n\n    semotus.logger.info(\n        {\n            component: 'semotus',\n            module: 'processCall',\n            activity: 'postCall.success',\n            data: {\n                call: remoteCall.name,\n                callTime: logTime(callContext),\n                sequence: remoteCall.sequence\n            }\n        },\n        remoteCall.name\n    );\n\n    packageChanges(semotus, session,{\n        type: 'response',\n        sync: true,\n        value: JSON.stringify(semotus._toTransport(ret)),\n        name: remoteCall.name,\n        remoteCallId: remoteCallId\n    });\n}\n\n/**\n * Helper function to identify if there's a postServerErrorHandler callback on the base controller\n * If there is, we execute the handler, and if we catch an error in the handler, we propogate it up to the logger.\n * @param logger\n * @param {*} controller\n * @param type\n * @param remoteCall\n * @param remoteCallId\n * @param callContext\n * @param changeString\n * @param session\n */\nasync function resolveErrorHandler(logger, controller: any, type, remoteCall: RemoteCall, remoteCallId, callContext: CallContext, changeString, session: Session) {\n\n    if (controller && controller.postServerErrorHandler) {\n        let errorType = type;\n        let functionName = remoteCall.name;\n        let obj = undefined;\n        if (session.objects[remoteCall.id]) {\n            obj = session.objects[remoteCall.id];\n        }\n        let logBody = {\n            component: 'semotus',\n            module: 'processCall.failure',\n            activity: 'postCall.resolveErrorHandler',\n            data: {\n                call: remoteCall.name,\n                message: undefined\n            }\n        };\n\n        try {\n            await controller.postServerErrorHandler.call(controller, errorType, remoteCallId, obj, functionName, callContext, changeString);\n        } catch (error) {\n            if (error.message) {\n                logBody.data.message = error.message;\n                logger.error(error.message);\n            } else {\n                logBody.data.message = JSON.stringify(error);\n            }\n\n            logger.error(logBody, 'User defined postServerErrorHandler threw an error');\n        }\n    }\n}\n\n/**\n * Handle errors by returning an apropriate message.  In all cases changes sent back though they\n *\n * @param semotus\n * @param remoteCall\n * @param callContext\n * @param session\n * @param {unknown} err unknown\n *\n * @returns {unknown} A Promise\n */\nasync function postCallFailure(payload: ProcessCallPayload, err) {\n    const {semotus, remoteCall, callContext, session, subscriptionId, remoteCallId, restoreSessionCallback} = payload;\n\n    let logString = '';\n\n    let packageChangesPayload = {};\n\n    let updateConflictRetry = false;\n\n    if (err === 'Sync Error') {\n        postCallErrorLog(semotus.logger, 'postCall.syncError', undefined, 'error', remoteCall.name, remoteCall, callContext);\n        packageChangesPayload = {\n            type: 'response',\n            sync: false,\n            changes: ''\n        };\n    } else if (err.message == 'Update Conflict') {\n        // Not this may be caught in the transport (e.g. Amorphic) and retried)\n\n        // increment callContext.retries after checking if < 3. Should retry 3 times.\n        if (callContext.retries++ < 3) {\n            postCallErrorLog(semotus.logger, 'postCall.updateConflict', undefined, 'warn', remoteCall.name, remoteCall, callContext);\n            updateConflictRetry = true;\n            // The following assignment is only used for the error handler\n            packageChangesPayload = {\n                type: 'retry'\n            };\n        } else {\n            postCallErrorLog(semotus.logger, 'postCall.updateConflict', undefined, 'error', remoteCall.name, remoteCall, callContext);\n            packageChangesPayload = {\n                type: 'retry',\n                sync: false\n            };\n        }\n    } else {\n        if (!(err instanceof Error)) {\n            postCallErrorLog(semotus.logger, 'postCall.error', JSON.stringify(err), 'info', remoteCall.name, remoteCall, callContext);\n        } else {\n            if (err.stack) {\n                logString = 'Exception in ' + remoteCall.name + ' - ' + err.message + (' ' + err.stack);\n            } else {\n                logString = 'Exception in ' + remoteCall.name + ' - ' + err.message;\n            }\n\n            postCallErrorLog(semotus.logger, 'postCall.exception', err.message, 'error', logString, remoteCall, callContext);\n        }\n\n        packageChangesPayload = {\n            type: 'error',\n            sync: true,\n            value: getError(err),\n            name: remoteCall.name\n        };\n    }\n\n    Object.assign(packageChangesPayload, {remoteCallId: remoteCallId});\n\n    await resolveErrorHandler(\n        semotus.logger,\n        semotus.controller,\n        // @ts-ignore\n        packageChangesPayload.type,\n        remoteCall,\n        remoteCallId,\n        callContext,\n        semotus.changeString,\n        session\n    );\n\n    if (updateConflictRetry) {\n        await delay(callContext.retries * 1000);\n        return retryCall(payload);\n    } else {\n        return packageChanges(semotus, session, packageChangesPayload);\n    }\n}\n\n/**\n * Deal with changes going back to the caller - Actually not Async!\n *\n * @param semotus\n * @param session\n * @param {unknown} message unknown\n */\nfunction packageChanges(semotus: Semotus, session: Session, message) {\n    semotus._convertArrayReferencesToChanges();\n    message.changes = JSON.stringify(semotus.getChanges());\n\n    if (semotus.memSession && semotus.memSession.semotus && semotus.memSession.semotus.callStartTime) {\n        semotus.memSession.semotus.callStartTime = 0;\n    }\n\n    // Inspect a copy of the message on the server only before sent out. Only for debug purposes\n    if (semotus.controller && semotus.controller.inspectMessage) {\n        semotus.controller.inspectMessage(Object.assign({}, message));\n    }\n\n    session.sendMessage(message);\n    semotus._deleteChanges();\n    semotus._processQueue();\n}\n\n\n/**\n *  Helper function to log amorphic errors.\n * @param {*} logger\n * @param {*} activity\n * @param {*} message\n * @param {*} logType\n * @param {*} logString\n * @param remoteCall\n * @param callContext\n */\nexport function postCallErrorLog(logger, activity, message, logType, logString, remoteCall: RemoteCall, callContext: CallContext) {\n    let logBody = {\n        component: 'semotus',\n        module: 'processCall.failure',\n        data: {\n            call: remoteCall.name,\n            callTime: logTime(callContext),\n            sequence: remoteCall.sequence,\n            message: undefined\n        },\n        activity: undefined\n    };\n\n    logBody.activity = activity;\n\n    if (logger.data) {\n        logBody.data.message = message;\n    }\n\n    logger[logType](logBody, logString);\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ProcessCall.js","sourceRoot":"","sources":["../../src/helpers/ProcessCall.ts"],"names":[],"mappings":"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","sourcesContent":["import {delay, getError, logTime} from './Utilities';\nimport {CallContext, ProcessCallPayload, RemoteCall, Semotus, Session} from './Types';\n\n/**\n * We process the call the remote method in stages starting by letting the controller examine the\n * changes (preCallHook) and giving it a chance to refresh data if it needs to.  Then we apply any\n * changes in the messages and give the object owning the method a chance to validate that the\n * call is valid and take care of any authorization concerns.  Finally we let the controller perform\n * any post-call processing such as commiting data and then we deal with a failure or success.\n *\n * @param payload\n * @param {unknown} forceupdate unknown\n *\n * @returns {unknown} unknown\n */\nexport async function processCall(payload: ProcessCallPayload, forceupdate?: boolean) {\n\n    try {\n        forceupdate = await Promise.resolve(forceupdate);\n        await preCallHook(payload, forceupdate);\n        let validation = await applyChangesAndValidateCall(payload);\n        validation = await customValidation(payload, validation);\n        const result = await callIfValid(payload, validation);\n        await postCallHook(payload, result);\n        await postCallSuccess(payload, result);\n    } catch (err) {\n        await postCallFailure(payload, err);\n    }\n}\n\n\n/**\n * If there is an update conflict we want to retry after restoring the session\n *\n * @returns {*} unknown\n */\nasync function retryCall(payload: ProcessCallPayload) {\n    const {semotus, remoteCall, callContext, session, subscriptionId, remoteCallId, restoreSessionCallback} = payload;\n\n    if (restoreSessionCallback) {\n        restoreSessionCallback();\n    }\n\n    return processCall(payload, true);\n}\n\n/**\n * Determine what objects changed and pass this to the preServerCall method on the controller\n *\n * @param payload\n * @param  forceupdate unknown\n *\n * @returns  unknown\n */\nfunction preCallHook(payload: ProcessCallPayload, forceupdate?: boolean): boolean {\n    const {semotus, remoteCall, session, callContext, HTTPObjs} = payload;\n    semotus.logger.info(\n        {\n            component: 'semotus',\n            module: 'processCall',\n            activity: 'preServerCall',\n            data: {\n                call: remoteCall.name,\n                sequence: remoteCall.sequence\n            }\n        },\n        remoteCall.name\n    );\n\n    if (semotus.controller && semotus.controller.preServerCall) {\n        let changes = {};\n\n        for (var objId in JSON.parse(remoteCall.changes)) {\n            changes[semotus.__dictionary__[objId.replace(/[^-]*-/, '').replace(/-.*/, '')].__name__] = true;\n        }\n\n        let remoteObject = session.objects[remoteCall.id];\n\n        let isPublic = semotus.role === 'server' && remoteObject[remoteCall.name].remotePublic;\n\n        return semotus.controller.preServerCall.call(\n            semotus.controller,\n            remoteCall.changes.length > 2,\n            changes,\n            callContext,\n            forceupdate,\n            remoteCall.name,\n            remoteCall,\n            isPublic,\n            HTTPObjs\n        );\n    } else {\n        return true;\n    }\n}\n\n/**\n * Apply changes in the message and then validate the call.  Throw \"Sync Error\" if changes can't be applied\n *\n * @returns {unknown} unknown\n */\nfunction applyChangesAndValidateCall(payload: ProcessCallPayload): boolean {\n    const {semotus, remoteCall, callContext, session, subscriptionId, remoteCallId, restoreSessionCallback} = payload;\n\n    semotus.logger.info(\n        {\n            component: 'semotus',\n            module: 'processCall',\n            activity: 'applyChangesAndValidateCall',\n            data: {\n                call: remoteCall.name,\n                sequence: remoteCall.sequence,\n                remoteCallId: remoteCall.id\n            }\n        },\n        remoteCall.name\n    );\n\n    let changes = JSON.parse(remoteCall.changes);\n\n    const res = semotus._applyChanges(changes, semotus.role === 'client', subscriptionId, callContext);\n    if (res) {\n        const obj = session.objects[remoteCall.id];\n\n        if (!obj) {\n            throw new Error(`Cannot find object for remote call ${remoteCall.id}`);\n        }\n\n        // check to see if this function is supposed to be called directly from client\n        if (obj.__proto__[remoteCall.name].__on__ !== 'server') {\n            throw 'Invalid Function Call; not an API function';\n        }\n\n        if (semotus.role === 'server' && obj.validateServerCall) {\n            return obj.validateServerCall.call(obj, remoteCall.name, callContext);\n        }\n\n        return true;\n    } else {\n        throw 'Sync Error';\n    }\n}\n\n/**\n * Apply function specific custom serverSide validation functions\n *\n * @param semotus\n * @param {boolean} isValid - Result of previous validation step (applyChangesAndValidateCall)\n * @param session\n * @param remoteCall\n * @returns {boolean} True if passed function\n */\nfunction customValidation(payload: ProcessCallPayload, isValid: boolean): boolean {\n    const {semotus, remoteCall, callContext, session, subscriptionId, remoteCallId, restoreSessionCallback} = payload;\n\n    let loggerObject = {\n        component: 'semotus',\n        module: 'processCall',\n        activity: 'customValidation',\n        data: {\n            call: remoteCall.name,\n            sequence: remoteCall.sequence,\n            remoteCallId: remoteCall.id\n        }\n    };\n\n    let remoteObject = session.objects[remoteCall.id];\n\n    semotus.logger.info(loggerObject, remoteCall.name);\n\n    if (!isValid) {\n        return false;\n    } else if (semotus.role === 'server' && remoteObject[remoteCall.name].serverValidation) {\n        let args = semotus._extractArguments(remoteCall);\n        args.unshift(remoteObject);\n\n        return remoteObject[remoteCall.name].serverValidation.apply(null, args);\n    } else {\n        return true;\n    }\n}\n\n/**\n * If the changes could be applied and the validation was successful call the method\n *\n * @param semotus\n * @param {boolean} isValid - takes a flag if the call is valid or not, if it is then we proceed normally,\n * otherwise, we throw an error and stop execution\n *\n * @param session\n * @param remoteCall\n * @returns {unknown} unknown\n */\nasync function callIfValid(payload: ProcessCallPayload, isValid: boolean) {\n    const {semotus, remoteCall, callContext, session, subscriptionId, remoteCallId, restoreSessionCallback} = payload;\n\n    let loggerObject = {\n        component: 'semotus',\n        module: 'processCall',\n        activity: 'callIfValid',\n        data: {\n            call: remoteCall.name,\n            sequence: remoteCall.sequence,\n            remoteCallId: remoteCall.id\n        }\n    };\n\n    semotus.logger.info(loggerObject, remoteCall.name);\n\n    let obj = session.objects[remoteCall.id];\n\n    if (!obj[remoteCall.name]) {\n        throw new Error(remoteCall.name + ' function does not exist.');\n    }\n\n    if (!isValid && remoteCall && remoteCall.name) {\n        throw new Error(remoteCall.name + ' refused');\n    }\n\n    let args = semotus._extractArguments(remoteCall);\n\n    return obj[remoteCall.name].apply(obj, args);\n}\n\n/**\n * Let the controller know that the method was completed and give it a chance to commit changes\n *\n * @param payload\n * @param  returnValue unknown\n *\n * @returns\n */\nasync function postCallHook(payload: ProcessCallPayload, returnValue) {\n    const {semotus, remoteCall, callContext, HTTPObjs} = payload;\n\n    if (semotus.controller && semotus.controller.postServerCall) {\n        const hasChanges: boolean = remoteCall.changes.length > 2;\n        await semotus.controller.postServerCall.call(semotus.controller, hasChanges, callContext, semotus.changeString, HTTPObjs);\n    }\n    return returnValue;\n}\n\n/**\n * Package up any changes resulting from the execution and send them back in the message, clearing\n * our change queue to accumulate more changes for the next call\n *\n * @param semotus\n * @param remoteCall\n * @param remoteCallId\n * @param {unknown} ret unknown\n */\nfunction postCallSuccess(payload: ProcessCallPayload, ret): void {\n    const {semotus, remoteCall, callContext, session, remoteCallId} = payload;\n\n    semotus.logger.info(\n        {\n            component: 'semotus',\n            module: 'processCall',\n            activity: 'postCall.success',\n            data: {\n                call: remoteCall.name,\n                callTime: logTime(callContext),\n                sequence: remoteCall.sequence\n            }\n        },\n        remoteCall.name\n    );\n\n    packageChanges(semotus, session,{\n        type: 'response',\n        sync: true,\n        value: JSON.stringify(semotus._toTransport(ret)),\n        name: remoteCall.name,\n        remoteCallId: remoteCallId\n    });\n}\n\n/**\n * Helper function to identify if there's a postServerErrorHandler callback on the base controller\n * If there is, we execute the handler, and if we catch an error in the handler, we propogate it up to the logger.\n * @param logger\n * @param {*} controller\n * @param type\n * @param remoteCall\n * @param remoteCallId\n * @param callContext\n * @param changeString\n * @param session\n */\nasync function resolveErrorHandler(logger, controller: any, type, remoteCall: RemoteCall, remoteCallId, callContext: CallContext, changeString, session: Session) {\n\n    if (controller && controller.postServerErrorHandler) {\n        let errorType = type;\n        let functionName = remoteCall.name;\n        let obj = undefined;\n        if (session.objects[remoteCall.id]) {\n            obj = session.objects[remoteCall.id];\n        }\n        let logBody = {\n            component: 'semotus',\n            module: 'processCall.failure',\n            activity: 'postCall.resolveErrorHandler',\n            data: {\n                call: remoteCall.name,\n                message: undefined\n            }\n        };\n\n        try {\n            await controller.postServerErrorHandler.call(controller, errorType, remoteCallId, obj, functionName, callContext, changeString);\n        } catch (error) {\n            if (error.message) {\n                logBody.data.message = error.message;\n                logger.error(error.message);\n            } else {\n                logBody.data.message = JSON.stringify(error);\n            }\n\n            logger.error(logBody, 'User defined postServerErrorHandler threw an error');\n        }\n    }\n}\n\n/**\n * Handle errors by returning an apropriate message.  In all cases changes sent back though they\n *\n * @param semotus\n * @param remoteCall\n * @param callContext\n * @param session\n * @param {unknown} err unknown\n *\n * @returns {unknown} A Promise\n */\nasync function postCallFailure(payload: ProcessCallPayload, err) {\n    const {semotus, remoteCall, callContext, session, subscriptionId, remoteCallId, restoreSessionCallback} = payload;\n\n    let logString = '';\n\n    let packageChangesPayload = {};\n\n    let updateConflictRetry = false;\n\n    if (err === 'Sync Error') {\n        postCallErrorLog(semotus.logger, 'postCall.syncError', undefined, 'error', remoteCall.name, remoteCall, callContext);\n        packageChangesPayload = {\n            type: 'response',\n            sync: false,\n            changes: ''\n        };\n    } else if (err.message == 'Update Conflict') {\n        // Not this may be caught in the transport (e.g. Amorphic) and retried)\n\n        // increment callContext.retries after checking if < 3. Should retry 3 times.\n        if (callContext.retries++ < 3) {\n            postCallErrorLog(semotus.logger, 'postCall.updateConflict', undefined, 'warn', remoteCall.name, remoteCall, callContext);\n            updateConflictRetry = true;\n            // The following assignment is only used for the error handler\n            packageChangesPayload = {\n                type: 'retry'\n            };\n        } else {\n            postCallErrorLog(semotus.logger, 'postCall.updateConflict', undefined, 'error', remoteCall.name, remoteCall, callContext);\n            packageChangesPayload = {\n                type: 'retry',\n                sync: false\n            };\n        }\n    } else {\n        if (!(err instanceof Error)) {\n            postCallErrorLog(semotus.logger, 'postCall.error', JSON.stringify(err), 'info', remoteCall.name, remoteCall, callContext);\n        } else {\n            if (err.stack) {\n                logString = 'Exception in ' + remoteCall.name + ' - ' + err.message + (' ' + err.stack);\n            } else {\n                logString = 'Exception in ' + remoteCall.name + ' - ' + err.message;\n            }\n\n            postCallErrorLog(semotus.logger, 'postCall.exception', err.message, 'error', logString, remoteCall, callContext);\n        }\n\n        packageChangesPayload = {\n            type: 'error',\n            sync: true,\n            value: getError(err),\n            name: remoteCall.name\n        };\n    }\n\n    Object.assign(packageChangesPayload, {remoteCallId: remoteCallId});\n\n    await resolveErrorHandler(\n        semotus.logger,\n        semotus.controller,\n        // @ts-ignore\n        packageChangesPayload.type,\n        remoteCall,\n        remoteCallId,\n        callContext,\n        semotus.changeString,\n        session\n    );\n\n    if (updateConflictRetry) {\n        await delay(callContext.retries * 1000);\n        return retryCall(payload);\n    } else {\n        return packageChanges(semotus, session, packageChangesPayload);\n    }\n}\n\n/**\n * Deal with changes going back to the caller - Actually not Async!\n *\n * @param semotus\n * @param session\n * @param {unknown} message unknown\n */\nfunction packageChanges(semotus: Semotus, session: Session, message) {\n    semotus._convertArrayReferencesToChanges();\n    message.changes = JSON.stringify(semotus.getChanges());\n\n    if (semotus.memSession && semotus.memSession.semotus && semotus.memSession.semotus.callStartTime) {\n        semotus.memSession.semotus.callStartTime = 0;\n    }\n\n    // Inspect a copy of the message on the server only before sent out. Only for debug purposes\n    if (semotus.controller && semotus.controller.inspectMessage) {\n        semotus.controller.inspectMessage(Object.assign({}, message));\n    }\n\n    session.sendMessage(message);\n    semotus._deleteChanges();\n    semotus._processQueue();\n}\n\n\n/**\n *  Helper function to log amorphic errors.\n * @param {*} logger\n * @param {*} activity\n * @param {*} message\n * @param {*} logType\n * @param {*} logString\n * @param remoteCall\n * @param callContext\n */\nexport function postCallErrorLog(logger, activity, message, logType, logString, remoteCall: RemoteCall, callContext: CallContext) {\n    let logBody = {\n        component: 'semotus',\n        module: 'processCall.failure',\n        data: {\n            call: remoteCall.name,\n            callTime: logTime(callContext),\n            sequence: remoteCall.sequence,\n            message: undefined\n        },\n        activity: undefined\n    };\n\n    logBody.activity = activity;\n\n    if (logger.data) {\n        logBody.data.message = message;\n    }\n\n    logger[logType](logBody, logString);\n}\n"]} |
"use strict"; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.Change = 'change'; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Types.js","sourceRoot":"","sources":["../../src/helpers/Types.ts"],"names":[],"mappings":";;AAWa,QAAA,MAAM,GAAG,QAAQ,CAAC","sourcesContent":["import {Supertype} from '@haventech/supertype';\n\nexport type Subscription = {\n    role: string;\n    log: {\n        array: ArrayGroup;\n        arrayDirty: ArrayGroup;\n        change: ChangeGroup;\n    }\n}\n\nexport const Change = 'change';\nexport type ArrayTypes = 'array' | 'arrayDirty';\nexport type ErrorType = 'error' | 'retry' | 'response';\n\nexport type CallContext = { retries: number; startTime: Date };\n\nexport type ProcessCallPayload = {\n    semotus: Semotus;\n    remoteCall: RemoteCall;\n    callContext: CallContext;\n    session: Session;\n    subscriptionId: any;\n    remoteCallId: any;\n    restoreSessionCallback?: Function;\n}\n\n/**\n *  id is the id of the object + '/' + property.\n *  Ex: there are 1 School to Many Students\n *\n *  School {\n *      students: Array<Students>\n *  }\n *\n *  In this case, one entry within this ArrayGroup may be 'server-School-1/students': [\"server-Student-3\", \"server-Student-5\"]\n *\n *  If that's all the changes then the ArrayGroup would be {'server-School-1/students': [\"server-Student-3\", \"server-Student-5\"]}\n */\nexport type ArrayGroup = { [id: string]: ArrayChanges };\n\n/**\n * An array of Id references to Supertype Objects or primitive values\n */\ntype ArrayChanges = Array<any>;\nexport type ChangeGroup = { [objId: string]: PropChanges };\ntype PropChanges = { [prop: string]: Changes };\n\n// Changes[0] is oldValue, Changes[1] is newValue\ntype oldVal = any;\ntype newVal = any;\ntype Changes = [oldVal, newVal][];\n\nexport type Subscriptions = { [key: string]: Subscription };\n\nexport type RemoteCall = {\n    remoteCallId: any;\n    id: any;\n    changes: string; // The string is of type ChangeGroup\n    name: string;\n    sequence: any;\n}\n\nexport type Session = {\n    subscriptions: Subscriptions;\n    sendMessage: SendMessage;\n    sendMessageEnabled: boolean;\n    remoteCalls: Array<any>;\n    pendingRemoteCalls: any;\n    nextPendingRemoteCallId: number;\n    nextSaveSessionId: number;\n    savedSessionId: number;\n    nextSubscriptionId: number;\n    objects: any;\n    nextObjId: number;\n    dispenseNextId: null; // not used anywhere\n}\n\n\nexport type Sessions = { [sessionId: number]: Session };\n\nexport type SendMessage = (message: any) => void;\n\nexport type SavedSession = {\n    revision: number;\n    data: string; // Serialized Session data\n    callCount: number;\n    referenced: number;\n};\n\nexport interface RemoteableClass extends Supertype {\n    syncStates: Array<string>;\n    __toClient__: boolean;\n    __toServer__: boolean;\n}\n\nexport type ChangeString = { [key: string]: string };\ntype PreServerCallChanges = { [key: string]: boolean };\n\nexport type ControllerSyncState = {\n    scope: '+' | '*' | '-'\n    state: string\n}\n\nexport interface ISemotusController {\n    __template__: any;\n    syncState?: ControllerSyncState\n\n    /**\n     * @server\n     *\n     * Callback after a successful remote function call (just the application of changes and the execution of the function call)\n     * Note that this doesn't mean we can't error out on this or subsequent steps of a remote call.\n     *\n     * We can utilize this function as a generic function handler to run after we have successfully called a remote function.\n     * One such use may be to see the changes that were applied from the client\n     *\n     * NOTE THAT THE CHANGESTRING DOES NOT INCLUDE CHANGES DONE WITHIN THE REMOTE FUNCTION CALL ITSELF, ONLY CHANGES FROM THE CLIENT\n     *\n     * See remote call documentation to know where this executes in the lifecycle\n     *\n     * @param {boolean} hasChanges - Whether or not we have applied client changes onto the server's object graph\n     * @param {CallContext} callContext - Context (number of retries etc)\n     * @param {changeString} changeString - Object of Changes - Key is [ClassName].[propertyName], Value is [changedValue] example: {'Customer.middlename': 'Karen'}, See above note\n     *\n     * @returns {Promise<void>}\n     * @memberof Controller\n     */\n    postServerCall?(hasChanges: boolean, callContext: CallContext, changeString: ChangeString): Promise<any>;\n\n    /**\n     * @server\n     *\n     * Callback to handle errors on a remote call.\n     *\n     * Executes after every other step in the remote call pipeline (see remote call documentation)\n     * but before retrying the call (or packaging response and sending back to client)\n     *\n     * NOTE THAT THE CHANGESTRING DOES NOT INCLUDE CHANGES DONE WITHIN THE REMOTE FUNCTION CALL ITSELF, ONLY CHANGES FROM THE CLIENT\n     *\n     * @param {ErrorType} errorType - Error type associated (error, retry, response)\n     * @param {number} remoteCallId - Id for remote call\n     * @param {extends Supertype} remoteObj - Instance for which the remote object function is called for - @TODO: revisit when we create a proper remoteable type\n     * @param {string} functionName - Name of function being called\n     * @param {CallContext} callContext - Context (number of retries etc)\n     * @param {ChangeString} changeString - Object of Changes - Key is [ClassName].[propertyName], Value is [changedValue] example: {'Customer.middlename': 'Karen'}, See above note\n     *\n     * @returns {Promise<void>}\n     * @memberof Controller\n     */\n    postServerErrorHandler?(errorType: ErrorType, remoteCallId: number, remoteObj: Supertype, functionName: string, callContext: CallContext, changeString: ChangeString): Promise<void>;\n\n    /**\n     * @server\n     *\n     * Callback before a remote function call (1st step of a remote call)\n     *\n     * We can utilize this function as a generic function handler to run before we call a\n     * remote function or before we apply changes from the client to the server\n     *\n     * We can also utilize this function to do any context-specific prep work / loading\n     * if this a subsequent try of this function due to an update conflict.\n     *\n     * See remote call documentation to know where this executes in the lifecycle\n     *\n     * @param {boolean} hasChanges - Whether or not we have applied client changes onto the server's object graph\n     * @param {PreServerCallChanges} changes - Dictionary of Objects that have been changed from the client\n     * @param {CallContext} callContext - Context (number of retries etc)\n     * @param {boolean} [forceUpdate] - Optional parameter passed in. True if this is a retry of the call based on an update conflict. False / undefined otherwise.\n     *\n     * @returns {Promise<void>}\n     * @memberof Controller\n     */\n    preServerCall?(hasChanges: boolean, changes: PreServerCallChanges, callContext: CallContext, forceUpdate?: boolean): Promise<void>;\n\n    /**\n     * This is a handler that will only be used for testing and debugging purposes\n     *\n     * When this is used, allows us to inspect the message body before we send it out in 'packageChanges'\n     *\n     * DO NOT USE THIS IN PRODUCTION CODE\n     *\n     * @param message\n     */\n    inspectMessage?(message): any;\n}\n\nexport interface Semotus {\n    setSyncState: (str) => void;\n    syncState: string;\n    getSyncState: () => any;\n    maxCallTime: number;\n    __dictionary__: any;\n    memSession: { semotus: { callStartTime: number } };\n    _injectIntoTemplate: (template) => void;\n    serializeAndGarbageCollect: () => any;\n    getMessage: (sessionId, forceMessage) => any;\n    clearPendingCalls: (sessionId) => void;\n    getChangeGroup: (type, subscriptionId) => ChangeGroup;\n    deleteChangeGroup: (type: any, subscriptionId: any) => void;\n    getChangeStatus: () => string;\n    _stashObject: (obj, template) => boolean;\n    sessionize: (obj, referencingObj) => (undefined | any);\n    _setupFunction: (propertyName, propertyValue, role, validate, serverValidation, template) => (any);\n    _setupProperty: (propertyName, defineProperty, objectProperties, defineProperties) => void;\n    withoutChangeTracking: (cb) => void;\n    _changedValue: (obj, prop, value) => void;\n    _referencedArray: (obj, prop, arrayRef, sessionId?) => void;\n    _convertArrayReferencesToChanges: () => void;\n    MarkChangedArrayReferences: () => void;\n    _convertValue: (value) => (any[] | null);\n    getObject: (objId, template) => (any | null);\n    _applyChanges: (changes, force, subscriptionId, callContext) => (number);\n    _applyObjectChanges: (changes, rollback, obj, force) => (boolean);\n    _validateServerIncomingProperty: (obj, prop, defineProperty, newValue) => (boolean);\n    _applyPropertyChange: (changes, rollback, obj, prop, ix, oldValue, newValue, force) => (boolean);\n    _rollback: (rollback) => void;\n    _rollbackChanges: () => void;\n    _createEmptyObject: (template, objId, defineProperty, isTransient) => any;\n    inject: (template, injector) => void;\n    _queueRemoteCall: (objId, functionName, deferred, args) => void;\n    _processQueue: () => void;\n    _toTransport: (obj) => any;\n    _fromTransport: (obj) => any;\n    _extractArguments: (remoteCall) => any;\n    _trimArray: (array) => void;\n    _deleteChangeGroups: (type) => void;\n    _getSubscriptions: (sessionId?) => Subscriptions | null;\n    _getSubscription: (subscriptionId?) => Subscription;\n    cleanPrivateValues: (prop, logValue, defineProperty) => (string | any);\n    Remoteable: (Base) => () => any;\n    Bindable: (Base) => () => any;\n    Persistable: (Base) => () => any;\n    bindDecorators: (objectTemplate?) => void;\n    processMessage: (remoteCall, subscriptionId, restoreSessionCallback) => (undefined | any);\n    enableSendMessage: (value, messageCallback, sessionId) => void;\n    syncSession: (sessionId) => void;\n    restoreSession: (sessionId, savedSession: SavedSession, sendMessage: SendMessage) => boolean;\n    saveSession: (sessionId) => SavedSession;\n    setMinimumSequence: (nextObjId) => void;\n    deleteSession: (sessionId) => void;\n    createSession: (role: any, sendMessage: SendMessage, sessionId: any) => any;\n    log: (level, data) => void;\n    nextObjId: number;\n    maxClientSequence: number;\n    logLevel: number;\n    __conflictMode__: string;\n    __changeTracking__: boolean;\n    _useGettersSetters: boolean;\n    logger: any;\n    role: any;\n    currentSession: any;\n    sessions?: Sessions;\n    nextSubscriptionId: number;\n    nextSessionId: number;\n    controller: ISemotusController;\n    changeString: string;\n\n\n    _getSession(_sid?: any): Session;\n\n    subscribe(role: any): number;\n\n    setSession(sessionId: any): void;\n\n    getChanges(subscriptionId?: any): ChangeGroup;\n\n    _deleteChanges(): void;\n\n    getPendingCallCount(sessionId: any): any;\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Types.js","sourceRoot":"","sources":["../../src/helpers/Types.ts"],"names":[],"mappings":";;AAaa,QAAA,MAAM,GAAG,QAAQ,CAAC","sourcesContent":["import {Supertype} from '@haventech/supertype';\nimport {Request, Response} from 'express';\nexport type HTTPObjs = {request: Request, response: Response};\n\nexport type Subscription = {\n    role: string;\n    log: {\n        array: ArrayGroup;\n        arrayDirty: ArrayGroup;\n        change: ChangeGroup;\n    }\n}\n\nexport const Change = 'change';\nexport type ArrayTypes = 'array' | 'arrayDirty';\nexport type ErrorType = 'error' | 'retry' | 'response';\n\nexport type CallContext = { retries: number; startTime: Date };\n\nexport type ProcessCallPayload = {\n    semotus: Semotus;\n    remoteCall: RemoteCall;\n    callContext: CallContext;\n    session: Session;\n    subscriptionId: any;\n    remoteCallId: any;\n    restoreSessionCallback?: Function;\n    HTTPObjs?: HTTPObjs;\n}\n\n/**\n *  id is the id of the object + '/' + property.\n *  Ex: there are 1 School to Many Students\n *\n *  School {\n *      students: Array<Students>\n *  }\n *\n *  In this case, one entry within this ArrayGroup may be 'server-School-1/students': [\"server-Student-3\", \"server-Student-5\"]\n *\n *  If that's all the changes then the ArrayGroup would be {'server-School-1/students': [\"server-Student-3\", \"server-Student-5\"]}\n */\nexport type ArrayGroup = { [id: string]: ArrayChanges };\n\n/**\n * An array of Id references to Supertype Objects or primitive values\n */\ntype ArrayChanges = Array<any>;\nexport type ChangeGroup = { [objId: string]: PropChanges };\ntype PropChanges = { [prop: string]: Changes };\n\n// Changes[0] is oldValue, Changes[1] is newValue\ntype oldVal = any;\ntype newVal = any;\ntype Changes = [oldVal, newVal][];\n\nexport type Subscriptions = { [key: string]: Subscription };\n\nexport type RemoteCall = {\n    remoteCallId: any;\n    id: any;\n    changes: string; // The string is of type ChangeGroup\n    name: string;\n    sequence: any;\n    type: any;\n    sync: any;\n    value: any;\n}\n\nexport type Session = {\n    subscriptions: Subscriptions;\n    sendMessage: SendMessage;\n    sendMessageEnabled: boolean;\n    remoteCalls: Array<any>;\n    pendingRemoteCalls: any;\n    nextPendingRemoteCallId: number;\n    nextSaveSessionId: number;\n    savedSessionId: number;\n    nextSubscriptionId: number;\n    objects: any;\n    nextObjId: number;\n    dispenseNextId: null; // not used anywhere\n}\n\n\nexport type Sessions = { [sessionId: number]: Session };\n\nexport type SendMessage = (message: any) => void;\n\nexport type SavedSession = {\n    revision: number;\n    data: string; // Serialized Session data\n    callCount: number;\n    referenced: number;\n};\n\nexport interface RemoteableClass extends Supertype {\n    syncStates: Array<string>;\n    __toClient__: boolean;\n    __toServer__: boolean;\n}\n\nexport type ChangeString = { [key: string]: string };\ntype PreServerCallChanges = { [key: string]: boolean };\n\nexport type ControllerSyncState = {\n    scope: '+' | '*' | '-'\n    state: string\n}\n\nexport interface ISemotusController {\n    __template__: any;\n    syncState?: ControllerSyncState\n\n    /**\n     * @server\n     *\n     * Callback after a successful remote function call (just the application of changes and the execution of the function call)\n     * Note that this doesn't mean we can't error out on this or subsequent steps of a remote call.\n     *\n     * We can utilize this function as a generic function handler to run after we have successfully called a remote function.\n     * One such use may be to see the changes that were applied from the client\n     *\n     * NOTE THAT THE CHANGESTRING DOES NOT INCLUDE CHANGES DONE WITHIN THE REMOTE FUNCTION CALL ITSELF, ONLY CHANGES FROM THE CLIENT\n     *\n     * See remote call documentation to know where this executes in the lifecycle\n     *\n     * @param {boolean} hasChanges - Whether or not we have applied client changes onto the server's object graph\n     * @param {CallContext} callContext - Context (number of retries etc)\n     * @param {changeString} changeString - Object of Changes - Key is [ClassName].[propertyName], Value is [changedValue] example: {'Customer.middlename': 'Karen'}, See above note\n     * @param HTTPObjs\n     *\n     * @returns {Promise<void>}\n     * @memberof Controller\n     */\n    postServerCall?(hasChanges: boolean, callContext: CallContext, changeString: ChangeString, HTTPObjs?: HTTPObjs): Promise<any>;\n\n    /**\n     * @server\n     *\n     * Callback to handle errors on a remote call.\n     *\n     * Executes after every other step in the remote call pipeline (see remote call documentation)\n     * but before retrying the call (or packaging response and sending back to client)\n     *\n     * NOTE THAT THE CHANGESTRING DOES NOT INCLUDE CHANGES DONE WITHIN THE REMOTE FUNCTION CALL ITSELF, ONLY CHANGES FROM THE CLIENT\n     *\n     * @param {ErrorType} errorType - Error type associated (error, retry, response)\n     * @param {number} remoteCallId - Id for remote call\n     * @param {extends Supertype} remoteObj - Instance for which the remote object function is called for - @TODO: revisit when we create a proper remoteable type\n     * @param {string} functionName - Name of function being called\n     * @param {CallContext} callContext - Context (number of retries etc)\n     * @param {ChangeString} changeString - Object of Changes - Key is [ClassName].[propertyName], Value is [changedValue] example: {'Customer.middlename': 'Karen'}, See above note\n     *\n     * @returns {Promise<void>}\n     * @memberof Controller\n     */\n    postServerErrorHandler?(errorType: ErrorType, remoteCallId: number, remoteObj: Supertype, functionName: string, callContext: CallContext, changeString: ChangeString): Promise<void>;\n\n    /**\n     * @server\n     *\n     * Callback before a remote function call (1st step of a remote call)\n     *\n     * We can utilize this function as a generic function handler to run before we call a\n     * remote function or before we apply changes from the client to the server\n     *\n     * We can also utilize this function to do any context-specific prep work / loading\n     * if this a subsequent try of this function due to an update conflict.\n     *\n     * See remote call documentation to know where this executes in the lifecycle\n     *\n     * @param {boolean} hasChanges - Whether or not we have applied client changes onto the server's object graph\n     * @param {PreServerCallChanges} changes - Dictionary of Objects that have been changed from the client\n     * @param {CallContext} callContext - Context (number of retries etc)\n     * @param {boolean} [forceUpdate] - Optional parameter passed in. True if this is a retry of the call based on an update conflict. False / undefined otherwise.\n     *\n     * @param functionName\n     * @param remoteCall\n     * @param isPublic\n     * @param HTTPObjs\n     * @returns {Promise<void>}\n     * @memberof Controller\n     */\n    preServerCall?(hasChanges: boolean, changes: PreServerCallChanges, callContext: CallContext, forceUpdate: undefined | boolean, functionName: string, remoteCall: RemoteCall, isPublic: boolean, HTTPObjs?: HTTPObjs ): Promise<void>;\n\n    /**\n     * This is a handler that will only be used for testing and debugging purposes\n     *\n     * When this is used, allows us to inspect the message body before we send it out in 'packageChanges'\n     *\n     * DO NOT USE THIS IN PRODUCTION CODE\n     *\n     * @param message\n     */\n    inspectMessage?(message): any;\n}\n\nexport interface Semotus {\n    setSyncState: (str) => void;\n    syncState: string;\n    getSyncState: () => any;\n    maxCallTime: number;\n    __dictionary__: any;\n    memSession: { semotus: { callStartTime: number } };\n    _injectIntoTemplate: (template) => void;\n    serializeAndGarbageCollect: () => any;\n    getMessage: (sessionId, forceMessage) => any;\n    clearPendingCalls: (sessionId) => void;\n    getChangeGroup: (type, subscriptionId) => ChangeGroup;\n    deleteChangeGroup: (type: any, subscriptionId: any) => void;\n    getChangeStatus: () => string;\n    _stashObject: (obj, template) => boolean;\n    sessionize: (obj, referencingObj) => (undefined | any);\n    _setupFunction: (propertyName, propertyValue, role, validate, serverValidation, isPublic: boolean, template) => (any);\n    _setupProperty: (propertyName, defineProperty, objectProperties, defineProperties) => void;\n    withoutChangeTracking: (cb) => void;\n    _changedValue: (obj, prop, value) => void;\n    _referencedArray: (obj, prop, arrayRef, sessionId?) => void;\n    _convertArrayReferencesToChanges: () => void;\n    MarkChangedArrayReferences: () => void;\n    _convertValue: (value) => (any[] | null);\n    getObject: (objId, template) => (any | null);\n    _applyChanges: (changes, force, subscriptionId, callContext) => (number);\n    _applyObjectChanges: (changes, rollback, obj, force) => (boolean);\n    _validateServerIncomingProperty: (obj, prop, defineProperty, newValue) => (boolean);\n    _applyPropertyChange: (changes, rollback, obj, prop, ix, oldValue, newValue, force) => (boolean);\n    _rollback: (rollback) => void;\n    _rollbackChanges: () => void;\n    _createEmptyObject: (template, objId, defineProperty, isTransient) => any;\n    inject: (template, injector) => void;\n    _queueRemoteCall: (objId, functionName, deferred, args) => void;\n    _processQueue: () => void;\n    _toTransport: (obj) => any;\n    _fromTransport: (obj) => any;\n    _extractArguments: (remoteCall) => any;\n    _trimArray: (array) => void;\n    _deleteChangeGroups: (type) => void;\n    _getSubscriptions: (sessionId?) => Subscriptions | null;\n    _getSubscription: (subscriptionId?) => Subscription;\n    cleanPrivateValues: (prop, logValue, defineProperty) => (string | any);\n    Remoteable: (Base) => () => any;\n    Bindable: (Base) => () => any;\n    Persistable: (Base) => () => any;\n    bindDecorators: (objectTemplate?) => void;\n    processMessage: (remoteCall, subscriptionId, restoreSessionCallback, req?: Request, res?: Response) => (undefined | any);\n    enableSendMessage: (value, messageCallback, sessionId) => void;\n    syncSession: (sessionId) => void;\n    restoreSession: (sessionId, savedSession: SavedSession, sendMessage: SendMessage) => boolean;\n    saveSession: (sessionId) => SavedSession;\n    setMinimumSequence: (nextObjId) => void;\n    deleteSession: (sessionId) => void;\n    createSession: (role: any, sendMessage: SendMessage, sessionId: any) => any;\n    log: (level, data) => void;\n    nextObjId: number;\n    maxClientSequence: number;\n    logLevel: number;\n    __conflictMode__: string;\n    __changeTracking__: boolean;\n    _useGettersSetters: boolean;\n    logger: any;\n    role: any;\n    currentSession: any;\n    sessions?: Sessions;\n    nextSubscriptionId: number;\n    nextSessionId: number;\n    controller: ISemotusController;\n    changeString: string;\n\n\n    _getSession(_sid?: any): Session;\n\n    subscribe(role: any): number;\n\n    setSession(sessionId: any): void;\n\n    getChanges(subscriptionId?: any): ChangeGroup;\n\n    _deleteChanges(): void;\n\n    getPendingCallCount(sessionId: any): any;\n}\n"]} |
@@ -0,1 +1,4 @@ | ||
## 4.1.0 | ||
* Added Semotus changes for `public: true` tag in remote decorator to help define public facing routes | ||
* Added ability for amorphic to pass in the request and response objects into semotus so it can pass them into the preserver and postServercall | ||
## 4.0.0 | ||
@@ -2,0 +5,0 @@ * Adding Sync State feature to limit object synchronization at stages in a session. See documentation in SyncStates.md |
{ | ||
"name": "@haventech/semotus", | ||
"version": "4.0.0", | ||
"version": "4.1.0", | ||
"description": "A subclass of supertype that synchronizes sets of objects.", | ||
@@ -42,4 +42,7 @@ "homepage": "https://github.com/haven-life/amorphic-framework", | ||
"@types/chai": "^3.4.34", | ||
"@types/express": "^4.17.11", | ||
"@types/mocha": "^2.2.39", | ||
"@types/node": "^7.10.14", | ||
"@types/node": "^14.14.37", | ||
"express": "^4.17.1", | ||
"mock-req-res": "^1.2.0", | ||
"chai": "3.x", | ||
@@ -46,0 +49,0 @@ "eslint": "3.7.x", |
@@ -122,2 +122,3 @@ /** | ||
let remoteValidator = defineProperty.serverValidation; | ||
let isPublic = !!defineProperty.public; | ||
@@ -131,2 +132,3 @@ return function (target, propertyName, descriptor) { | ||
remoteValidator, | ||
isPublic, | ||
defineProperty.target | ||
@@ -133,0 +135,0 @@ ); |
@@ -56,3 +56,3 @@ import {delay, getError, logTime} from './Utilities'; | ||
function preCallHook(payload: ProcessCallPayload, forceupdate?: boolean): boolean { | ||
const {semotus, remoteCall, callContext, session, subscriptionId, remoteCallId, restoreSessionCallback} = payload; | ||
const {semotus, remoteCall, session, callContext, HTTPObjs} = payload; | ||
semotus.logger.info( | ||
@@ -78,2 +78,6 @@ { | ||
let remoteObject = session.objects[remoteCall.id]; | ||
let isPublic = semotus.role === 'server' && remoteObject[remoteCall.name].remotePublic; | ||
return semotus.controller.preServerCall.call( | ||
@@ -84,3 +88,7 @@ semotus.controller, | ||
callContext, | ||
forceupdate | ||
forceupdate, | ||
remoteCall.name, | ||
remoteCall, | ||
isPublic, | ||
HTTPObjs | ||
); | ||
@@ -229,7 +237,7 @@ } else { | ||
async function postCallHook(payload: ProcessCallPayload, returnValue) { | ||
const {semotus, remoteCall, callContext} = payload; | ||
const {semotus, remoteCall, callContext, HTTPObjs} = payload; | ||
if (semotus.controller && semotus.controller.postServerCall) { | ||
const hasChanges: boolean = remoteCall.changes.length > 2; | ||
await semotus.controller.postServerCall.call(semotus.controller, hasChanges, callContext, semotus.changeString); | ||
await semotus.controller.postServerCall.call(semotus.controller, hasChanges, callContext, semotus.changeString, HTTPObjs); | ||
} | ||
@@ -236,0 +244,0 @@ return returnValue; |
"use strict"; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.Change = 'change'; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Types.js","sourceRoot":"","sources":["Types.ts"],"names":[],"mappings":";;AAWa,QAAA,MAAM,GAAG,QAAQ,CAAC","sourcesContent":["import {Supertype} from '@haventech/supertype';\n\nexport type Subscription = {\n    role: string;\n    log: {\n        array: ArrayGroup;\n        arrayDirty: ArrayGroup;\n        change: ChangeGroup;\n    }\n}\n\nexport const Change = 'change';\nexport type ArrayTypes = 'array' | 'arrayDirty';\nexport type ErrorType = 'error' | 'retry' | 'response';\n\nexport type CallContext = { retries: number; startTime: Date };\n\nexport type ProcessCallPayload = {\n    semotus: Semotus;\n    remoteCall: RemoteCall;\n    callContext: CallContext;\n    session: Session;\n    subscriptionId: any;\n    remoteCallId: any;\n    restoreSessionCallback?: Function;\n}\n\n/**\n *  id is the id of the object + '/' + property.\n *  Ex: there are 1 School to Many Students\n *\n *  School {\n *      students: Array<Students>\n *  }\n *\n *  In this case, one entry within this ArrayGroup may be 'server-School-1/students': [\"server-Student-3\", \"server-Student-5\"]\n *\n *  If that's all the changes then the ArrayGroup would be {'server-School-1/students': [\"server-Student-3\", \"server-Student-5\"]}\n */\nexport type ArrayGroup = { [id: string]: ArrayChanges };\n\n/**\n * An array of Id references to Supertype Objects or primitive values\n */\ntype ArrayChanges = Array<any>;\nexport type ChangeGroup = { [objId: string]: PropChanges };\ntype PropChanges = { [prop: string]: Changes };\n\n// Changes[0] is oldValue, Changes[1] is newValue\ntype oldVal = any;\ntype newVal = any;\ntype Changes = [oldVal, newVal][];\n\nexport type Subscriptions = { [key: string]: Subscription };\n\nexport type RemoteCall = {\n    remoteCallId: any;\n    id: any;\n    changes: string; // The string is of type ChangeGroup\n    name: string;\n    sequence: any;\n}\n\nexport type Session = {\n    subscriptions: Subscriptions;\n    sendMessage: SendMessage;\n    sendMessageEnabled: boolean;\n    remoteCalls: Array<any>;\n    pendingRemoteCalls: any;\n    nextPendingRemoteCallId: number;\n    nextSaveSessionId: number;\n    savedSessionId: number;\n    nextSubscriptionId: number;\n    objects: any;\n    nextObjId: number;\n    dispenseNextId: null; // not used anywhere\n}\n\n\nexport type Sessions = { [sessionId: number]: Session };\n\nexport type SendMessage = (message: any) => void;\n\nexport type SavedSession = {\n    revision: number;\n    data: string; // Serialized Session data\n    callCount: number;\n    referenced: number;\n};\n\nexport interface RemoteableClass extends Supertype {\n    syncStates: Array<string>;\n    __toClient__: boolean;\n    __toServer__: boolean;\n}\n\nexport type ChangeString = { [key: string]: string };\ntype PreServerCallChanges = { [key: string]: boolean };\n\nexport type ControllerSyncState = {\n    scope: '+' | '*' | '-'\n    state: string\n}\n\nexport interface ISemotusController {\n    __template__: any;\n    syncState?: ControllerSyncState\n\n    /**\n     * @server\n     *\n     * Callback after a successful remote function call (just the application of changes and the execution of the function call)\n     * Note that this doesn't mean we can't error out on this or subsequent steps of a remote call.\n     *\n     * We can utilize this function as a generic function handler to run after we have successfully called a remote function.\n     * One such use may be to see the changes that were applied from the client\n     *\n     * NOTE THAT THE CHANGESTRING DOES NOT INCLUDE CHANGES DONE WITHIN THE REMOTE FUNCTION CALL ITSELF, ONLY CHANGES FROM THE CLIENT\n     *\n     * See remote call documentation to know where this executes in the lifecycle\n     *\n     * @param {boolean} hasChanges - Whether or not we have applied client changes onto the server's object graph\n     * @param {CallContext} callContext - Context (number of retries etc)\n     * @param {changeString} changeString - Object of Changes - Key is [ClassName].[propertyName], Value is [changedValue] example: {'Customer.middlename': 'Karen'}, See above note\n     *\n     * @returns {Promise<void>}\n     * @memberof Controller\n     */\n    postServerCall?(hasChanges: boolean, callContext: CallContext, changeString: ChangeString): Promise<any>;\n\n    /**\n     * @server\n     *\n     * Callback to handle errors on a remote call.\n     *\n     * Executes after every other step in the remote call pipeline (see remote call documentation)\n     * but before retrying the call (or packaging response and sending back to client)\n     *\n     * NOTE THAT THE CHANGESTRING DOES NOT INCLUDE CHANGES DONE WITHIN THE REMOTE FUNCTION CALL ITSELF, ONLY CHANGES FROM THE CLIENT\n     *\n     * @param {ErrorType} errorType - Error type associated (error, retry, response)\n     * @param {number} remoteCallId - Id for remote call\n     * @param {extends Supertype} remoteObj - Instance for which the remote object function is called for - @TODO: revisit when we create a proper remoteable type\n     * @param {string} functionName - Name of function being called\n     * @param {CallContext} callContext - Context (number of retries etc)\n     * @param {ChangeString} changeString - Object of Changes - Key is [ClassName].[propertyName], Value is [changedValue] example: {'Customer.middlename': 'Karen'}, See above note\n     *\n     * @returns {Promise<void>}\n     * @memberof Controller\n     */\n    postServerErrorHandler?(errorType: ErrorType, remoteCallId: number, remoteObj: Supertype, functionName: string, callContext: CallContext, changeString: ChangeString): Promise<void>;\n\n    /**\n     * @server\n     *\n     * Callback before a remote function call (1st step of a remote call)\n     *\n     * We can utilize this function as a generic function handler to run before we call a\n     * remote function or before we apply changes from the client to the server\n     *\n     * We can also utilize this function to do any context-specific prep work / loading\n     * if this a subsequent try of this function due to an update conflict.\n     *\n     * See remote call documentation to know where this executes in the lifecycle\n     *\n     * @param {boolean} hasChanges - Whether or not we have applied client changes onto the server's object graph\n     * @param {PreServerCallChanges} changes - Dictionary of Objects that have been changed from the client\n     * @param {CallContext} callContext - Context (number of retries etc)\n     * @param {boolean} [forceUpdate] - Optional parameter passed in. True if this is a retry of the call based on an update conflict. False / undefined otherwise.\n     *\n     * @returns {Promise<void>}\n     * @memberof Controller\n     */\n    preServerCall?(hasChanges: boolean, changes: PreServerCallChanges, callContext: CallContext, forceUpdate?: boolean): Promise<void>;\n\n    /**\n     * This is a handler that will only be used for testing and debugging purposes\n     *\n     * When this is used, allows us to inspect the message body before we send it out in 'packageChanges'\n     *\n     * DO NOT USE THIS IN PRODUCTION CODE\n     *\n     * @param message\n     */\n    inspectMessage?(message): any;\n}\n\nexport interface Semotus {\n    setSyncState: (str) => void;\n    syncState: string;\n    getSyncState: () => any;\n    maxCallTime: number;\n    __dictionary__: any;\n    memSession: { semotus: { callStartTime: number } };\n    _injectIntoTemplate: (template) => void;\n    serializeAndGarbageCollect: () => any;\n    getMessage: (sessionId, forceMessage) => any;\n    clearPendingCalls: (sessionId) => void;\n    getChangeGroup: (type, subscriptionId) => ChangeGroup;\n    deleteChangeGroup: (type: any, subscriptionId: any) => void;\n    getChangeStatus: () => string;\n    _stashObject: (obj, template) => boolean;\n    sessionize: (obj, referencingObj) => (undefined | any);\n    _setupFunction: (propertyName, propertyValue, role, validate, serverValidation, template) => (any);\n    _setupProperty: (propertyName, defineProperty, objectProperties, defineProperties) => void;\n    withoutChangeTracking: (cb) => void;\n    _changedValue: (obj, prop, value) => void;\n    _referencedArray: (obj, prop, arrayRef, sessionId?) => void;\n    _convertArrayReferencesToChanges: () => void;\n    MarkChangedArrayReferences: () => void;\n    _convertValue: (value) => (any[] | null);\n    getObject: (objId, template) => (any | null);\n    _applyChanges: (changes, force, subscriptionId, callContext) => (number);\n    _applyObjectChanges: (changes, rollback, obj, force) => (boolean);\n    _validateServerIncomingProperty: (obj, prop, defineProperty, newValue) => (boolean);\n    _applyPropertyChange: (changes, rollback, obj, prop, ix, oldValue, newValue, force) => (boolean);\n    _rollback: (rollback) => void;\n    _rollbackChanges: () => void;\n    _createEmptyObject: (template, objId, defineProperty, isTransient) => any;\n    inject: (template, injector) => void;\n    _queueRemoteCall: (objId, functionName, deferred, args) => void;\n    _processQueue: () => void;\n    _toTransport: (obj) => any;\n    _fromTransport: (obj) => any;\n    _extractArguments: (remoteCall) => any;\n    _trimArray: (array) => void;\n    _deleteChangeGroups: (type) => void;\n    _getSubscriptions: (sessionId?) => Subscriptions | null;\n    _getSubscription: (subscriptionId?) => Subscription;\n    cleanPrivateValues: (prop, logValue, defineProperty) => (string | any);\n    Remoteable: (Base) => () => any;\n    Bindable: (Base) => () => any;\n    Persistable: (Base) => () => any;\n    bindDecorators: (objectTemplate?) => void;\n    processMessage: (remoteCall, subscriptionId, restoreSessionCallback) => (undefined | any);\n    enableSendMessage: (value, messageCallback, sessionId) => void;\n    syncSession: (sessionId) => void;\n    restoreSession: (sessionId, savedSession: SavedSession, sendMessage: SendMessage) => boolean;\n    saveSession: (sessionId) => SavedSession;\n    setMinimumSequence: (nextObjId) => void;\n    deleteSession: (sessionId) => void;\n    createSession: (role: any, sendMessage: SendMessage, sessionId: any) => any;\n    log: (level, data) => void;\n    nextObjId: number;\n    maxClientSequence: number;\n    logLevel: number;\n    __conflictMode__: string;\n    __changeTracking__: boolean;\n    _useGettersSetters: boolean;\n    logger: any;\n    role: any;\n    currentSession: any;\n    sessions?: Sessions;\n    nextSubscriptionId: number;\n    nextSessionId: number;\n    controller: ISemotusController;\n    changeString: string;\n\n\n    _getSession(_sid?: any): Session;\n\n    subscribe(role: any): number;\n\n    setSession(sessionId: any): void;\n\n    getChanges(subscriptionId?: any): ChangeGroup;\n\n    _deleteChanges(): void;\n\n    getPendingCallCount(sessionId: any): any;\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Types.js","sourceRoot":"","sources":["Types.ts"],"names":[],"mappings":";;AAaa,QAAA,MAAM,GAAG,QAAQ,CAAC","sourcesContent":["import {Supertype} from '@haventech/supertype';\nimport {Request, Response} from 'express';\nexport type HTTPObjs = {request: Request, response: Response};\n\nexport type Subscription = {\n    role: string;\n    log: {\n        array: ArrayGroup;\n        arrayDirty: ArrayGroup;\n        change: ChangeGroup;\n    }\n}\n\nexport const Change = 'change';\nexport type ArrayTypes = 'array' | 'arrayDirty';\nexport type ErrorType = 'error' | 'retry' | 'response';\n\nexport type CallContext = { retries: number; startTime: Date };\n\nexport type ProcessCallPayload = {\n    semotus: Semotus;\n    remoteCall: RemoteCall;\n    callContext: CallContext;\n    session: Session;\n    subscriptionId: any;\n    remoteCallId: any;\n    restoreSessionCallback?: Function;\n    HTTPObjs?: HTTPObjs;\n}\n\n/**\n *  id is the id of the object + '/' + property.\n *  Ex: there are 1 School to Many Students\n *\n *  School {\n *      students: Array<Students>\n *  }\n *\n *  In this case, one entry within this ArrayGroup may be 'server-School-1/students': [\"server-Student-3\", \"server-Student-5\"]\n *\n *  If that's all the changes then the ArrayGroup would be {'server-School-1/students': [\"server-Student-3\", \"server-Student-5\"]}\n */\nexport type ArrayGroup = { [id: string]: ArrayChanges };\n\n/**\n * An array of Id references to Supertype Objects or primitive values\n */\ntype ArrayChanges = Array<any>;\nexport type ChangeGroup = { [objId: string]: PropChanges };\ntype PropChanges = { [prop: string]: Changes };\n\n// Changes[0] is oldValue, Changes[1] is newValue\ntype oldVal = any;\ntype newVal = any;\ntype Changes = [oldVal, newVal][];\n\nexport type Subscriptions = { [key: string]: Subscription };\n\nexport type RemoteCall = {\n    remoteCallId: any;\n    id: any;\n    changes: string; // The string is of type ChangeGroup\n    name: string;\n    sequence: any;\n    type: any;\n    sync: any;\n    value: any;\n}\n\nexport type Session = {\n    subscriptions: Subscriptions;\n    sendMessage: SendMessage;\n    sendMessageEnabled: boolean;\n    remoteCalls: Array<any>;\n    pendingRemoteCalls: any;\n    nextPendingRemoteCallId: number;\n    nextSaveSessionId: number;\n    savedSessionId: number;\n    nextSubscriptionId: number;\n    objects: any;\n    nextObjId: number;\n    dispenseNextId: null; // not used anywhere\n}\n\n\nexport type Sessions = { [sessionId: number]: Session };\n\nexport type SendMessage = (message: any) => void;\n\nexport type SavedSession = {\n    revision: number;\n    data: string; // Serialized Session data\n    callCount: number;\n    referenced: number;\n};\n\nexport interface RemoteableClass extends Supertype {\n    syncStates: Array<string>;\n    __toClient__: boolean;\n    __toServer__: boolean;\n}\n\nexport type ChangeString = { [key: string]: string };\ntype PreServerCallChanges = { [key: string]: boolean };\n\nexport type ControllerSyncState = {\n    scope: '+' | '*' | '-'\n    state: string\n}\n\nexport interface ISemotusController {\n    __template__: any;\n    syncState?: ControllerSyncState\n\n    /**\n     * @server\n     *\n     * Callback after a successful remote function call (just the application of changes and the execution of the function call)\n     * Note that this doesn't mean we can't error out on this or subsequent steps of a remote call.\n     *\n     * We can utilize this function as a generic function handler to run after we have successfully called a remote function.\n     * One such use may be to see the changes that were applied from the client\n     *\n     * NOTE THAT THE CHANGESTRING DOES NOT INCLUDE CHANGES DONE WITHIN THE REMOTE FUNCTION CALL ITSELF, ONLY CHANGES FROM THE CLIENT\n     *\n     * See remote call documentation to know where this executes in the lifecycle\n     *\n     * @param {boolean} hasChanges - Whether or not we have applied client changes onto the server's object graph\n     * @param {CallContext} callContext - Context (number of retries etc)\n     * @param {changeString} changeString - Object of Changes - Key is [ClassName].[propertyName], Value is [changedValue] example: {'Customer.middlename': 'Karen'}, See above note\n     * @param HTTPObjs\n     *\n     * @returns {Promise<void>}\n     * @memberof Controller\n     */\n    postServerCall?(hasChanges: boolean, callContext: CallContext, changeString: ChangeString, HTTPObjs?: HTTPObjs): Promise<any>;\n\n    /**\n     * @server\n     *\n     * Callback to handle errors on a remote call.\n     *\n     * Executes after every other step in the remote call pipeline (see remote call documentation)\n     * but before retrying the call (or packaging response and sending back to client)\n     *\n     * NOTE THAT THE CHANGESTRING DOES NOT INCLUDE CHANGES DONE WITHIN THE REMOTE FUNCTION CALL ITSELF, ONLY CHANGES FROM THE CLIENT\n     *\n     * @param {ErrorType} errorType - Error type associated (error, retry, response)\n     * @param {number} remoteCallId - Id for remote call\n     * @param {extends Supertype} remoteObj - Instance for which the remote object function is called for - @TODO: revisit when we create a proper remoteable type\n     * @param {string} functionName - Name of function being called\n     * @param {CallContext} callContext - Context (number of retries etc)\n     * @param {ChangeString} changeString - Object of Changes - Key is [ClassName].[propertyName], Value is [changedValue] example: {'Customer.middlename': 'Karen'}, See above note\n     *\n     * @returns {Promise<void>}\n     * @memberof Controller\n     */\n    postServerErrorHandler?(errorType: ErrorType, remoteCallId: number, remoteObj: Supertype, functionName: string, callContext: CallContext, changeString: ChangeString): Promise<void>;\n\n    /**\n     * @server\n     *\n     * Callback before a remote function call (1st step of a remote call)\n     *\n     * We can utilize this function as a generic function handler to run before we call a\n     * remote function or before we apply changes from the client to the server\n     *\n     * We can also utilize this function to do any context-specific prep work / loading\n     * if this a subsequent try of this function due to an update conflict.\n     *\n     * See remote call documentation to know where this executes in the lifecycle\n     *\n     * @param {boolean} hasChanges - Whether or not we have applied client changes onto the server's object graph\n     * @param {PreServerCallChanges} changes - Dictionary of Objects that have been changed from the client\n     * @param {CallContext} callContext - Context (number of retries etc)\n     * @param {boolean} [forceUpdate] - Optional parameter passed in. True if this is a retry of the call based on an update conflict. False / undefined otherwise.\n     *\n     * @param functionName\n     * @param remoteCall\n     * @param isPublic\n     * @param HTTPObjs\n     * @returns {Promise<void>}\n     * @memberof Controller\n     */\n    preServerCall?(hasChanges: boolean, changes: PreServerCallChanges, callContext: CallContext, forceUpdate: undefined | boolean, functionName: string, remoteCall: RemoteCall, isPublic: boolean, HTTPObjs?: HTTPObjs ): Promise<void>;\n\n    /**\n     * This is a handler that will only be used for testing and debugging purposes\n     *\n     * When this is used, allows us to inspect the message body before we send it out in 'packageChanges'\n     *\n     * DO NOT USE THIS IN PRODUCTION CODE\n     *\n     * @param message\n     */\n    inspectMessage?(message): any;\n}\n\nexport interface Semotus {\n    setSyncState: (str) => void;\n    syncState: string;\n    getSyncState: () => any;\n    maxCallTime: number;\n    __dictionary__: any;\n    memSession: { semotus: { callStartTime: number } };\n    _injectIntoTemplate: (template) => void;\n    serializeAndGarbageCollect: () => any;\n    getMessage: (sessionId, forceMessage) => any;\n    clearPendingCalls: (sessionId) => void;\n    getChangeGroup: (type, subscriptionId) => ChangeGroup;\n    deleteChangeGroup: (type: any, subscriptionId: any) => void;\n    getChangeStatus: () => string;\n    _stashObject: (obj, template) => boolean;\n    sessionize: (obj, referencingObj) => (undefined | any);\n    _setupFunction: (propertyName, propertyValue, role, validate, serverValidation, isPublic: boolean, template) => (any);\n    _setupProperty: (propertyName, defineProperty, objectProperties, defineProperties) => void;\n    withoutChangeTracking: (cb) => void;\n    _changedValue: (obj, prop, value) => void;\n    _referencedArray: (obj, prop, arrayRef, sessionId?) => void;\n    _convertArrayReferencesToChanges: () => void;\n    MarkChangedArrayReferences: () => void;\n    _convertValue: (value) => (any[] | null);\n    getObject: (objId, template) => (any | null);\n    _applyChanges: (changes, force, subscriptionId, callContext) => (number);\n    _applyObjectChanges: (changes, rollback, obj, force) => (boolean);\n    _validateServerIncomingProperty: (obj, prop, defineProperty, newValue) => (boolean);\n    _applyPropertyChange: (changes, rollback, obj, prop, ix, oldValue, newValue, force) => (boolean);\n    _rollback: (rollback) => void;\n    _rollbackChanges: () => void;\n    _createEmptyObject: (template, objId, defineProperty, isTransient) => any;\n    inject: (template, injector) => void;\n    _queueRemoteCall: (objId, functionName, deferred, args) => void;\n    _processQueue: () => void;\n    _toTransport: (obj) => any;\n    _fromTransport: (obj) => any;\n    _extractArguments: (remoteCall) => any;\n    _trimArray: (array) => void;\n    _deleteChangeGroups: (type) => void;\n    _getSubscriptions: (sessionId?) => Subscriptions | null;\n    _getSubscription: (subscriptionId?) => Subscription;\n    cleanPrivateValues: (prop, logValue, defineProperty) => (string | any);\n    Remoteable: (Base) => () => any;\n    Bindable: (Base) => () => any;\n    Persistable: (Base) => () => any;\n    bindDecorators: (objectTemplate?) => void;\n    processMessage: (remoteCall, subscriptionId, restoreSessionCallback, req?: Request, res?: Response) => (undefined | any);\n    enableSendMessage: (value, messageCallback, sessionId) => void;\n    syncSession: (sessionId) => void;\n    restoreSession: (sessionId, savedSession: SavedSession, sendMessage: SendMessage) => boolean;\n    saveSession: (sessionId) => SavedSession;\n    setMinimumSequence: (nextObjId) => void;\n    deleteSession: (sessionId) => void;\n    createSession: (role: any, sendMessage: SendMessage, sessionId: any) => any;\n    log: (level, data) => void;\n    nextObjId: number;\n    maxClientSequence: number;\n    logLevel: number;\n    __conflictMode__: string;\n    __changeTracking__: boolean;\n    _useGettersSetters: boolean;\n    logger: any;\n    role: any;\n    currentSession: any;\n    sessions?: Sessions;\n    nextSubscriptionId: number;\n    nextSessionId: number;\n    controller: ISemotusController;\n    changeString: string;\n\n\n    _getSession(_sid?: any): Session;\n\n    subscribe(role: any): number;\n\n    setSession(sessionId: any): void;\n\n    getChanges(subscriptionId?: any): ChangeGroup;\n\n    _deleteChanges(): void;\n\n    getPendingCallCount(sessionId: any): any;\n}\n"]} |
import {Supertype} from '@haventech/supertype'; | ||
import {Request, Response} from 'express'; | ||
export type HTTPObjs = {request: Request, response: Response}; | ||
@@ -26,2 +28,3 @@ export type Subscription = { | ||
restoreSessionCallback?: Function; | ||
HTTPObjs?: HTTPObjs; | ||
} | ||
@@ -63,2 +66,5 @@ | ||
sequence: any; | ||
type: any; | ||
sync: any; | ||
value: any; | ||
} | ||
@@ -127,2 +133,3 @@ | ||
* @param {changeString} changeString - Object of Changes - Key is [ClassName].[propertyName], Value is [changedValue] example: {'Customer.middlename': 'Karen'}, See above note | ||
* @param HTTPObjs | ||
* | ||
@@ -132,3 +139,3 @@ * @returns {Promise<void>} | ||
*/ | ||
postServerCall?(hasChanges: boolean, callContext: CallContext, changeString: ChangeString): Promise<any>; | ||
postServerCall?(hasChanges: boolean, callContext: CallContext, changeString: ChangeString, HTTPObjs?: HTTPObjs): Promise<any>; | ||
@@ -175,6 +182,10 @@ /** | ||
* | ||
* @param functionName | ||
* @param remoteCall | ||
* @param isPublic | ||
* @param HTTPObjs | ||
* @returns {Promise<void>} | ||
* @memberof Controller | ||
*/ | ||
preServerCall?(hasChanges: boolean, changes: PreServerCallChanges, callContext: CallContext, forceUpdate?: boolean): Promise<void>; | ||
preServerCall?(hasChanges: boolean, changes: PreServerCallChanges, callContext: CallContext, forceUpdate: undefined | boolean, functionName: string, remoteCall: RemoteCall, isPublic: boolean, HTTPObjs?: HTTPObjs ): Promise<void>; | ||
@@ -209,3 +220,3 @@ /** | ||
sessionize: (obj, referencingObj) => (undefined | any); | ||
_setupFunction: (propertyName, propertyValue, role, validate, serverValidation, template) => (any); | ||
_setupFunction: (propertyName, propertyValue, role, validate, serverValidation, isPublic: boolean, template) => (any); | ||
_setupProperty: (propertyName, defineProperty, objectProperties, defineProperties) => void; | ||
@@ -241,3 +252,3 @@ withoutChangeTracking: (cb) => void; | ||
bindDecorators: (objectTemplate?) => void; | ||
processMessage: (remoteCall, subscriptionId, restoreSessionCallback) => (undefined | any); | ||
processMessage: (remoteCall, subscriptionId, restoreSessionCallback, req?: Request, res?: Response) => (undefined | any); | ||
enableSendMessage: (value, messageCallback, sessionId) => void; | ||
@@ -244,0 +255,0 @@ syncSession: (sessionId) => void; |
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
979558
39
9868
15