@aerokit/sdk 12.57.0 → 12.58.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -39,16 +39,45 @@ __export(decorators_exports, {
|
|
|
39
39
|
});
|
|
40
40
|
module.exports = __toCommonJS(decorators_exports);
|
|
41
41
|
var rs = __toESM(require("@aerokit/sdk/http/rs"));
|
|
42
|
+
const ROUTES_KEY = /* @__PURE__ */ Symbol.for("dirigible.controller.routes");
|
|
43
|
+
const GLOBAL_ROUTES = globalThis[ROUTES_KEY] ?? (globalThis[ROUTES_KEY] = []);
|
|
42
44
|
const router = rs.service();
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
45
|
+
function Controller(ctr) {
|
|
46
|
+
const instance = new ctr();
|
|
47
|
+
const routes = GLOBAL_ROUTES;
|
|
48
|
+
for (const route of routes) {
|
|
49
|
+
const fn = instance[route.propertyKey.name];
|
|
50
|
+
if (typeof fn === "function") {
|
|
51
|
+
((fn2, instance2) => {
|
|
52
|
+
router.resource(route.path)[route.method]((ctx, req, res) => {
|
|
53
|
+
const body = req.json ? req.json() : null;
|
|
54
|
+
const result = fn2.call(instance2, body, ctx, req, res);
|
|
55
|
+
if (result !== void 0) {
|
|
56
|
+
res.json(result);
|
|
57
|
+
}
|
|
58
|
+
});
|
|
59
|
+
})(fn, instance);
|
|
60
|
+
}
|
|
61
|
+
}
|
|
46
62
|
router.execute();
|
|
63
|
+
GLOBAL_ROUTES.length = 0;
|
|
47
64
|
}
|
|
48
65
|
function Documentation(documentation) {
|
|
49
66
|
return function(value, context) {
|
|
50
67
|
};
|
|
51
68
|
}
|
|
69
|
+
function createRequestDecorator(httpMethod) {
|
|
70
|
+
return function(path) {
|
|
71
|
+
return function(target, propertyKey) {
|
|
72
|
+
GLOBAL_ROUTES.push({
|
|
73
|
+
controller: target.constructor,
|
|
74
|
+
method: httpMethod,
|
|
75
|
+
path: path || "/",
|
|
76
|
+
propertyKey
|
|
77
|
+
});
|
|
78
|
+
};
|
|
79
|
+
};
|
|
80
|
+
}
|
|
52
81
|
const Get = createRequestDecorator("get");
|
|
53
82
|
const Post = createRequestDecorator("post");
|
|
54
83
|
const Put = createRequestDecorator("put");
|
|
@@ -56,21 +85,4 @@ const Patch = createRequestDecorator("patch");
|
|
|
56
85
|
const Delete = createRequestDecorator("delete");
|
|
57
86
|
const Head = createRequestDecorator("head");
|
|
58
87
|
const Options = createRequestDecorator("options");
|
|
59
|
-
|
|
60
|
-
return function(path, consumesMimeTypes = ["*/*"], producesMimeTypes = ["application/json"]) {
|
|
61
|
-
return function(target, propertyKey, descriptor) {
|
|
62
|
-
const handler = descriptor ? descriptor.value : target;
|
|
63
|
-
router.resource(path)[httpMethod]((ctx, req, res) => {
|
|
64
|
-
handleRequest(req, res, ctx, handler);
|
|
65
|
-
}).consumes(consumesMimeTypes).produces(producesMimeTypes);
|
|
66
|
-
};
|
|
67
|
-
};
|
|
68
|
-
}
|
|
69
|
-
function handleRequest(req, res, ctx, handler) {
|
|
70
|
-
const body = req.json();
|
|
71
|
-
const maybeResponseBody = handler.apply(instance || {}, [body, ctx, req, res]);
|
|
72
|
-
if (maybeResponseBody) {
|
|
73
|
-
res.json(maybeResponseBody);
|
|
74
|
-
}
|
|
75
|
-
}
|
|
76
|
-
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vc3JjL2h0dHAvZGVjb3JhdG9ycy50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiaW1wb3J0ICogYXMgcnMgZnJvbSBcIkBhZXJva2l0L3Nkay9odHRwL3JzXCJcblxuY29uc3Qgcm91dGVyID0gcnMuc2VydmljZSgpO1xubGV0IGluc3RhbmNlID0gbnVsbDtcblxuZXhwb3J0IGZ1bmN0aW9uIENvbnRyb2xsZXIoY3RyOiB7IG5ldygpIH0sIGNvbnRleHQ6IENsYXNzRGVjb3JhdG9yQ29udGV4dCk6IHZvaWQge1xuICAgIGluc3RhbmNlID0gbmV3IGN0cigpO1xuICAgIHJvdXRlci5leGVjdXRlKCk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBEb2N1bWVudGF0aW9uKGRvY3VtZW50YXRpb246IHN0cmluZykge1xuICByZXR1cm4gZnVuY3Rpb24gKFxuICAgIHZhbHVlOiBhbnksXG4gICAgY29udGV4dDogQ2xhc3NEZWNvcmF0b3JDb250ZXh0IHwgQ2xhc3NGaWVsZERlY29yYXRvckNvbnRleHQgfCBDbGFzc01ldGhvZERlY29yYXRvckNvbnRleHRcbiAgKSB7fTtcbn1cblxuZXhwb3J0IGNvbnN0IEdldCA9IGNyZWF0ZVJlcXVlc3REZWNvcmF0b3IoXCJnZXRcIilcbmV4cG9ydCBjb25zdCBQb3N0ID0gY3JlYXRlUmVxdWVzdERlY29yYXRvcihcInBvc3RcIilcbmV4cG9ydCBjb25zdCBQdXQgPSBjcmVhdGVSZXF1ZXN0RGVjb3JhdG9yKFwicHV0XCIpXG5leHBvcnQgY29uc3QgUGF0Y2ggPSBjcmVhdGVSZXF1ZXN0RGVjb3JhdG9yKFwicGF0Y2hcIilcbmV4cG9ydCBjb25zdCBEZWxldGUgPSBjcmVhdGVSZXF1ZXN0RGVjb3JhdG9yKFwiZGVsZXRlXCIpXG5leHBvcnQgY29uc3QgSGVhZCA9IGNyZWF0ZVJlcXVlc3REZWNvcmF0b3IoXCJoZWFkXCIpXG5leHBvcnQgY29uc3QgT3B0aW9ucyA9IGNyZWF0ZVJlcXVlc3REZWNvcmF0b3IoXCJvcHRpb25zXCIpXG5cbmZ1bmN0aW9uIGNyZWF0ZVJlcXVlc3REZWNvcmF0b3IoaHR0cE1ldGhvZCkge1xuICAgIHJldHVybiBmdW5jdGlvbiAocGF0aDogc3RyaW5nLCBjb25zdW1lc01pbWVUeXBlczogdW5kZWZpbmVkIHwgc3RyaW5nIHwgc3RyaW5nW10gPSBbJyovKiddLCBwcm9kdWNlc01pbWVUeXBlczogdW5kZWZpbmVkIHwgc3RyaW5nIHwgc3RyaW5nW10gPSBbJ2FwcGxpY2F0aW9uL2pzb24nXSk6IGFueSB7XG4gICAgICAgIHJldHVybiBmdW5jdGlvbiAodGFyZ2V0LCBwcm9wZXJ0eUtleSwgZGVzY3JpcHRvcikge1xuICAgICAgICAgICAgY29uc3QgaGFuZGxlciA9IGRlc2NyaXB0b3IgPyBkZXNjcmlwdG9yLnZhbHVlIDogdGFyZ2V0O1xuICAgICAgICAgICAgcm91dGVyLnJlc291cmNlKHBhdGgpW2h0dHBNZXRob2RdKChjdHgsIHJlcSwgcmVzKSA9PiB7XG4gICAgICAgICAgICAgICAgaGFuZGxlUmVxdWVzdChyZXEsIHJlcywgY3R4LCBoYW5kbGVyKTtcbiAgICAgICAgICAgIH0pLmNvbnN1bWVzKGNvbnN1bWVzTWltZVR5cGVzKS5wcm9kdWNlcyhwcm9kdWNlc01pbWVUeXBlcyk7XG4gICAgICAgIH07XG4gICAgfVxufVxuXG5mdW5jdGlvbiBoYW5kbGVSZXF1ZXN0KHJlcSwgcmVzLCBjdHgsIGhhbmRsZXIpIHtcbiAgICBjb25zdCBib2R5ID0gcmVxLmpzb24oKTtcbiAgICBjb25zdCBtYXliZVJlc3BvbnNlQm9keSA9IGhhbmRsZXIuYXBwbHkoaW5zdGFuY2UgfHwge30sIFtib2R5LCBjdHgsIHJlcSwgcmVzXSk7XG4gICAgaWYgKG1heWJlUmVzcG9uc2VCb2R5KSB7XG4gICAgICAgIHJlcy5qc29uKG1heWJlUmVzcG9uc2VCb2R5KTtcbiAgICB9XG59XG4iXSwKICAibWFwcGluZ3MiOiAiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBLFNBQW9CO0FBRXBCLE1BQU0sU0FBUyxHQUFHLFFBQVE7QUFDMUIsSUFBSSxXQUFXO0FBRVIsU0FBUyxXQUFXLEtBQWdCLFNBQXNDO0FBQzdFLGFBQVcsSUFBSSxJQUFJO0FBQ25CLFNBQU8sUUFBUTtBQUNuQjtBQUVPLFNBQVMsY0FBYyxlQUF1QjtBQUNuRCxTQUFPLFNBQ0wsT0FDQSxTQUNBO0FBQUEsRUFBQztBQUNMO0FBRU8sTUFBTSxNQUFNLHVCQUF1QixLQUFLO0FBQ3hDLE1BQU0sT0FBTyx1QkFBdUIsTUFBTTtBQUMxQyxNQUFNLE1BQU0sdUJBQXVCLEtBQUs7QUFDeEMsTUFBTSxRQUFRLHVCQUF1QixPQUFPO0FBQzVDLE1BQU0sU0FBUyx1QkFBdUIsUUFBUTtBQUM5QyxNQUFNLE9BQU8sdUJBQXVCLE1BQU07QUFDMUMsTUFBTSxVQUFVLHVCQUF1QixTQUFTO0FBRXZELFNBQVMsdUJBQXVCLFlBQVk7QUFDeEMsU0FBTyxTQUFVLE1BQWMsb0JBQW1ELENBQUMsS0FBSyxHQUFHLG9CQUFtRCxDQUFDLGtCQUFrQixHQUFRO0FBQ3JLLFdBQU8sU0FBVSxRQUFRLGFBQWEsWUFBWTtBQUM5QyxZQUFNLFVBQVUsYUFBYSxXQUFXLFFBQVE7QUFDaEQsYUFBTyxTQUFTLElBQUksRUFBRSxVQUFVLEVBQUUsQ0FBQyxLQUFLLEtBQUssUUFBUTtBQUNqRCxzQkFBYyxLQUFLLEtBQUssS0FBSyxPQUFPO0FBQUEsTUFDeEMsQ0FBQyxFQUFFLFNBQVMsaUJBQWlCLEVBQUUsU0FBUyxpQkFBaUI7QUFBQSxJQUM3RDtBQUFBLEVBQ0o7QUFDSjtBQUVBLFNBQVMsY0FBYyxLQUFLLEtBQUssS0FBSyxTQUFTO0FBQzNDLFFBQU0sT0FBTyxJQUFJLEtBQUs7QUFDdEIsUUFBTSxvQkFBb0IsUUFBUSxNQUFNLFlBQVksQ0FBQyxHQUFHLENBQUMsTUFBTSxLQUFLLEtBQUssR0FBRyxDQUFDO0FBQzdFLE1BQUksbUJBQW1CO0FBQ25CLFFBQUksS0FBSyxpQkFBaUI7QUFBQSxFQUM5QjtBQUNKOyIsCiAgIm5hbWVzIjogW10KfQo=
|
|
88
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vc3JjL2h0dHAvZGVjb3JhdG9ycy50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiaW1wb3J0ICogYXMgcnMgZnJvbSBcIkBhZXJva2l0L3Nkay9odHRwL3JzXCI7XG5cbmNvbnN0IFJPVVRFU19LRVkgPSBTeW1ib2wuZm9yKFwiZGlyaWdpYmxlLmNvbnRyb2xsZXIucm91dGVzXCIpO1xuXG5jb25zdCBHTE9CQUxfUk9VVEVTOiBhbnlbXSA9XG4gICAgKGdsb2JhbFRoaXMgYXMgYW55KVtST1VURVNfS0VZXSA/P1xuICAgICgoZ2xvYmFsVGhpcyBhcyBhbnkpW1JPVVRFU19LRVldID0gW10pO1xuXG5jb25zdCByb3V0ZXIgPSBycy5zZXJ2aWNlKCk7XG5cbmV4cG9ydCBmdW5jdGlvbiBDb250cm9sbGVyKGN0cjogeyBuZXcoKTogYW55IH0pIHtcbiAgICBjb25zdCBpbnN0YW5jZSA9IG5ldyBjdHIoKTtcbiAgICBjb25zdCByb3V0ZXMgPSBHTE9CQUxfUk9VVEVTO1xuICAgIGZvciAoY29uc3Qgcm91dGUgb2Ygcm91dGVzKSB7XG4gICAgICAgIGNvbnN0IGZuID0gaW5zdGFuY2Vbcm91dGUucHJvcGVydHlLZXkubmFtZV07XG4gICAgICAgIGlmICh0eXBlb2YgZm4gPT09IFwiZnVuY3Rpb25cIikge1xuXHRcdFx0KChmbiwgaW5zdGFuY2UpID0+IHtcblx0XHQgICAgICAgIHJvdXRlci5yZXNvdXJjZShyb3V0ZS5wYXRoKVtyb3V0ZS5tZXRob2RdKChjdHgsIHJlcSwgcmVzKSA9PiB7XG5cdFx0ICAgICAgICAgICAgY29uc3QgYm9keSA9IHJlcS5qc29uID8gcmVxLmpzb24oKSA6IG51bGw7XG5cdFx0ICAgICAgICAgICAgY29uc3QgcmVzdWx0ID0gZm4uY2FsbChpbnN0YW5jZSwgYm9keSwgY3R4LCByZXEsIHJlcyk7XG5cdFx0ICAgICAgICAgICAgaWYgKHJlc3VsdCAhPT0gdW5kZWZpbmVkKSB7XG5cdFx0ICAgICAgICAgICAgICAgIHJlcy5qc29uKHJlc3VsdCk7XG5cdFx0ICAgICAgICAgICAgfVxuXHRcdCAgICAgICAgfSk7XG5cdFx0ICAgIH0pKGZuLCBpbnN0YW5jZSk7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICByb3V0ZXIuZXhlY3V0ZSgpO1xuXHRHTE9CQUxfUk9VVEVTLmxlbmd0aCA9IDA7XG59XG5cblxuZXhwb3J0IGZ1bmN0aW9uIERvY3VtZW50YXRpb24oZG9jdW1lbnRhdGlvbjogc3RyaW5nKSB7XG4gICAgcmV0dXJuIGZ1bmN0aW9uIChcbiAgICAgICAgdmFsdWU6IGFueSxcbiAgICAgICAgY29udGV4dDogQ2xhc3NEZWNvcmF0b3JDb250ZXh0IHwgQ2xhc3NGaWVsZERlY29yYXRvckNvbnRleHQgfCBDbGFzc01ldGhvZERlY29yYXRvckNvbnRleHRcbiAgICApIHt9O1xufVxuXG5mdW5jdGlvbiBjcmVhdGVSZXF1ZXN0RGVjb3JhdG9yKGh0dHBNZXRob2Q6IHN0cmluZykge1xuICAgIHJldHVybiBmdW5jdGlvbiAocGF0aDogc3RyaW5nKSB7XG4gICAgICAgIHJldHVybiBmdW5jdGlvbiAodGFyZ2V0OiBhbnksIHByb3BlcnR5S2V5OiBzdHJpbmcpIHtcbiAgICAgICAgICAgIEdMT0JBTF9ST1VURVMucHVzaCh7XG4gICAgICAgICAgICAgICAgY29udHJvbGxlcjogdGFyZ2V0LmNvbnN0cnVjdG9yLFxuICAgICAgICAgICAgICAgIG1ldGhvZDogaHR0cE1ldGhvZCxcbiAgICAgICAgICAgICAgICBwYXRoOiBwYXRoIHx8IFwiL1wiLFxuICAgICAgICAgICAgICAgIHByb3BlcnR5S2V5XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfTtcbiAgICB9O1xufVxuXG5leHBvcnQgY29uc3QgR2V0ID0gY3JlYXRlUmVxdWVzdERlY29yYXRvcihcImdldFwiKTtcbmV4cG9ydCBjb25zdCBQb3N0ID0gY3JlYXRlUmVxdWVzdERlY29yYXRvcihcInBvc3RcIik7XG5leHBvcnQgY29uc3QgUHV0ID0gY3JlYXRlUmVxdWVzdERlY29yYXRvcihcInB1dFwiKTtcbmV4cG9ydCBjb25zdCBQYXRjaCA9IGNyZWF0ZVJlcXVlc3REZWNvcmF0b3IoXCJwYXRjaFwiKTtcbmV4cG9ydCBjb25zdCBEZWxldGUgPSBjcmVhdGVSZXF1ZXN0RGVjb3JhdG9yKFwiZGVsZXRlXCIpO1xuZXhwb3J0IGNvbnN0IEhlYWQgPSBjcmVhdGVSZXF1ZXN0RGVjb3JhdG9yKFwiaGVhZFwiKTtcbmV4cG9ydCBjb25zdCBPcHRpb25zID0gY3JlYXRlUmVxdWVzdERlY29yYXRvcihcIm9wdGlvbnNcIik7XG4iXSwKICAibWFwcGluZ3MiOiAiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBLFNBQW9CO0FBRXBCLE1BQU0sYUFBYSx1QkFBTyxJQUFJLDZCQUE2QjtBQUUzRCxNQUFNLGdCQUNELFdBQW1CLFVBQVUsTUFDNUIsV0FBbUIsVUFBVSxJQUFJLENBQUM7QUFFeEMsTUFBTSxTQUFTLEdBQUcsUUFBUTtBQUVuQixTQUFTLFdBQVcsS0FBcUI7QUFDNUMsUUFBTSxXQUFXLElBQUksSUFBSTtBQUN6QixRQUFNLFNBQVM7QUFDZixhQUFXLFNBQVMsUUFBUTtBQUN4QixVQUFNLEtBQUssU0FBUyxNQUFNLFlBQVksSUFBSTtBQUMxQyxRQUFJLE9BQU8sT0FBTyxZQUFZO0FBQ25DLE9BQUMsQ0FBQ0EsS0FBSUMsY0FBYTtBQUNaLGVBQU8sU0FBUyxNQUFNLElBQUksRUFBRSxNQUFNLE1BQU0sRUFBRSxDQUFDLEtBQUssS0FBSyxRQUFRO0FBQ3pELGdCQUFNLE9BQU8sSUFBSSxPQUFPLElBQUksS0FBSyxJQUFJO0FBQ3JDLGdCQUFNLFNBQVNELElBQUcsS0FBS0MsV0FBVSxNQUFNLEtBQUssS0FBSyxHQUFHO0FBQ3BELGNBQUksV0FBVyxRQUFXO0FBQ3RCLGdCQUFJLEtBQUssTUFBTTtBQUFBLFVBQ25CO0FBQUEsUUFDSixDQUFDO0FBQUEsTUFDTCxHQUFHLElBQUksUUFBUTtBQUFBLElBQ2I7QUFBQSxFQUNKO0FBRUEsU0FBTyxRQUFRO0FBQ2xCLGdCQUFjLFNBQVM7QUFDeEI7QUFHTyxTQUFTLGNBQWMsZUFBdUI7QUFDakQsU0FBTyxTQUNILE9BQ0EsU0FDRjtBQUFBLEVBQUM7QUFDUDtBQUVBLFNBQVMsdUJBQXVCLFlBQW9CO0FBQ2hELFNBQU8sU0FBVSxNQUFjO0FBQzNCLFdBQU8sU0FBVSxRQUFhLGFBQXFCO0FBQy9DLG9CQUFjLEtBQUs7QUFBQSxRQUNmLFlBQVksT0FBTztBQUFBLFFBQ25CLFFBQVE7QUFBQSxRQUNSLE1BQU0sUUFBUTtBQUFBLFFBQ2Q7QUFBQSxNQUNKLENBQUM7QUFBQSxJQUNMO0FBQUEsRUFDSjtBQUNKO0FBRU8sTUFBTSxNQUFNLHVCQUF1QixLQUFLO0FBQ3hDLE1BQU0sT0FBTyx1QkFBdUIsTUFBTTtBQUMxQyxNQUFNLE1BQU0sdUJBQXVCLEtBQUs7QUFDeEMsTUFBTSxRQUFRLHVCQUF1QixPQUFPO0FBQzVDLE1BQU0sU0FBUyx1QkFBdUIsUUFBUTtBQUM5QyxNQUFNLE9BQU8sdUJBQXVCLE1BQU07QUFDMUMsTUFBTSxVQUFVLHVCQUF1QixTQUFTOyIsCiAgIm5hbWVzIjogWyJmbiIsICJpbnN0YW5jZSJdCn0K
|
|
@@ -155,6 +155,7 @@ class HttpController {
|
|
|
155
155
|
}
|
|
156
156
|
} else {
|
|
157
157
|
logger.error("No suitable resource handler for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] found", resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);
|
|
158
|
+
logger.error("Registered resource handlers for Resource[{}] are [{}]", resourcePath, JSON.stringify(_oConfiguration));
|
|
158
159
|
this.sendError(response2.BAD_REQUEST, void 0, "Bad Request", "No suitable processor for this request.");
|
|
159
160
|
}
|
|
160
161
|
}
|
|
@@ -303,4 +304,4 @@ const matchMediaType = function(request2, producesMediaTypes, consumesMediaTypes
|
|
|
303
304
|
}
|
|
304
305
|
return isProduceMatched && isConsumeMatched;
|
|
305
306
|
};
|
|
306
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/http/rs/resource-http-controller.ts"],
  "sourcesContent": ["import { Response as response } from \"../response\";\nimport { Request as request } from \"../request\";\nimport { ResourceMappings } from \"./resource-mappings\";\nimport { Logging } from \"@aerokit/sdk/log\";\n\n// Declaration for the external dirigibleRequire function and its dependency\ndeclare function dirigibleRequire(module: string): any;\nconst { match } = dirigibleRequire(\"modules/src/http/path-to-regexp/6.2.1/index.js\");\n\nconst logger: any = Logging.getLogger('http.rs.controller');\n\n/**\n * Interface for the context object passed to handler functions (before, serve, catch).\n */\ninterface RequestContext {\n    pathParameters: { [key: string]: any };\n    queryParameters: { [key: string]: any };\n    response: any;\n    res: any;\n    request: any;\n    req: any;\n    // Context properties for error handling might also be attached here\n    suppressStack?: boolean;\n    httpErrorCode?: number;\n    errorMessage?: string;\n    errorName?: string;\n    errorCode?: any;\n}\n\nfunction getRequest(): any {\n    return request;\n}\nfunction getResponse(): any {\n    return response;\n}\n\n/**\n * Creates a service (HttpController) instance, optionally initialized with oMappings.\n *\n * @param oConfig Configuration object or configuration builder with configuration() getter function.\n * @returns A new HttpController instance.\n */\nexport function service(oConfig?: any): HttpController {\n    let config: ResourceMappings | any;\n    if (oConfig !== undefined) {\n        if (typeof oConfig === 'object' || oConfig instanceof ResourceMappings) {\n            config = oConfig;\n        }\n    }\n    return new HttpController(config);\n}\n\n/**\n * The main class for handling HTTP requests and routing them to the correct resource handlers.\n */\nexport class HttpController {\n\n    resource: Function;\n    resourcePath: Function;\n    resourceMappings: ResourceMappings;\n\n    // Index signature to allow dynamic method assignment in the constructor\n    [key: string]: any;\n\n    /**\n     * Constructor function for HttpController instances.\n     *\n     * @param oMappings The mappings configuration for this controller.\n     */\n    constructor(oMappings?: ResourceMappings | any) {\n        if (oMappings instanceof ResourceMappings) {\n            this.resourceMappings = oMappings;\n        } else if (typeof oMappings === 'object' || oMappings === undefined) {\n            this.resourceMappings = new ResourceMappings(oMappings, this);\n        } else {\n            // Default initialization if input is unexpected\n            this.resourceMappings = new ResourceMappings({}, this);\n        }\n\n        // Alias for resourceMappings.resource\n        this.resource = this.resourcePath = this.resourceMappings.resourcePath.bind(this.resourceMappings);\n\n        // weave-in HTTP method-based factory functions - shortcut for service().resource(sPath).method\n        ['get', 'post', 'put', 'delete', 'remove', 'method'].forEach((sMethodName: string) => {\n            this[sMethodName] = (...allArguments: any[]): HttpController => {\n                if (allArguments.length < 1)\n                    throw Error('Insufficient arguments provided to HttpController method ' + sMethodName + '.');\n\n                // sPath is always the first argument\n                let sPath = allArguments[0];\n                if (sPath === undefined)\n                    sPath = \"\";\n\n                // The next arguments (sVerb, arrConsumes, arrProduces) are used by ResourceMappings.find/resource\n                const sVerb = allArguments[1];\n                const arrConsumes = allArguments[2];\n                const arrProduces = allArguments[3];\n\n                // Find existing resource or create a new one based on the path (and other optional constraints)\n                const resource: any = this.resourceMappings.find(sPath, sVerb, arrConsumes, arrProduces) || this.resourceMappings.resource(sPath);\n\n                // Pass all arguments *except* the sPath to the corresponding method of the resource\n                const resourceMethodArgs = allArguments.slice(1);\n                \n                // Execute the method on the resource instance\n                resource[sMethodName].apply(resource, resourceMethodArgs);\n                \n                return this;\n            };\n        });\n    }\n\n    /**\n     * Alias for execute.\n     */\n    listen(request: any, response: any): void {\n        this.execute(request, response);\n    }\n\n    /**\n     * Executes the request handling logic, finding the best matching resource and handler.\n     */\n    execute(request?: any, response?: any): void {\n        request = request || getRequest();\n        const requestPath: string = request.getResourcePath();\n        const method: string = request.getMethod().toLowerCase();\n        const _oConfiguration: any = this.resourceMappings.configuration();\n\n        const matches: any[] = matchRequestUrl(requestPath, method, _oConfiguration);\n        let resourceHandler: any;\n\n        if (matches && matches[0]) {\n            const verbHandlers: any[] = _oConfiguration[matches[0].d][method];\n            if (verbHandlers) {\n                resourceHandler = verbHandlers.filter((handlerDef) => {\n                    return matchMediaType(request, handlerDef.produces, handlerDef.consumes);\n                })[0];\n            }\n        }\n\n        response = response || getResponse();\n        const queryParams: { [key: string]: any } = request.getQueryParametersMap() || {};\n        const acceptsHeader: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Accept')) || [];\n        const contentTypeHeader: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Content-Type')) || [];\n        const resourcePath: string = requestPath;\n\n        if (resourceHandler) {\n            const ctx: RequestContext = {\n                \"pathParameters\": {},\n                \"queryParameters\": {},\n                \"response\": response,\n                \"res\": response,\n                \"request\": request,\n                \"req\": request\n            };\n            if (matches[0].pathParams) {\n                ctx.pathParameters = matches[0].pathParams;\n            }\n            ctx.queryParameters = queryParams;\n\n            const noop = function () { };\n            let _before: Function, _serve: Function, _catch: Function, _finally: Function;\n            _before = resourceHandler.before || noop;\n            _serve = resourceHandler.handler || resourceHandler.serve || noop;\n            _catch = resourceHandler.catch || catchErrorHandler.bind(this, {\n                path: resourcePath,\n                method: method.toUpperCase(),\n                contentType: contentTypeHeader.join(','),\n                accepts: acceptsHeader.join(',')\n            });\n            _finally = resourceHandler.finally || noop;\n\n            const callbackArgs: any[] = [ctx, request, response, resourceHandler, this];\n\n            try {\n                logger.trace('Before serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}]', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                _before.apply(this, callbackArgs);\n\n                if (!response.isCommitted()) {\n                    logger.trace('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}]', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                    _serve.apply(this, callbackArgs);\n                    logger.trace('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] finished', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                }\n            } catch (err: any) {\n                try {\n                    callbackArgs.splice(1, 0, err);\n                    _catch.apply(this, callbackArgs);\n                } catch (_catchErr) {\n                    logger.error('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] error handler threw error', _catchErr);\n                    throw _catchErr;\n                }\n            } finally {\n                HttpController.prototype.closeResponse.call(this);\n                try {\n                    _finally.apply(this, []);\n                } catch (_finallyErr) {\n                    logger.error('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] post handler threw error', _finallyErr);\n                }\n            }\n        } else {\n            logger.error('No suitable resource handler for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] found', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n            this.sendError(response.BAD_REQUEST, undefined, 'Bad Request', 'No suitable processor for this request.');\n        }\n    }\n\n    /**\n     * Returns the ResourceMappings instance of this controller.\n     */\n    mappings(): ResourceMappings {\n        return this.resourceMappings;\n    };\n\n    /**\n     * Sends an error response to the client, formatted based on the accepted media type.\n     */\n    sendError(httpErrorCode: number, applicationErrorCode: any, errorName: string, errorDetails: string): void {\n        const clientAcceptMediaTypes: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Accept')) || ['application/json'];\n        const isHtml: boolean = clientAcceptMediaTypes.some((acceptMediaType) => isMimeTypeCompatible('*/html', acceptMediaType));\n\n        response.setStatus(httpErrorCode || response.INTERNAL_SERVER_ERROR);\n\n        if (isHtml) {\n            const message: string = errorName + (applicationErrorCode !== undefined ? '[' + applicationErrorCode + ']' : '') + (errorDetails ? ': ' + errorDetails : '');\n            response.sendError(httpErrorCode || response.INTERNAL_SERVER_ERROR, message);\n        } else {\n            const body = {\n                \"code\": applicationErrorCode,\n                \"error\": errorName,\n                \"details\": errorDetails\n            };\n            response.setHeader(\"Content-Type\", \"application/json\");\n            response.print(JSON.stringify(body, null, 2));\n        }\n        this.closeResponse();\n    };\n\n    /**\n     * Flushes and closes the HTTP response stream.\n     */\n    closeResponse(): void {\n        response.flush();\n        response.close();\n    };\n}\n\n/**\n * Custom sort function for matched route definitions, preferring exact matches over those with placeholders.\n */\nfunction matchedRouteDefinitionsSorter(p: any, n: any): number {\n    p.w = calculateMatchedRouteWeight(p);\n    n.w = calculateMatchedRouteWeight(n);\n\n    if (n.w === p.w) {\n        // The one with less placeholders wins\n        const m1 = p.d.match(/{(.*?)}/g);\n        const placeholdersCount1 = m1 !== null ? m1.length : 0;\n        const m2 = n.d.match(/{(.*?)}/g);\n        const placeholdersCount2 = m2 !== null ? m2.length : 0;\n        if (placeholdersCount1 > placeholdersCount2) {\n            n.w = n.w + 1;\n        } else if (placeholdersCount1 < placeholdersCount2) {\n            p.w = p.w + 1;\n        }\n    }\n    return n.w - p.w;\n}\n\n/**\n * Calculates the initial weight of a matched route definition.\n */\nfunction calculateMatchedRouteWeight(matchedRoute: any): number {\n    return (matchedRoute.params && Object.keys(matchedRoute.params).length > 0) ? 0 : 1; // always prefer exact route definitions - set weight to 1\n}\n\n/**\n * Transforms path parameters declared in braces (e.g., '/api/{pathParam}') to path-to-regexp format (e.g., '/api/:pathParam').\n */\nfunction transformPathParamsDeclaredInBraces(pathDefinition: string): string {\n    const pathParamsInBracesMatcher = /({(\\w*\\*?)})/g; // matches cases like '/api/{pathParam}' or '/api/{pathParam*}'\n    return pathDefinition.replace(pathParamsInBracesMatcher, \":$2\"); // transforms matched cases to '/api/:pathParam' or '/api/:pathParam*'\n}\n\n/**\n * Finds all routes in the configuration that match the request path and method.\n */\nfunction matchRequestUrl(requestPath: string, method: string, cfg: any): any[] {\n    return Object.entries(cfg)\n        .filter(([_, handlers]) => handlers && (handlers as any)[method])\n        .map(([path, _]) => path)\n        .reduce((matches: any[], path: string) => matchingRouteDefinitionsReducer(matches, path, requestPath), [])\n        .sort(matchedRouteDefinitionsSorter);\n}\n\n/**\n * Reducer function to attempt matching a defined path against the request path using path-to-regexp.\n */\nfunction matchingRouteDefinitionsReducer(matchedDefinitions: any[], definedPath: string, requestPath: string): any[] {\n    // 'match' from path-to-regexp is used here\n    const pathMatcher = match(transformPathParamsDeclaredInBraces(definedPath));\n    const matched = pathMatcher(requestPath);\n\n    if (matched) {\n        // Ensure pathParams is an object of key/value pairs\n        const pathParams = matched.params;\n\n        const matchedDefinition = {\n            p: requestPath,\n            d: definedPath,\n            pathParams: pathParams\n        };\n        matchedDefinitions.push(matchedDefinition);\n    }\n    return matchedDefinitions;\n}\n\n/**\n * Normalizes an HTTP media type header value (e.g., \"text/plain; q=0.9, application/json\").\n */\nfunction normalizeMediaTypeHeaderValue(sMediaType: string | undefined | null): string[] | undefined {\n    if (sMediaType === undefined || sMediaType === null)\n        return;\n    // convert to array of individual types\n    let arrMediaType = sMediaType.split(',');\n    arrMediaType = arrMediaType.map((mimeTypeEntry) => {\n        // remove escaping, remove quality or other attributes (e.g., '; q=0.9')\n        return mimeTypeEntry.replace('\\\\', '').split(';')[0].trim();\n    });\n    return arrMediaType.filter(type => type.length > 0);\n}\n\n/**\n * Checks if a source MIME type is compatible with a target MIME type, supporting wildcards (*).\n */\nfunction isMimeTypeCompatible(source: string, target: string): boolean {\n    if (source === target)\n        return true;\n\n    const targetM = target.split('/');\n    const sourceM = source.split('/');\n\n    // Target is wildcard type, Source has a specific subtype (e.g., target=*/json, source=application/json)\n    if (targetM[0] === '*' && targetM[1] === sourceM[1])\n        return true;\n    // Source is wildcard type, Target has a specific subtype (e.g., source=*/json, target=application/json)\n    if (sourceM[0] === '*' && targetM[1] === sourceM[1])\n        return true;\n\n    // Target is wildcard subtype, Source has a specific type (e.g., target=application/*, source=application/json)\n    if (targetM[1] === '*' && targetM[0] === sourceM[0])\n        return true;\n    // Source is wildcard subtype, Target has a specific type (e.g., source=application/*, target=application/json)\n    if (sourceM[1] === '*' && targetM[0] === sourceM[0])\n        return true;\n\n    return false;\n}\n\n/**\n * Default error handler function.\n */\nconst catchErrorHandler = function (this: HttpController, logctx: any, ctx: RequestContext, err: any, request: any, response: any): void {\n    if (ctx.suppressStack) {\n        const detailsMsg = (ctx.errorName || \"\") + (ctx.errorCode ? \" [\" + ctx.errorCode + \"]\" : \"\") + (ctx.errorMessage ? \": \" + ctx.errorMessage : \"\");\n        logger.info('Serving resource[{}], Verb[{}], Content-Type[{}], Accept[{}] finished in error. {}', logctx.path, logctx.method, logctx.contentType, logctx.accepts, detailsMsg);\n    } else {\n        logger.error('Serving resource[{}], Verb[{}], Content-Type[{}], Accept[{}] finished in error', logctx.path, logctx.method, logctx.contentType, logctx.accepts, err);\n    }\n\n    const httpErrorCode = ctx.httpErrorCode || response.INTERNAL_SERVER_ERROR;\n    const errorMessage = ctx.errorMessage || (err && err.message);\n    const errorName = ctx.errorName || (err && err.name);\n    const errorCode = ctx.errorCode;\n\n    this.sendError(httpErrorCode, errorCode, errorName, errorMessage);\n};\n\n/**\n * Checks if the request media types match the resource handler's consumes and produces constraints.\n */\nconst matchMediaType = function (request: any, producesMediaTypes: string[], consumesMediaTypes: string[]): boolean {\n    let isProduceMatched = false;\n    const acceptsMediaTypes = normalizeMediaTypeHeaderValue(request.getHeader('Accept'));\n\n    // 1. Check Produces (Accepts header)\n    if (!acceptsMediaTypes || acceptsMediaTypes.length === 0 || acceptsMediaTypes.includes('*/*')) {\n        // Output media type is not restricted by client or client accepts anything\n        isProduceMatched = true;\n    } else {\n        if (producesMediaTypes && producesMediaTypes.length) {\n            const matchedProducesMIME = acceptsMediaTypes.filter((acceptsMediaType) => {\n                return producesMediaTypes.some((producesMediaType) => {\n                    return isMimeTypeCompatible(acceptsMediaType, producesMediaType);\n                });\n            });\n            isProduceMatched = matchedProducesMIME && matchedProducesMIME.length > 0;\n        } else {\n            // Resource doesn't specify produces, so it matches if the client accepts anything or if produces is an empty array\n            isProduceMatched = true;\n        }\n    }\n\n    // 2. Check Consumes (Content-Type header)\n    let isConsumeMatched = false;\n    const contentTypeMediaTypes = normalizeMediaTypeHeaderValue(request.getContentType());\n\n    if (!consumesMediaTypes || consumesMediaTypes.length === 0 || consumesMediaTypes.includes('*/*')) {\n        // Input media type is not restricted by resource or resource accepts anything\n        isConsumeMatched = true;\n    } else {\n        if (contentTypeMediaTypes && consumesMediaTypes && consumesMediaTypes.length) {\n            const matchedConsumesMIME = contentTypeMediaTypes.filter((contentTypeMediaType) => {\n                return consumesMediaTypes.some((consumesMediaType) => {\n                    return isMimeTypeCompatible(contentTypeMediaType, consumesMediaType);\n                });\n            });\n            isConsumeMatched = matchedConsumesMIME && matchedConsumesMIME.length > 0;\n        }\n    }\n\n    return isProduceMatched && isConsumeMatched;\n};\n"],
  "mappings": "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",
  "names": ["request", "response"]
}

|
|
307
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/http/rs/resource-http-controller.ts"],
  "sourcesContent": ["import { Response as response } from \"../response\";\nimport { Request as request } from \"../request\";\nimport { ResourceMappings } from \"./resource-mappings\";\nimport { Logging } from \"@aerokit/sdk/log\";\n\n// Declaration for the external dirigibleRequire function and its dependency\ndeclare function dirigibleRequire(module: string): any;\nconst { match } = dirigibleRequire(\"modules/src/http/path-to-regexp/6.2.1/index.js\");\n\nconst logger: any = Logging.getLogger('http.rs.controller');\n\n/**\n * Interface for the context object passed to handler functions (before, serve, catch).\n */\ninterface RequestContext {\n    pathParameters: { [key: string]: any };\n    queryParameters: { [key: string]: any };\n    response: any;\n    res: any;\n    request: any;\n    req: any;\n    // Context properties for error handling might also be attached here\n    suppressStack?: boolean;\n    httpErrorCode?: number;\n    errorMessage?: string;\n    errorName?: string;\n    errorCode?: any;\n}\n\nfunction getRequest(): any {\n    return request;\n}\nfunction getResponse(): any {\n    return response;\n}\n\n/**\n * Creates a service (HttpController) instance, optionally initialized with oMappings.\n *\n * @param oConfig Configuration object or configuration builder with configuration() getter function.\n * @returns A new HttpController instance.\n */\nexport function service(oConfig?: any): HttpController {\n    let config: ResourceMappings | any;\n    if (oConfig !== undefined) {\n        if (typeof oConfig === 'object' || oConfig instanceof ResourceMappings) {\n            config = oConfig;\n        }\n    }\n    return new HttpController(config);\n}\n\n/**\n * The main class for handling HTTP requests and routing them to the correct resource handlers.\n */\nexport class HttpController {\n\n    resource: Function;\n    resourcePath: Function;\n    resourceMappings: ResourceMappings;\n\n    // Index signature to allow dynamic method assignment in the constructor\n    [key: string]: any;\n\n    /**\n     * Constructor function for HttpController instances.\n     *\n     * @param oMappings The mappings configuration for this controller.\n     */\n    constructor(oMappings?: ResourceMappings | any) {\n        if (oMappings instanceof ResourceMappings) {\n            this.resourceMappings = oMappings;\n        } else if (typeof oMappings === 'object' || oMappings === undefined) {\n            this.resourceMappings = new ResourceMappings(oMappings, this);\n        } else {\n            // Default initialization if input is unexpected\n            this.resourceMappings = new ResourceMappings({}, this);\n        }\n\n        // Alias for resourceMappings.resource\n        this.resource = this.resourcePath = this.resourceMappings.resourcePath.bind(this.resourceMappings);\n\n        // weave-in HTTP method-based factory functions - shortcut for service().resource(sPath).method\n        ['get', 'post', 'put', 'delete', 'remove', 'method'].forEach((sMethodName: string) => {\n            this[sMethodName] = (...allArguments: any[]): HttpController => {\n                if (allArguments.length < 1)\n                    throw Error('Insufficient arguments provided to HttpController method ' + sMethodName + '.');\n\n                // sPath is always the first argument\n                let sPath = allArguments[0];\n                if (sPath === undefined)\n                    sPath = \"\";\n\n                // The next arguments (sVerb, arrConsumes, arrProduces) are used by ResourceMappings.find/resource\n                const sVerb = allArguments[1];\n                const arrConsumes = allArguments[2];\n                const arrProduces = allArguments[3];\n\n                // Find existing resource or create a new one based on the path (and other optional constraints)\n                const resource: any = this.resourceMappings.find(sPath, sVerb, arrConsumes, arrProduces) || this.resourceMappings.resource(sPath);\n\n                // Pass all arguments *except* the sPath to the corresponding method of the resource\n                const resourceMethodArgs = allArguments.slice(1);\n                \n                // Execute the method on the resource instance\n                resource[sMethodName].apply(resource, resourceMethodArgs);\n                \n                return this;\n            };\n        });\n    }\n\n    /**\n     * Alias for execute.\n     */\n    listen(request: any, response: any): void {\n        this.execute(request, response);\n    }\n\n    /**\n     * Executes the request handling logic, finding the best matching resource and handler.\n     */\n    execute(request?: any, response?: any): void {\n        request = request || getRequest();\n        const requestPath: string = request.getResourcePath();\n        const method: string = request.getMethod().toLowerCase();\n        const _oConfiguration: any = this.resourceMappings.configuration();\n\n        const matches: any[] = matchRequestUrl(requestPath, method, _oConfiguration);\n        let resourceHandler: any;\n\n        if (matches && matches[0]) {\n            const verbHandlers: any[] = _oConfiguration[matches[0].d][method];\n            if (verbHandlers) {\n                resourceHandler = verbHandlers.filter((handlerDef) => {\n                    return matchMediaType(request, handlerDef.produces, handlerDef.consumes);\n                })[0];\n            }\n        }\n\n        response = response || getResponse();\n        const queryParams: { [key: string]: any } = request.getQueryParametersMap() || {};\n        const acceptsHeader: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Accept')) || [];\n        const contentTypeHeader: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Content-Type')) || [];\n        const resourcePath: string = requestPath;\n\n        if (resourceHandler) {\n            const ctx: RequestContext = {\n                \"pathParameters\": {},\n                \"queryParameters\": {},\n                \"response\": response,\n                \"res\": response,\n                \"request\": request,\n                \"req\": request\n            };\n            if (matches[0].pathParams) {\n                ctx.pathParameters = matches[0].pathParams;\n            }\n            ctx.queryParameters = queryParams;\n\n            const noop = function () { };\n            let _before: Function, _serve: Function, _catch: Function, _finally: Function;\n            _before = resourceHandler.before || noop;\n            _serve = resourceHandler.handler || resourceHandler.serve || noop;\n            _catch = resourceHandler.catch || catchErrorHandler.bind(this, {\n                path: resourcePath,\n                method: method.toUpperCase(),\n                contentType: contentTypeHeader.join(','),\n                accepts: acceptsHeader.join(',')\n            });\n            _finally = resourceHandler.finally || noop;\n\n            const callbackArgs: any[] = [ctx, request, response, resourceHandler, this];\n\n            try {\n                logger.trace('Before serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}]', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                _before.apply(this, callbackArgs);\n\n                if (!response.isCommitted()) {\n                    logger.trace('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}]', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                    _serve.apply(this, callbackArgs);\n                    logger.trace('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] finished', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                }\n            } catch (err: any) {\n                try {\n                    callbackArgs.splice(1, 0, err);\n                    _catch.apply(this, callbackArgs);\n                } catch (_catchErr) {\n                    logger.error('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] error handler threw error', _catchErr);\n                    throw _catchErr;\n                }\n            } finally {\n                HttpController.prototype.closeResponse.call(this);\n                try {\n                    _finally.apply(this, []);\n                } catch (_finallyErr) {\n                    logger.error('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] post handler threw error', _finallyErr);\n                }\n            }\n        } else {\n            logger.error('No suitable resource handler for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] found', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n\t\t\tlogger.error('Registered resource handlers for Resource[{}] are [{}]', resourcePath, JSON.stringify(_oConfiguration));\n            this.sendError(response.BAD_REQUEST, undefined, 'Bad Request', 'No suitable processor for this request.');\n        }\n    }\n\n    /**\n     * Returns the ResourceMappings instance of this controller.\n     */\n    mappings(): ResourceMappings {\n        return this.resourceMappings;\n    };\n\n    /**\n     * Sends an error response to the client, formatted based on the accepted media type.\n     */\n    sendError(httpErrorCode: number, applicationErrorCode: any, errorName: string, errorDetails: string): void {\n        const clientAcceptMediaTypes: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Accept')) || ['application/json'];\n        const isHtml: boolean = clientAcceptMediaTypes.some((acceptMediaType) => isMimeTypeCompatible('*/html', acceptMediaType));\n\n        response.setStatus(httpErrorCode || response.INTERNAL_SERVER_ERROR);\n\n        if (isHtml) {\n            const message: string = errorName + (applicationErrorCode !== undefined ? '[' + applicationErrorCode + ']' : '') + (errorDetails ? ': ' + errorDetails : '');\n            response.sendError(httpErrorCode || response.INTERNAL_SERVER_ERROR, message);\n        } else {\n            const body = {\n                \"code\": applicationErrorCode,\n                \"error\": errorName,\n                \"details\": errorDetails\n            };\n            response.setHeader(\"Content-Type\", \"application/json\");\n            response.print(JSON.stringify(body, null, 2));\n        }\n        this.closeResponse();\n    };\n\n    /**\n     * Flushes and closes the HTTP response stream.\n     */\n    closeResponse(): void {\n        response.flush();\n        response.close();\n    };\n}\n\n/**\n * Custom sort function for matched route definitions, preferring exact matches over those with placeholders.\n */\nfunction matchedRouteDefinitionsSorter(p: any, n: any): number {\n    p.w = calculateMatchedRouteWeight(p);\n    n.w = calculateMatchedRouteWeight(n);\n\n    if (n.w === p.w) {\n        // The one with less placeholders wins\n        const m1 = p.d.match(/{(.*?)}/g);\n        const placeholdersCount1 = m1 !== null ? m1.length : 0;\n        const m2 = n.d.match(/{(.*?)}/g);\n        const placeholdersCount2 = m2 !== null ? m2.length : 0;\n        if (placeholdersCount1 > placeholdersCount2) {\n            n.w = n.w + 1;\n        } else if (placeholdersCount1 < placeholdersCount2) {\n            p.w = p.w + 1;\n        }\n    }\n    return n.w - p.w;\n}\n\n/**\n * Calculates the initial weight of a matched route definition.\n */\nfunction calculateMatchedRouteWeight(matchedRoute: any): number {\n    return (matchedRoute.params && Object.keys(matchedRoute.params).length > 0) ? 0 : 1; // always prefer exact route definitions - set weight to 1\n}\n\n/**\n * Transforms path parameters declared in braces (e.g., '/api/{pathParam}') to path-to-regexp format (e.g., '/api/:pathParam').\n */\nfunction transformPathParamsDeclaredInBraces(pathDefinition: string): string {\n    const pathParamsInBracesMatcher = /({(\\w*\\*?)})/g; // matches cases like '/api/{pathParam}' or '/api/{pathParam*}'\n    return pathDefinition.replace(pathParamsInBracesMatcher, \":$2\"); // transforms matched cases to '/api/:pathParam' or '/api/:pathParam*'\n}\n\n/**\n * Finds all routes in the configuration that match the request path and method.\n */\nfunction matchRequestUrl(requestPath: string, method: string, cfg: any): any[] {\n    return Object.entries(cfg)\n        .filter(([_, handlers]) => handlers && (handlers as any)[method])\n        .map(([path, _]) => path)\n        .reduce((matches: any[], path: string) => matchingRouteDefinitionsReducer(matches, path, requestPath), [])\n        .sort(matchedRouteDefinitionsSorter);\n}\n\n/**\n * Reducer function to attempt matching a defined path against the request path using path-to-regexp.\n */\nfunction matchingRouteDefinitionsReducer(matchedDefinitions: any[], definedPath: string, requestPath: string): any[] {\n    // 'match' from path-to-regexp is used here\n    const pathMatcher = match(transformPathParamsDeclaredInBraces(definedPath));\n    const matched = pathMatcher(requestPath);\n\n    if (matched) {\n        // Ensure pathParams is an object of key/value pairs\n        const pathParams = matched.params;\n\n        const matchedDefinition = {\n            p: requestPath,\n            d: definedPath,\n            pathParams: pathParams\n        };\n        matchedDefinitions.push(matchedDefinition);\n    }\n    return matchedDefinitions;\n}\n\n/**\n * Normalizes an HTTP media type header value (e.g., \"text/plain; q=0.9, application/json\").\n */\nfunction normalizeMediaTypeHeaderValue(sMediaType: string | undefined | null): string[] | undefined {\n    if (sMediaType === undefined || sMediaType === null)\n        return;\n    // convert to array of individual types\n    let arrMediaType = sMediaType.split(',');\n    arrMediaType = arrMediaType.map((mimeTypeEntry) => {\n        // remove escaping, remove quality or other attributes (e.g., '; q=0.9')\n        return mimeTypeEntry.replace('\\\\', '').split(';')[0].trim();\n    });\n    return arrMediaType.filter(type => type.length > 0);\n}\n\n/**\n * Checks if a source MIME type is compatible with a target MIME type, supporting wildcards (*).\n */\nfunction isMimeTypeCompatible(source: string, target: string): boolean {\n    if (source === target)\n        return true;\n\n    const targetM = target.split('/');\n    const sourceM = source.split('/');\n\n    // Target is wildcard type, Source has a specific subtype (e.g., target=*/json, source=application/json)\n    if (targetM[0] === '*' && targetM[1] === sourceM[1])\n        return true;\n    // Source is wildcard type, Target has a specific subtype (e.g., source=*/json, target=application/json)\n    if (sourceM[0] === '*' && targetM[1] === sourceM[1])\n        return true;\n\n    // Target is wildcard subtype, Source has a specific type (e.g., target=application/*, source=application/json)\n    if (targetM[1] === '*' && targetM[0] === sourceM[0])\n        return true;\n    // Source is wildcard subtype, Target has a specific type (e.g., source=application/*, target=application/json)\n    if (sourceM[1] === '*' && targetM[0] === sourceM[0])\n        return true;\n\n    return false;\n}\n\n/**\n * Default error handler function.\n */\nconst catchErrorHandler = function (this: HttpController, logctx: any, ctx: RequestContext, err: any, request: any, response: any): void {\n    if (ctx.suppressStack) {\n        const detailsMsg = (ctx.errorName || \"\") + (ctx.errorCode ? \" [\" + ctx.errorCode + \"]\" : \"\") + (ctx.errorMessage ? \": \" + ctx.errorMessage : \"\");\n        logger.info('Serving resource[{}], Verb[{}], Content-Type[{}], Accept[{}] finished in error. {}', logctx.path, logctx.method, logctx.contentType, logctx.accepts, detailsMsg);\n    } else {\n        logger.error('Serving resource[{}], Verb[{}], Content-Type[{}], Accept[{}] finished in error', logctx.path, logctx.method, logctx.contentType, logctx.accepts, err);\n    }\n\n    const httpErrorCode = ctx.httpErrorCode || response.INTERNAL_SERVER_ERROR;\n    const errorMessage = ctx.errorMessage || (err && err.message);\n    const errorName = ctx.errorName || (err && err.name);\n    const errorCode = ctx.errorCode;\n\n    this.sendError(httpErrorCode, errorCode, errorName, errorMessage);\n};\n\n/**\n * Checks if the request media types match the resource handler's consumes and produces constraints.\n */\nconst matchMediaType = function (request: any, producesMediaTypes: string[], consumesMediaTypes: string[]): boolean {\n    let isProduceMatched = false;\n    const acceptsMediaTypes = normalizeMediaTypeHeaderValue(request.getHeader('Accept'));\n\n    // 1. Check Produces (Accepts header)\n    if (!acceptsMediaTypes || acceptsMediaTypes.length === 0 || acceptsMediaTypes.includes('*/*')) {\n        // Output media type is not restricted by client or client accepts anything\n        isProduceMatched = true;\n    } else {\n        if (producesMediaTypes && producesMediaTypes.length) {\n            const matchedProducesMIME = acceptsMediaTypes.filter((acceptsMediaType) => {\n                return producesMediaTypes.some((producesMediaType) => {\n                    return isMimeTypeCompatible(acceptsMediaType, producesMediaType);\n                });\n            });\n            isProduceMatched = matchedProducesMIME && matchedProducesMIME.length > 0;\n        } else {\n            // Resource doesn't specify produces, so it matches if the client accepts anything or if produces is an empty array\n            isProduceMatched = true;\n        }\n    }\n\n    // 2. Check Consumes (Content-Type header)\n    let isConsumeMatched = false;\n    const contentTypeMediaTypes = normalizeMediaTypeHeaderValue(request.getContentType());\n\n    if (!consumesMediaTypes || consumesMediaTypes.length === 0 || consumesMediaTypes.includes('*/*')) {\n        // Input media type is not restricted by resource or resource accepts anything\n        isConsumeMatched = true;\n    } else {\n        if (contentTypeMediaTypes && consumesMediaTypes && consumesMediaTypes.length) {\n            const matchedConsumesMIME = contentTypeMediaTypes.filter((contentTypeMediaType) => {\n                return consumesMediaTypes.some((consumesMediaType) => {\n                    return isMimeTypeCompatible(contentTypeMediaType, consumesMediaType);\n                });\n            });\n            isConsumeMatched = matchedConsumesMIME && matchedConsumesMIME.length > 0;\n        }\n    }\n\n    return isProduceMatched && isConsumeMatched;\n};\n"],
  "mappings": "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",
  "names": ["request", "response"]
}

|
|
@@ -1,11 +1,11 @@
|
|
|
1
1
|
export declare function Controller(ctr: {
|
|
2
2
|
new (): any;
|
|
3
|
-
}
|
|
3
|
+
}): void;
|
|
4
4
|
export declare function Documentation(documentation: string): (value: any, context: ClassDecoratorContext | ClassFieldDecoratorContext | ClassMethodDecoratorContext) => void;
|
|
5
|
-
export declare const Get: (path: string
|
|
6
|
-
export declare const Post: (path: string
|
|
7
|
-
export declare const Put: (path: string
|
|
8
|
-
export declare const Patch: (path: string
|
|
9
|
-
export declare const Delete: (path: string
|
|
10
|
-
export declare const Head: (path: string
|
|
11
|
-
export declare const Options: (path: string
|
|
5
|
+
export declare const Get: (path: string) => (target: any, propertyKey: string) => void;
|
|
6
|
+
export declare const Post: (path: string) => (target: any, propertyKey: string) => void;
|
|
7
|
+
export declare const Put: (path: string) => (target: any, propertyKey: string) => void;
|
|
8
|
+
export declare const Patch: (path: string) => (target: any, propertyKey: string) => void;
|
|
9
|
+
export declare const Delete: (path: string) => (target: any, propertyKey: string) => void;
|
|
10
|
+
export declare const Head: (path: string) => (target: any, propertyKey: string) => void;
|
|
11
|
+
export declare const Options: (path: string) => (target: any, propertyKey: string) => void;
|
|
@@ -1,14 +1,43 @@
|
|
|
1
1
|
import * as rs from "@aerokit/sdk/http/rs";
|
|
2
|
+
const ROUTES_KEY = /* @__PURE__ */ Symbol.for("dirigible.controller.routes");
|
|
3
|
+
const GLOBAL_ROUTES = globalThis[ROUTES_KEY] ?? (globalThis[ROUTES_KEY] = []);
|
|
2
4
|
const router = rs.service();
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
5
|
+
function Controller(ctr) {
|
|
6
|
+
const instance = new ctr();
|
|
7
|
+
const routes = GLOBAL_ROUTES;
|
|
8
|
+
for (const route of routes) {
|
|
9
|
+
const fn = instance[route.propertyKey.name];
|
|
10
|
+
if (typeof fn === "function") {
|
|
11
|
+
((fn2, instance2) => {
|
|
12
|
+
router.resource(route.path)[route.method]((ctx, req, res) => {
|
|
13
|
+
const body = req.json ? req.json() : null;
|
|
14
|
+
const result = fn2.call(instance2, body, ctx, req, res);
|
|
15
|
+
if (result !== void 0) {
|
|
16
|
+
res.json(result);
|
|
17
|
+
}
|
|
18
|
+
});
|
|
19
|
+
})(fn, instance);
|
|
20
|
+
}
|
|
21
|
+
}
|
|
6
22
|
router.execute();
|
|
23
|
+
GLOBAL_ROUTES.length = 0;
|
|
7
24
|
}
|
|
8
25
|
function Documentation(documentation) {
|
|
9
26
|
return function(value, context) {
|
|
10
27
|
};
|
|
11
28
|
}
|
|
29
|
+
function createRequestDecorator(httpMethod) {
|
|
30
|
+
return function(path) {
|
|
31
|
+
return function(target, propertyKey) {
|
|
32
|
+
GLOBAL_ROUTES.push({
|
|
33
|
+
controller: target.constructor,
|
|
34
|
+
method: httpMethod,
|
|
35
|
+
path: path || "/",
|
|
36
|
+
propertyKey
|
|
37
|
+
});
|
|
38
|
+
};
|
|
39
|
+
};
|
|
40
|
+
}
|
|
12
41
|
const Get = createRequestDecorator("get");
|
|
13
42
|
const Post = createRequestDecorator("post");
|
|
14
43
|
const Put = createRequestDecorator("put");
|
|
@@ -16,23 +45,6 @@ const Patch = createRequestDecorator("patch");
|
|
|
16
45
|
const Delete = createRequestDecorator("delete");
|
|
17
46
|
const Head = createRequestDecorator("head");
|
|
18
47
|
const Options = createRequestDecorator("options");
|
|
19
|
-
function createRequestDecorator(httpMethod) {
|
|
20
|
-
return function(path, consumesMimeTypes = ["*/*"], producesMimeTypes = ["application/json"]) {
|
|
21
|
-
return function(target, propertyKey, descriptor) {
|
|
22
|
-
const handler = descriptor ? descriptor.value : target;
|
|
23
|
-
router.resource(path)[httpMethod]((ctx, req, res) => {
|
|
24
|
-
handleRequest(req, res, ctx, handler);
|
|
25
|
-
}).consumes(consumesMimeTypes).produces(producesMimeTypes);
|
|
26
|
-
};
|
|
27
|
-
};
|
|
28
|
-
}
|
|
29
|
-
function handleRequest(req, res, ctx, handler) {
|
|
30
|
-
const body = req.json();
|
|
31
|
-
const maybeResponseBody = handler.apply(instance || {}, [body, ctx, req, res]);
|
|
32
|
-
if (maybeResponseBody) {
|
|
33
|
-
res.json(maybeResponseBody);
|
|
34
|
-
}
|
|
35
|
-
}
|
|
36
48
|
export {
|
|
37
49
|
Controller,
|
|
38
50
|
Delete,
|
|
@@ -44,4 +56,4 @@ export {
|
|
|
44
56
|
Post,
|
|
45
57
|
Put
|
|
46
58
|
};
|
|
47
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
59
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vc3JjL2h0dHAvZGVjb3JhdG9ycy50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiaW1wb3J0ICogYXMgcnMgZnJvbSBcIkBhZXJva2l0L3Nkay9odHRwL3JzXCI7XG5cbmNvbnN0IFJPVVRFU19LRVkgPSBTeW1ib2wuZm9yKFwiZGlyaWdpYmxlLmNvbnRyb2xsZXIucm91dGVzXCIpO1xuXG5jb25zdCBHTE9CQUxfUk9VVEVTOiBhbnlbXSA9XG4gICAgKGdsb2JhbFRoaXMgYXMgYW55KVtST1VURVNfS0VZXSA/P1xuICAgICgoZ2xvYmFsVGhpcyBhcyBhbnkpW1JPVVRFU19LRVldID0gW10pO1xuXG5jb25zdCByb3V0ZXIgPSBycy5zZXJ2aWNlKCk7XG5cbmV4cG9ydCBmdW5jdGlvbiBDb250cm9sbGVyKGN0cjogeyBuZXcoKTogYW55IH0pIHtcbiAgICBjb25zdCBpbnN0YW5jZSA9IG5ldyBjdHIoKTtcbiAgICBjb25zdCByb3V0ZXMgPSBHTE9CQUxfUk9VVEVTO1xuICAgIGZvciAoY29uc3Qgcm91dGUgb2Ygcm91dGVzKSB7XG4gICAgICAgIGNvbnN0IGZuID0gaW5zdGFuY2Vbcm91dGUucHJvcGVydHlLZXkubmFtZV07XG4gICAgICAgIGlmICh0eXBlb2YgZm4gPT09IFwiZnVuY3Rpb25cIikge1xuXHRcdFx0KChmbiwgaW5zdGFuY2UpID0+IHtcblx0XHQgICAgICAgIHJvdXRlci5yZXNvdXJjZShyb3V0ZS5wYXRoKVtyb3V0ZS5tZXRob2RdKChjdHgsIHJlcSwgcmVzKSA9PiB7XG5cdFx0ICAgICAgICAgICAgY29uc3QgYm9keSA9IHJlcS5qc29uID8gcmVxLmpzb24oKSA6IG51bGw7XG5cdFx0ICAgICAgICAgICAgY29uc3QgcmVzdWx0ID0gZm4uY2FsbChpbnN0YW5jZSwgYm9keSwgY3R4LCByZXEsIHJlcyk7XG5cdFx0ICAgICAgICAgICAgaWYgKHJlc3VsdCAhPT0gdW5kZWZpbmVkKSB7XG5cdFx0ICAgICAgICAgICAgICAgIHJlcy5qc29uKHJlc3VsdCk7XG5cdFx0ICAgICAgICAgICAgfVxuXHRcdCAgICAgICAgfSk7XG5cdFx0ICAgIH0pKGZuLCBpbnN0YW5jZSk7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICByb3V0ZXIuZXhlY3V0ZSgpO1xuXHRHTE9CQUxfUk9VVEVTLmxlbmd0aCA9IDA7XG59XG5cblxuZXhwb3J0IGZ1bmN0aW9uIERvY3VtZW50YXRpb24oZG9jdW1lbnRhdGlvbjogc3RyaW5nKSB7XG4gICAgcmV0dXJuIGZ1bmN0aW9uIChcbiAgICAgICAgdmFsdWU6IGFueSxcbiAgICAgICAgY29udGV4dDogQ2xhc3NEZWNvcmF0b3JDb250ZXh0IHwgQ2xhc3NGaWVsZERlY29yYXRvckNvbnRleHQgfCBDbGFzc01ldGhvZERlY29yYXRvckNvbnRleHRcbiAgICApIHt9O1xufVxuXG5mdW5jdGlvbiBjcmVhdGVSZXF1ZXN0RGVjb3JhdG9yKGh0dHBNZXRob2Q6IHN0cmluZykge1xuICAgIHJldHVybiBmdW5jdGlvbiAocGF0aDogc3RyaW5nKSB7XG4gICAgICAgIHJldHVybiBmdW5jdGlvbiAodGFyZ2V0OiBhbnksIHByb3BlcnR5S2V5OiBzdHJpbmcpIHtcbiAgICAgICAgICAgIEdMT0JBTF9ST1VURVMucHVzaCh7XG4gICAgICAgICAgICAgICAgY29udHJvbGxlcjogdGFyZ2V0LmNvbnN0cnVjdG9yLFxuICAgICAgICAgICAgICAgIG1ldGhvZDogaHR0cE1ldGhvZCxcbiAgICAgICAgICAgICAgICBwYXRoOiBwYXRoIHx8IFwiL1wiLFxuICAgICAgICAgICAgICAgIHByb3BlcnR5S2V5XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfTtcbiAgICB9O1xufVxuXG5leHBvcnQgY29uc3QgR2V0ID0gY3JlYXRlUmVxdWVzdERlY29yYXRvcihcImdldFwiKTtcbmV4cG9ydCBjb25zdCBQb3N0ID0gY3JlYXRlUmVxdWVzdERlY29yYXRvcihcInBvc3RcIik7XG5leHBvcnQgY29uc3QgUHV0ID0gY3JlYXRlUmVxdWVzdERlY29yYXRvcihcInB1dFwiKTtcbmV4cG9ydCBjb25zdCBQYXRjaCA9IGNyZWF0ZVJlcXVlc3REZWNvcmF0b3IoXCJwYXRjaFwiKTtcbmV4cG9ydCBjb25zdCBEZWxldGUgPSBjcmVhdGVSZXF1ZXN0RGVjb3JhdG9yKFwiZGVsZXRlXCIpO1xuZXhwb3J0IGNvbnN0IEhlYWQgPSBjcmVhdGVSZXF1ZXN0RGVjb3JhdG9yKFwiaGVhZFwiKTtcbmV4cG9ydCBjb25zdCBPcHRpb25zID0gY3JlYXRlUmVxdWVzdERlY29yYXRvcihcIm9wdGlvbnNcIik7XG4iXSwKICAibWFwcGluZ3MiOiAiQUFBQSxZQUFZLFFBQVE7QUFFcEIsTUFBTSxhQUFhLHVCQUFPLElBQUksNkJBQTZCO0FBRTNELE1BQU0sZ0JBQ0QsV0FBbUIsVUFBVSxNQUM1QixXQUFtQixVQUFVLElBQUksQ0FBQztBQUV4QyxNQUFNLFNBQVMsR0FBRyxRQUFRO0FBRW5CLFNBQVMsV0FBVyxLQUFxQjtBQUM1QyxRQUFNLFdBQVcsSUFBSSxJQUFJO0FBQ3pCLFFBQU0sU0FBUztBQUNmLGFBQVcsU0FBUyxRQUFRO0FBQ3hCLFVBQU0sS0FBSyxTQUFTLE1BQU0sWUFBWSxJQUFJO0FBQzFDLFFBQUksT0FBTyxPQUFPLFlBQVk7QUFDbkMsT0FBQyxDQUFDQSxLQUFJQyxjQUFhO0FBQ1osZUFBTyxTQUFTLE1BQU0sSUFBSSxFQUFFLE1BQU0sTUFBTSxFQUFFLENBQUMsS0FBSyxLQUFLLFFBQVE7QUFDekQsZ0JBQU0sT0FBTyxJQUFJLE9BQU8sSUFBSSxLQUFLLElBQUk7QUFDckMsZ0JBQU0sU0FBU0QsSUFBRyxLQUFLQyxXQUFVLE1BQU0sS0FBSyxLQUFLLEdBQUc7QUFDcEQsY0FBSSxXQUFXLFFBQVc7QUFDdEIsZ0JBQUksS0FBSyxNQUFNO0FBQUEsVUFDbkI7QUFBQSxRQUNKLENBQUM7QUFBQSxNQUNMLEdBQUcsSUFBSSxRQUFRO0FBQUEsSUFDYjtBQUFBLEVBQ0o7QUFFQSxTQUFPLFFBQVE7QUFDbEIsZ0JBQWMsU0FBUztBQUN4QjtBQUdPLFNBQVMsY0FBYyxlQUF1QjtBQUNqRCxTQUFPLFNBQ0gsT0FDQSxTQUNGO0FBQUEsRUFBQztBQUNQO0FBRUEsU0FBUyx1QkFBdUIsWUFBb0I7QUFDaEQsU0FBTyxTQUFVLE1BQWM7QUFDM0IsV0FBTyxTQUFVLFFBQWEsYUFBcUI7QUFDL0Msb0JBQWMsS0FBSztBQUFBLFFBQ2YsWUFBWSxPQUFPO0FBQUEsUUFDbkIsUUFBUTtBQUFBLFFBQ1IsTUFBTSxRQUFRO0FBQUEsUUFDZDtBQUFBLE1BQ0osQ0FBQztBQUFBLElBQ0w7QUFBQSxFQUNKO0FBQ0o7QUFFTyxNQUFNLE1BQU0sdUJBQXVCLEtBQUs7QUFDeEMsTUFBTSxPQUFPLHVCQUF1QixNQUFNO0FBQzFDLE1BQU0sTUFBTSx1QkFBdUIsS0FBSztBQUN4QyxNQUFNLFFBQVEsdUJBQXVCLE9BQU87QUFDNUMsTUFBTSxTQUFTLHVCQUF1QixRQUFRO0FBQzlDLE1BQU0sT0FBTyx1QkFBdUIsTUFBTTtBQUMxQyxNQUFNLFVBQVUsdUJBQXVCLFNBQVM7IiwKICAibmFtZXMiOiBbImZuIiwgImluc3RhbmNlIl0KfQo=
|
|
@@ -132,6 +132,7 @@ class HttpController {
|
|
|
132
132
|
}
|
|
133
133
|
} else {
|
|
134
134
|
logger.error("No suitable resource handler for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] found", resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);
|
|
135
|
+
logger.error("Registered resource handlers for Resource[{}] are [{}]", resourcePath, JSON.stringify(_oConfiguration));
|
|
135
136
|
this.sendError(response2.BAD_REQUEST, void 0, "Bad Request", "No suitable processor for this request.");
|
|
136
137
|
}
|
|
137
138
|
}
|
|
@@ -284,4 +285,4 @@ export {
|
|
|
284
285
|
HttpController,
|
|
285
286
|
service
|
|
286
287
|
};
|
|
287
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/http/rs/resource-http-controller.ts"],
  "sourcesContent": ["import { Response as response } from \"../response\";\nimport { Request as request } from \"../request\";\nimport { ResourceMappings } from \"./resource-mappings\";\nimport { Logging } from \"@aerokit/sdk/log\";\n\n// Declaration for the external dirigibleRequire function and its dependency\ndeclare function dirigibleRequire(module: string): any;\nconst { match } = dirigibleRequire(\"modules/src/http/path-to-regexp/6.2.1/index.js\");\n\nconst logger: any = Logging.getLogger('http.rs.controller');\n\n/**\n * Interface for the context object passed to handler functions (before, serve, catch).\n */\ninterface RequestContext {\n    pathParameters: { [key: string]: any };\n    queryParameters: { [key: string]: any };\n    response: any;\n    res: any;\n    request: any;\n    req: any;\n    // Context properties for error handling might also be attached here\n    suppressStack?: boolean;\n    httpErrorCode?: number;\n    errorMessage?: string;\n    errorName?: string;\n    errorCode?: any;\n}\n\nfunction getRequest(): any {\n    return request;\n}\nfunction getResponse(): any {\n    return response;\n}\n\n/**\n * Creates a service (HttpController) instance, optionally initialized with oMappings.\n *\n * @param oConfig Configuration object or configuration builder with configuration() getter function.\n * @returns A new HttpController instance.\n */\nexport function service(oConfig?: any): HttpController {\n    let config: ResourceMappings | any;\n    if (oConfig !== undefined) {\n        if (typeof oConfig === 'object' || oConfig instanceof ResourceMappings) {\n            config = oConfig;\n        }\n    }\n    return new HttpController(config);\n}\n\n/**\n * The main class for handling HTTP requests and routing them to the correct resource handlers.\n */\nexport class HttpController {\n\n    resource: Function;\n    resourcePath: Function;\n    resourceMappings: ResourceMappings;\n\n    // Index signature to allow dynamic method assignment in the constructor\n    [key: string]: any;\n\n    /**\n     * Constructor function for HttpController instances.\n     *\n     * @param oMappings The mappings configuration for this controller.\n     */\n    constructor(oMappings?: ResourceMappings | any) {\n        if (oMappings instanceof ResourceMappings) {\n            this.resourceMappings = oMappings;\n        } else if (typeof oMappings === 'object' || oMappings === undefined) {\n            this.resourceMappings = new ResourceMappings(oMappings, this);\n        } else {\n            // Default initialization if input is unexpected\n            this.resourceMappings = new ResourceMappings({}, this);\n        }\n\n        // Alias for resourceMappings.resource\n        this.resource = this.resourcePath = this.resourceMappings.resourcePath.bind(this.resourceMappings);\n\n        // weave-in HTTP method-based factory functions - shortcut for service().resource(sPath).method\n        ['get', 'post', 'put', 'delete', 'remove', 'method'].forEach((sMethodName: string) => {\n            this[sMethodName] = (...allArguments: any[]): HttpController => {\n                if (allArguments.length < 1)\n                    throw Error('Insufficient arguments provided to HttpController method ' + sMethodName + '.');\n\n                // sPath is always the first argument\n                let sPath = allArguments[0];\n                if (sPath === undefined)\n                    sPath = \"\";\n\n                // The next arguments (sVerb, arrConsumes, arrProduces) are used by ResourceMappings.find/resource\n                const sVerb = allArguments[1];\n                const arrConsumes = allArguments[2];\n                const arrProduces = allArguments[3];\n\n                // Find existing resource or create a new one based on the path (and other optional constraints)\n                const resource: any = this.resourceMappings.find(sPath, sVerb, arrConsumes, arrProduces) || this.resourceMappings.resource(sPath);\n\n                // Pass all arguments *except* the sPath to the corresponding method of the resource\n                const resourceMethodArgs = allArguments.slice(1);\n                \n                // Execute the method on the resource instance\n                resource[sMethodName].apply(resource, resourceMethodArgs);\n                \n                return this;\n            };\n        });\n    }\n\n    /**\n     * Alias for execute.\n     */\n    listen(request: any, response: any): void {\n        this.execute(request, response);\n    }\n\n    /**\n     * Executes the request handling logic, finding the best matching resource and handler.\n     */\n    execute(request?: any, response?: any): void {\n        request = request || getRequest();\n        const requestPath: string = request.getResourcePath();\n        const method: string = request.getMethod().toLowerCase();\n        const _oConfiguration: any = this.resourceMappings.configuration();\n\n        const matches: any[] = matchRequestUrl(requestPath, method, _oConfiguration);\n        let resourceHandler: any;\n\n        if (matches && matches[0]) {\n            const verbHandlers: any[] = _oConfiguration[matches[0].d][method];\n            if (verbHandlers) {\n                resourceHandler = verbHandlers.filter((handlerDef) => {\n                    return matchMediaType(request, handlerDef.produces, handlerDef.consumes);\n                })[0];\n            }\n        }\n\n        response = response || getResponse();\n        const queryParams: { [key: string]: any } = request.getQueryParametersMap() || {};\n        const acceptsHeader: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Accept')) || [];\n        const contentTypeHeader: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Content-Type')) || [];\n        const resourcePath: string = requestPath;\n\n        if (resourceHandler) {\n            const ctx: RequestContext = {\n                \"pathParameters\": {},\n                \"queryParameters\": {},\n                \"response\": response,\n                \"res\": response,\n                \"request\": request,\n                \"req\": request\n            };\n            if (matches[0].pathParams) {\n                ctx.pathParameters = matches[0].pathParams;\n            }\n            ctx.queryParameters = queryParams;\n\n            const noop = function () { };\n            let _before: Function, _serve: Function, _catch: Function, _finally: Function;\n            _before = resourceHandler.before || noop;\n            _serve = resourceHandler.handler || resourceHandler.serve || noop;\n            _catch = resourceHandler.catch || catchErrorHandler.bind(this, {\n                path: resourcePath,\n                method: method.toUpperCase(),\n                contentType: contentTypeHeader.join(','),\n                accepts: acceptsHeader.join(',')\n            });\n            _finally = resourceHandler.finally || noop;\n\n            const callbackArgs: any[] = [ctx, request, response, resourceHandler, this];\n\n            try {\n                logger.trace('Before serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}]', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                _before.apply(this, callbackArgs);\n\n                if (!response.isCommitted()) {\n                    logger.trace('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}]', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                    _serve.apply(this, callbackArgs);\n                    logger.trace('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] finished', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                }\n            } catch (err: any) {\n                try {\n                    callbackArgs.splice(1, 0, err);\n                    _catch.apply(this, callbackArgs);\n                } catch (_catchErr) {\n                    logger.error('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] error handler threw error', _catchErr);\n                    throw _catchErr;\n                }\n            } finally {\n                HttpController.prototype.closeResponse.call(this);\n                try {\n                    _finally.apply(this, []);\n                } catch (_finallyErr) {\n                    logger.error('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] post handler threw error', _finallyErr);\n                }\n            }\n        } else {\n            logger.error('No suitable resource handler for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] found', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n            this.sendError(response.BAD_REQUEST, undefined, 'Bad Request', 'No suitable processor for this request.');\n        }\n    }\n\n    /**\n     * Returns the ResourceMappings instance of this controller.\n     */\n    mappings(): ResourceMappings {\n        return this.resourceMappings;\n    };\n\n    /**\n     * Sends an error response to the client, formatted based on the accepted media type.\n     */\n    sendError(httpErrorCode: number, applicationErrorCode: any, errorName: string, errorDetails: string): void {\n        const clientAcceptMediaTypes: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Accept')) || ['application/json'];\n        const isHtml: boolean = clientAcceptMediaTypes.some((acceptMediaType) => isMimeTypeCompatible('*/html', acceptMediaType));\n\n        response.setStatus(httpErrorCode || response.INTERNAL_SERVER_ERROR);\n\n        if (isHtml) {\n            const message: string = errorName + (applicationErrorCode !== undefined ? '[' + applicationErrorCode + ']' : '') + (errorDetails ? ': ' + errorDetails : '');\n            response.sendError(httpErrorCode || response.INTERNAL_SERVER_ERROR, message);\n        } else {\n            const body = {\n                \"code\": applicationErrorCode,\n                \"error\": errorName,\n                \"details\": errorDetails\n            };\n            response.setHeader(\"Content-Type\", \"application/json\");\n            response.print(JSON.stringify(body, null, 2));\n        }\n        this.closeResponse();\n    };\n\n    /**\n     * Flushes and closes the HTTP response stream.\n     */\n    closeResponse(): void {\n        response.flush();\n        response.close();\n    };\n}\n\n/**\n * Custom sort function for matched route definitions, preferring exact matches over those with placeholders.\n */\nfunction matchedRouteDefinitionsSorter(p: any, n: any): number {\n    p.w = calculateMatchedRouteWeight(p);\n    n.w = calculateMatchedRouteWeight(n);\n\n    if (n.w === p.w) {\n        // The one with less placeholders wins\n        const m1 = p.d.match(/{(.*?)}/g);\n        const placeholdersCount1 = m1 !== null ? m1.length : 0;\n        const m2 = n.d.match(/{(.*?)}/g);\n        const placeholdersCount2 = m2 !== null ? m2.length : 0;\n        if (placeholdersCount1 > placeholdersCount2) {\n            n.w = n.w + 1;\n        } else if (placeholdersCount1 < placeholdersCount2) {\n            p.w = p.w + 1;\n        }\n    }\n    return n.w - p.w;\n}\n\n/**\n * Calculates the initial weight of a matched route definition.\n */\nfunction calculateMatchedRouteWeight(matchedRoute: any): number {\n    return (matchedRoute.params && Object.keys(matchedRoute.params).length > 0) ? 0 : 1; // always prefer exact route definitions - set weight to 1\n}\n\n/**\n * Transforms path parameters declared in braces (e.g., '/api/{pathParam}') to path-to-regexp format (e.g., '/api/:pathParam').\n */\nfunction transformPathParamsDeclaredInBraces(pathDefinition: string): string {\n    const pathParamsInBracesMatcher = /({(\\w*\\*?)})/g; // matches cases like '/api/{pathParam}' or '/api/{pathParam*}'\n    return pathDefinition.replace(pathParamsInBracesMatcher, \":$2\"); // transforms matched cases to '/api/:pathParam' or '/api/:pathParam*'\n}\n\n/**\n * Finds all routes in the configuration that match the request path and method.\n */\nfunction matchRequestUrl(requestPath: string, method: string, cfg: any): any[] {\n    return Object.entries(cfg)\n        .filter(([_, handlers]) => handlers && (handlers as any)[method])\n        .map(([path, _]) => path)\n        .reduce((matches: any[], path: string) => matchingRouteDefinitionsReducer(matches, path, requestPath), [])\n        .sort(matchedRouteDefinitionsSorter);\n}\n\n/**\n * Reducer function to attempt matching a defined path against the request path using path-to-regexp.\n */\nfunction matchingRouteDefinitionsReducer(matchedDefinitions: any[], definedPath: string, requestPath: string): any[] {\n    // 'match' from path-to-regexp is used here\n    const pathMatcher = match(transformPathParamsDeclaredInBraces(definedPath));\n    const matched = pathMatcher(requestPath);\n\n    if (matched) {\n        // Ensure pathParams is an object of key/value pairs\n        const pathParams = matched.params;\n\n        const matchedDefinition = {\n            p: requestPath,\n            d: definedPath,\n            pathParams: pathParams\n        };\n        matchedDefinitions.push(matchedDefinition);\n    }\n    return matchedDefinitions;\n}\n\n/**\n * Normalizes an HTTP media type header value (e.g., \"text/plain; q=0.9, application/json\").\n */\nfunction normalizeMediaTypeHeaderValue(sMediaType: string | undefined | null): string[] | undefined {\n    if (sMediaType === undefined || sMediaType === null)\n        return;\n    // convert to array of individual types\n    let arrMediaType = sMediaType.split(',');\n    arrMediaType = arrMediaType.map((mimeTypeEntry) => {\n        // remove escaping, remove quality or other attributes (e.g., '; q=0.9')\n        return mimeTypeEntry.replace('\\\\', '').split(';')[0].trim();\n    });\n    return arrMediaType.filter(type => type.length > 0);\n}\n\n/**\n * Checks if a source MIME type is compatible with a target MIME type, supporting wildcards (*).\n */\nfunction isMimeTypeCompatible(source: string, target: string): boolean {\n    if (source === target)\n        return true;\n\n    const targetM = target.split('/');\n    const sourceM = source.split('/');\n\n    // Target is wildcard type, Source has a specific subtype (e.g., target=*/json, source=application/json)\n    if (targetM[0] === '*' && targetM[1] === sourceM[1])\n        return true;\n    // Source is wildcard type, Target has a specific subtype (e.g., source=*/json, target=application/json)\n    if (sourceM[0] === '*' && targetM[1] === sourceM[1])\n        return true;\n\n    // Target is wildcard subtype, Source has a specific type (e.g., target=application/*, source=application/json)\n    if (targetM[1] === '*' && targetM[0] === sourceM[0])\n        return true;\n    // Source is wildcard subtype, Target has a specific type (e.g., source=application/*, target=application/json)\n    if (sourceM[1] === '*' && targetM[0] === sourceM[0])\n        return true;\n\n    return false;\n}\n\n/**\n * Default error handler function.\n */\nconst catchErrorHandler = function (this: HttpController, logctx: any, ctx: RequestContext, err: any, request: any, response: any): void {\n    if (ctx.suppressStack) {\n        const detailsMsg = (ctx.errorName || \"\") + (ctx.errorCode ? \" [\" + ctx.errorCode + \"]\" : \"\") + (ctx.errorMessage ? \": \" + ctx.errorMessage : \"\");\n        logger.info('Serving resource[{}], Verb[{}], Content-Type[{}], Accept[{}] finished in error. {}', logctx.path, logctx.method, logctx.contentType, logctx.accepts, detailsMsg);\n    } else {\n        logger.error('Serving resource[{}], Verb[{}], Content-Type[{}], Accept[{}] finished in error', logctx.path, logctx.method, logctx.contentType, logctx.accepts, err);\n    }\n\n    const httpErrorCode = ctx.httpErrorCode || response.INTERNAL_SERVER_ERROR;\n    const errorMessage = ctx.errorMessage || (err && err.message);\n    const errorName = ctx.errorName || (err && err.name);\n    const errorCode = ctx.errorCode;\n\n    this.sendError(httpErrorCode, errorCode, errorName, errorMessage);\n};\n\n/**\n * Checks if the request media types match the resource handler's consumes and produces constraints.\n */\nconst matchMediaType = function (request: any, producesMediaTypes: string[], consumesMediaTypes: string[]): boolean {\n    let isProduceMatched = false;\n    const acceptsMediaTypes = normalizeMediaTypeHeaderValue(request.getHeader('Accept'));\n\n    // 1. Check Produces (Accepts header)\n    if (!acceptsMediaTypes || acceptsMediaTypes.length === 0 || acceptsMediaTypes.includes('*/*')) {\n        // Output media type is not restricted by client or client accepts anything\n        isProduceMatched = true;\n    } else {\n        if (producesMediaTypes && producesMediaTypes.length) {\n            const matchedProducesMIME = acceptsMediaTypes.filter((acceptsMediaType) => {\n                return producesMediaTypes.some((producesMediaType) => {\n                    return isMimeTypeCompatible(acceptsMediaType, producesMediaType);\n                });\n            });\n            isProduceMatched = matchedProducesMIME && matchedProducesMIME.length > 0;\n        } else {\n            // Resource doesn't specify produces, so it matches if the client accepts anything or if produces is an empty array\n            isProduceMatched = true;\n        }\n    }\n\n    // 2. Check Consumes (Content-Type header)\n    let isConsumeMatched = false;\n    const contentTypeMediaTypes = normalizeMediaTypeHeaderValue(request.getContentType());\n\n    if (!consumesMediaTypes || consumesMediaTypes.length === 0 || consumesMediaTypes.includes('*/*')) {\n        // Input media type is not restricted by resource or resource accepts anything\n        isConsumeMatched = true;\n    } else {\n        if (contentTypeMediaTypes && consumesMediaTypes && consumesMediaTypes.length) {\n            const matchedConsumesMIME = contentTypeMediaTypes.filter((contentTypeMediaType) => {\n                return consumesMediaTypes.some((consumesMediaType) => {\n                    return isMimeTypeCompatible(contentTypeMediaType, consumesMediaType);\n                });\n            });\n            isConsumeMatched = matchedConsumesMIME && matchedConsumesMIME.length > 0;\n        }\n    }\n\n    return isProduceMatched && isConsumeMatched;\n};\n"],
  "mappings": "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",
  "names": ["request", "response"]
}

|
|
288
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/http/rs/resource-http-controller.ts"],
  "sourcesContent": ["import { Response as response } from \"../response\";\nimport { Request as request } from \"../request\";\nimport { ResourceMappings } from \"./resource-mappings\";\nimport { Logging } from \"@aerokit/sdk/log\";\n\n// Declaration for the external dirigibleRequire function and its dependency\ndeclare function dirigibleRequire(module: string): any;\nconst { match } = dirigibleRequire(\"modules/src/http/path-to-regexp/6.2.1/index.js\");\n\nconst logger: any = Logging.getLogger('http.rs.controller');\n\n/**\n * Interface for the context object passed to handler functions (before, serve, catch).\n */\ninterface RequestContext {\n    pathParameters: { [key: string]: any };\n    queryParameters: { [key: string]: any };\n    response: any;\n    res: any;\n    request: any;\n    req: any;\n    // Context properties for error handling might also be attached here\n    suppressStack?: boolean;\n    httpErrorCode?: number;\n    errorMessage?: string;\n    errorName?: string;\n    errorCode?: any;\n}\n\nfunction getRequest(): any {\n    return request;\n}\nfunction getResponse(): any {\n    return response;\n}\n\n/**\n * Creates a service (HttpController) instance, optionally initialized with oMappings.\n *\n * @param oConfig Configuration object or configuration builder with configuration() getter function.\n * @returns A new HttpController instance.\n */\nexport function service(oConfig?: any): HttpController {\n    let config: ResourceMappings | any;\n    if (oConfig !== undefined) {\n        if (typeof oConfig === 'object' || oConfig instanceof ResourceMappings) {\n            config = oConfig;\n        }\n    }\n    return new HttpController(config);\n}\n\n/**\n * The main class for handling HTTP requests and routing them to the correct resource handlers.\n */\nexport class HttpController {\n\n    resource: Function;\n    resourcePath: Function;\n    resourceMappings: ResourceMappings;\n\n    // Index signature to allow dynamic method assignment in the constructor\n    [key: string]: any;\n\n    /**\n     * Constructor function for HttpController instances.\n     *\n     * @param oMappings The mappings configuration for this controller.\n     */\n    constructor(oMappings?: ResourceMappings | any) {\n        if (oMappings instanceof ResourceMappings) {\n            this.resourceMappings = oMappings;\n        } else if (typeof oMappings === 'object' || oMappings === undefined) {\n            this.resourceMappings = new ResourceMappings(oMappings, this);\n        } else {\n            // Default initialization if input is unexpected\n            this.resourceMappings = new ResourceMappings({}, this);\n        }\n\n        // Alias for resourceMappings.resource\n        this.resource = this.resourcePath = this.resourceMappings.resourcePath.bind(this.resourceMappings);\n\n        // weave-in HTTP method-based factory functions - shortcut for service().resource(sPath).method\n        ['get', 'post', 'put', 'delete', 'remove', 'method'].forEach((sMethodName: string) => {\n            this[sMethodName] = (...allArguments: any[]): HttpController => {\n                if (allArguments.length < 1)\n                    throw Error('Insufficient arguments provided to HttpController method ' + sMethodName + '.');\n\n                // sPath is always the first argument\n                let sPath = allArguments[0];\n                if (sPath === undefined)\n                    sPath = \"\";\n\n                // The next arguments (sVerb, arrConsumes, arrProduces) are used by ResourceMappings.find/resource\n                const sVerb = allArguments[1];\n                const arrConsumes = allArguments[2];\n                const arrProduces = allArguments[3];\n\n                // Find existing resource or create a new one based on the path (and other optional constraints)\n                const resource: any = this.resourceMappings.find(sPath, sVerb, arrConsumes, arrProduces) || this.resourceMappings.resource(sPath);\n\n                // Pass all arguments *except* the sPath to the corresponding method of the resource\n                const resourceMethodArgs = allArguments.slice(1);\n                \n                // Execute the method on the resource instance\n                resource[sMethodName].apply(resource, resourceMethodArgs);\n                \n                return this;\n            };\n        });\n    }\n\n    /**\n     * Alias for execute.\n     */\n    listen(request: any, response: any): void {\n        this.execute(request, response);\n    }\n\n    /**\n     * Executes the request handling logic, finding the best matching resource and handler.\n     */\n    execute(request?: any, response?: any): void {\n        request = request || getRequest();\n        const requestPath: string = request.getResourcePath();\n        const method: string = request.getMethod().toLowerCase();\n        const _oConfiguration: any = this.resourceMappings.configuration();\n\n        const matches: any[] = matchRequestUrl(requestPath, method, _oConfiguration);\n        let resourceHandler: any;\n\n        if (matches && matches[0]) {\n            const verbHandlers: any[] = _oConfiguration[matches[0].d][method];\n            if (verbHandlers) {\n                resourceHandler = verbHandlers.filter((handlerDef) => {\n                    return matchMediaType(request, handlerDef.produces, handlerDef.consumes);\n                })[0];\n            }\n        }\n\n        response = response || getResponse();\n        const queryParams: { [key: string]: any } = request.getQueryParametersMap() || {};\n        const acceptsHeader: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Accept')) || [];\n        const contentTypeHeader: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Content-Type')) || [];\n        const resourcePath: string = requestPath;\n\n        if (resourceHandler) {\n            const ctx: RequestContext = {\n                \"pathParameters\": {},\n                \"queryParameters\": {},\n                \"response\": response,\n                \"res\": response,\n                \"request\": request,\n                \"req\": request\n            };\n            if (matches[0].pathParams) {\n                ctx.pathParameters = matches[0].pathParams;\n            }\n            ctx.queryParameters = queryParams;\n\n            const noop = function () { };\n            let _before: Function, _serve: Function, _catch: Function, _finally: Function;\n            _before = resourceHandler.before || noop;\n            _serve = resourceHandler.handler || resourceHandler.serve || noop;\n            _catch = resourceHandler.catch || catchErrorHandler.bind(this, {\n                path: resourcePath,\n                method: method.toUpperCase(),\n                contentType: contentTypeHeader.join(','),\n                accepts: acceptsHeader.join(',')\n            });\n            _finally = resourceHandler.finally || noop;\n\n            const callbackArgs: any[] = [ctx, request, response, resourceHandler, this];\n\n            try {\n                logger.trace('Before serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}]', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                _before.apply(this, callbackArgs);\n\n                if (!response.isCommitted()) {\n                    logger.trace('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}]', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                    _serve.apply(this, callbackArgs);\n                    logger.trace('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] finished', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n                }\n            } catch (err: any) {\n                try {\n                    callbackArgs.splice(1, 0, err);\n                    _catch.apply(this, callbackArgs);\n                } catch (_catchErr) {\n                    logger.error('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] error handler threw error', _catchErr);\n                    throw _catchErr;\n                }\n            } finally {\n                HttpController.prototype.closeResponse.call(this);\n                try {\n                    _finally.apply(this, []);\n                } catch (_finallyErr) {\n                    logger.error('Serving request for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] post handler threw error', _finallyErr);\n                }\n            }\n        } else {\n            logger.error('No suitable resource handler for Resource[{}], Method[{}], Content-Type[{}], Accept[{}] found', resourcePath, method.toUpperCase(), contentTypeHeader, acceptsHeader);\n\t\t\tlogger.error('Registered resource handlers for Resource[{}] are [{}]', resourcePath, JSON.stringify(_oConfiguration));\n            this.sendError(response.BAD_REQUEST, undefined, 'Bad Request', 'No suitable processor for this request.');\n        }\n    }\n\n    /**\n     * Returns the ResourceMappings instance of this controller.\n     */\n    mappings(): ResourceMappings {\n        return this.resourceMappings;\n    };\n\n    /**\n     * Sends an error response to the client, formatted based on the accepted media type.\n     */\n    sendError(httpErrorCode: number, applicationErrorCode: any, errorName: string, errorDetails: string): void {\n        const clientAcceptMediaTypes: string[] = normalizeMediaTypeHeaderValue(request.getHeader('Accept')) || ['application/json'];\n        const isHtml: boolean = clientAcceptMediaTypes.some((acceptMediaType) => isMimeTypeCompatible('*/html', acceptMediaType));\n\n        response.setStatus(httpErrorCode || response.INTERNAL_SERVER_ERROR);\n\n        if (isHtml) {\n            const message: string = errorName + (applicationErrorCode !== undefined ? '[' + applicationErrorCode + ']' : '') + (errorDetails ? ': ' + errorDetails : '');\n            response.sendError(httpErrorCode || response.INTERNAL_SERVER_ERROR, message);\n        } else {\n            const body = {\n                \"code\": applicationErrorCode,\n                \"error\": errorName,\n                \"details\": errorDetails\n            };\n            response.setHeader(\"Content-Type\", \"application/json\");\n            response.print(JSON.stringify(body, null, 2));\n        }\n        this.closeResponse();\n    };\n\n    /**\n     * Flushes and closes the HTTP response stream.\n     */\n    closeResponse(): void {\n        response.flush();\n        response.close();\n    };\n}\n\n/**\n * Custom sort function for matched route definitions, preferring exact matches over those with placeholders.\n */\nfunction matchedRouteDefinitionsSorter(p: any, n: any): number {\n    p.w = calculateMatchedRouteWeight(p);\n    n.w = calculateMatchedRouteWeight(n);\n\n    if (n.w === p.w) {\n        // The one with less placeholders wins\n        const m1 = p.d.match(/{(.*?)}/g);\n        const placeholdersCount1 = m1 !== null ? m1.length : 0;\n        const m2 = n.d.match(/{(.*?)}/g);\n        const placeholdersCount2 = m2 !== null ? m2.length : 0;\n        if (placeholdersCount1 > placeholdersCount2) {\n            n.w = n.w + 1;\n        } else if (placeholdersCount1 < placeholdersCount2) {\n            p.w = p.w + 1;\n        }\n    }\n    return n.w - p.w;\n}\n\n/**\n * Calculates the initial weight of a matched route definition.\n */\nfunction calculateMatchedRouteWeight(matchedRoute: any): number {\n    return (matchedRoute.params && Object.keys(matchedRoute.params).length > 0) ? 0 : 1; // always prefer exact route definitions - set weight to 1\n}\n\n/**\n * Transforms path parameters declared in braces (e.g., '/api/{pathParam}') to path-to-regexp format (e.g., '/api/:pathParam').\n */\nfunction transformPathParamsDeclaredInBraces(pathDefinition: string): string {\n    const pathParamsInBracesMatcher = /({(\\w*\\*?)})/g; // matches cases like '/api/{pathParam}' or '/api/{pathParam*}'\n    return pathDefinition.replace(pathParamsInBracesMatcher, \":$2\"); // transforms matched cases to '/api/:pathParam' or '/api/:pathParam*'\n}\n\n/**\n * Finds all routes in the configuration that match the request path and method.\n */\nfunction matchRequestUrl(requestPath: string, method: string, cfg: any): any[] {\n    return Object.entries(cfg)\n        .filter(([_, handlers]) => handlers && (handlers as any)[method])\n        .map(([path, _]) => path)\n        .reduce((matches: any[], path: string) => matchingRouteDefinitionsReducer(matches, path, requestPath), [])\n        .sort(matchedRouteDefinitionsSorter);\n}\n\n/**\n * Reducer function to attempt matching a defined path against the request path using path-to-regexp.\n */\nfunction matchingRouteDefinitionsReducer(matchedDefinitions: any[], definedPath: string, requestPath: string): any[] {\n    // 'match' from path-to-regexp is used here\n    const pathMatcher = match(transformPathParamsDeclaredInBraces(definedPath));\n    const matched = pathMatcher(requestPath);\n\n    if (matched) {\n        // Ensure pathParams is an object of key/value pairs\n        const pathParams = matched.params;\n\n        const matchedDefinition = {\n            p: requestPath,\n            d: definedPath,\n            pathParams: pathParams\n        };\n        matchedDefinitions.push(matchedDefinition);\n    }\n    return matchedDefinitions;\n}\n\n/**\n * Normalizes an HTTP media type header value (e.g., \"text/plain; q=0.9, application/json\").\n */\nfunction normalizeMediaTypeHeaderValue(sMediaType: string | undefined | null): string[] | undefined {\n    if (sMediaType === undefined || sMediaType === null)\n        return;\n    // convert to array of individual types\n    let arrMediaType = sMediaType.split(',');\n    arrMediaType = arrMediaType.map((mimeTypeEntry) => {\n        // remove escaping, remove quality or other attributes (e.g., '; q=0.9')\n        return mimeTypeEntry.replace('\\\\', '').split(';')[0].trim();\n    });\n    return arrMediaType.filter(type => type.length > 0);\n}\n\n/**\n * Checks if a source MIME type is compatible with a target MIME type, supporting wildcards (*).\n */\nfunction isMimeTypeCompatible(source: string, target: string): boolean {\n    if (source === target)\n        return true;\n\n    const targetM = target.split('/');\n    const sourceM = source.split('/');\n\n    // Target is wildcard type, Source has a specific subtype (e.g., target=*/json, source=application/json)\n    if (targetM[0] === '*' && targetM[1] === sourceM[1])\n        return true;\n    // Source is wildcard type, Target has a specific subtype (e.g., source=*/json, target=application/json)\n    if (sourceM[0] === '*' && targetM[1] === sourceM[1])\n        return true;\n\n    // Target is wildcard subtype, Source has a specific type (e.g., target=application/*, source=application/json)\n    if (targetM[1] === '*' && targetM[0] === sourceM[0])\n        return true;\n    // Source is wildcard subtype, Target has a specific type (e.g., source=application/*, target=application/json)\n    if (sourceM[1] === '*' && targetM[0] === sourceM[0])\n        return true;\n\n    return false;\n}\n\n/**\n * Default error handler function.\n */\nconst catchErrorHandler = function (this: HttpController, logctx: any, ctx: RequestContext, err: any, request: any, response: any): void {\n    if (ctx.suppressStack) {\n        const detailsMsg = (ctx.errorName || \"\") + (ctx.errorCode ? \" [\" + ctx.errorCode + \"]\" : \"\") + (ctx.errorMessage ? \": \" + ctx.errorMessage : \"\");\n        logger.info('Serving resource[{}], Verb[{}], Content-Type[{}], Accept[{}] finished in error. {}', logctx.path, logctx.method, logctx.contentType, logctx.accepts, detailsMsg);\n    } else {\n        logger.error('Serving resource[{}], Verb[{}], Content-Type[{}], Accept[{}] finished in error', logctx.path, logctx.method, logctx.contentType, logctx.accepts, err);\n    }\n\n    const httpErrorCode = ctx.httpErrorCode || response.INTERNAL_SERVER_ERROR;\n    const errorMessage = ctx.errorMessage || (err && err.message);\n    const errorName = ctx.errorName || (err && err.name);\n    const errorCode = ctx.errorCode;\n\n    this.sendError(httpErrorCode, errorCode, errorName, errorMessage);\n};\n\n/**\n * Checks if the request media types match the resource handler's consumes and produces constraints.\n */\nconst matchMediaType = function (request: any, producesMediaTypes: string[], consumesMediaTypes: string[]): boolean {\n    let isProduceMatched = false;\n    const acceptsMediaTypes = normalizeMediaTypeHeaderValue(request.getHeader('Accept'));\n\n    // 1. Check Produces (Accepts header)\n    if (!acceptsMediaTypes || acceptsMediaTypes.length === 0 || acceptsMediaTypes.includes('*/*')) {\n        // Output media type is not restricted by client or client accepts anything\n        isProduceMatched = true;\n    } else {\n        if (producesMediaTypes && producesMediaTypes.length) {\n            const matchedProducesMIME = acceptsMediaTypes.filter((acceptsMediaType) => {\n                return producesMediaTypes.some((producesMediaType) => {\n                    return isMimeTypeCompatible(acceptsMediaType, producesMediaType);\n                });\n            });\n            isProduceMatched = matchedProducesMIME && matchedProducesMIME.length > 0;\n        } else {\n            // Resource doesn't specify produces, so it matches if the client accepts anything or if produces is an empty array\n            isProduceMatched = true;\n        }\n    }\n\n    // 2. Check Consumes (Content-Type header)\n    let isConsumeMatched = false;\n    const contentTypeMediaTypes = normalizeMediaTypeHeaderValue(request.getContentType());\n\n    if (!consumesMediaTypes || consumesMediaTypes.length === 0 || consumesMediaTypes.includes('*/*')) {\n        // Input media type is not restricted by resource or resource accepts anything\n        isConsumeMatched = true;\n    } else {\n        if (contentTypeMediaTypes && consumesMediaTypes && consumesMediaTypes.length) {\n            const matchedConsumesMIME = contentTypeMediaTypes.filter((contentTypeMediaType) => {\n                return consumesMediaTypes.some((consumesMediaType) => {\n                    return isMimeTypeCompatible(contentTypeMediaType, consumesMediaType);\n                });\n            });\n            isConsumeMatched = matchedConsumesMIME && matchedConsumesMIME.length > 0;\n        }\n    }\n\n    return isProduceMatched && isConsumeMatched;\n};\n"],
  "mappings": "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",
  "names": ["request", "response"]
}

|