A generic binary http receiver for reusable purposes

Signed-off-by: Fabio José <fabiojose@gmail.com>
This commit is contained in:
Fabio José 2019-07-29 16:33:48 -03:00
parent 49119d707f
commit ef6db28464
2 changed files with 133 additions and 94 deletions

View File

@ -0,0 +1,122 @@
const Constants = require("./constants.js");
const Commons = require("./commons.js");
const Cloudevent = require("../../cloudevent.js");
const {
isDefinedOrThrow,
isStringOrObjectOrThrow
} = require("../../utils/fun.js");
function validateArgs(payload, attributes) {
Array.of(payload)
.filter((p) => isDefinedOrThrow(p,
{message: "payload is null or undefined"}))
.filter((p) => isStringOrObjectOrThrow(p,
{message: "payload must be an object or a string"}))
.shift();
Array.of(attributes)
.filter((a) => isDefinedOrThrow(a,
{message: "attributes is null or undefined"}))
.shift();
}
function BinaryHTTPReceiver(
parserByType,
setterByHeader,
allowedContentTypes,
requiredHeaders,
Spec) {
this.parserByType = parserByType;
this.setterByHeader = setterByHeader;
this.allowedContentTypes = allowedContentTypes;
this.requiredHeaders = requiredHeaders;
this.Spec = Spec;
this.spec = new Spec();
}
BinaryHTTPReceiver.prototype.check = function(payload, headers) {
// Validation Level 0
validateArgs(payload, headers);
// Clone and low case all headers names
var sanityHeaders = Commons.sanityAndClone(headers);
// Validation Level 1
if(!this.allowedContentTypes
.includes(sanityHeaders[Constants.HEADER_CONTENT_TYPE])){
throw {
message: "invalid content type",
errors: [sanityHeaders[Constants.HEADER_CONTENT_TYPE]]
};
}
this.requiredHeaders
.filter((required) => !sanityHeaders[required])
.forEach((required) => {
throw {message: "header '" + required + "' not found"};
});
if(sanityHeaders[Constants.BINARY_HEADERS_02.SPEC_VERSION] !== "0.2"){
throw {
message: "invalid spec version",
errors: [sanityHeaders[Constants.BINARY_HEADERS_02.SPEC_VERSION]]
};
}
// No erros! Its contains the minimum required attributes
};
BinaryHTTPReceiver.prototype.parse = function(payload, headers) {
this.check(payload, headers);
// Clone and low case all headers names
var sanityHeaders = Commons.sanityAndClone(headers);
var processedHeaders = [];
var cloudevent = new Cloudevent(this.Spec);
// dont worry, check() have seen what was required or not
Array.from(Object.keys(this.setterByHeader))
.filter((header) => sanityHeaders[header])
.forEach((header) => {
var setterName = this.setterByHeader[header].name;
var parserFun = this.setterByHeader[header].parser;
// invoke the setter function
cloudevent[setterName](parserFun(sanityHeaders[header]));
// to use ahead, for extensions processing
processedHeaders.push(header);
});
// Parses the payload
var parsedPayload =
this.parserByType[sanityHeaders[Constants.HEADER_CONTENT_TYPE]]
.parse(payload);
// Every unprocessed header can be an extension
Array.from(Object.keys(sanityHeaders))
.filter((value) => !processedHeaders.includes(value))
.filter((value) =>
value.startsWith(Constants.BINARY_HEADERS_02.EXTENSIONS_PREFIX))
.map((extension) =>
extension.substring(Constants.BINARY_HEADERS_02.EXTENSIONS_PREFIX.length)
).forEach((extension) =>
cloudevent.addExtension(extension,
sanityHeaders[Constants.BINARY_HEADERS_02.EXTENSIONS_PREFIX+extension])
);
// Sets the data
cloudevent.data(parsedPayload);
// Checks the event spec
cloudevent.format();
// return the result
return cloudevent;
};
module.exports = BinaryHTTPReceiver;

View File

@ -1,10 +1,10 @@
const Constants = require("./constants.js");
const Commons = require("./commons.js");
const Cloudevent = require("../../cloudevent.js");
const Spec02 = require("../../specs/spec_0_2.js");
const JSONParser = require("../../formats/json/parser.js");
const BinaryHTTPReceiver = require("./receiver_binary.js");
const {
isDefinedOrThrow,
isStringOrObjectOrThrow
@ -56,105 +56,22 @@ setterByHeader[Constants.HEADER_CONTENT_TYPE] = {
parser: (v) => v
};
function validateArgs(payload, attributes) {
Array.of(payload)
.filter((p) => isDefinedOrThrow(p,
{message: "payload is null or undefined"}))
.filter((p) => isStringOrObjectOrThrow(p,
{message: "payload must be an object or a string"}))
.shift();
Array.of(attributes)
.filter((a) => isDefinedOrThrow(a,
{message: "attributes is null or undefined"}))
.shift();
}
function Receiver(configuration) {
this.receiver = new BinaryHTTPReceiver(
parserByType,
setterByHeader,
allowedContentTypes,
requiredHeaders,
Spec02
);
}
Receiver.prototype.check = function(payload, headers) {
// Validation Level 0
validateArgs(payload, headers);
// Clone and low case all headers names
var sanityHeaders = Commons.sanityAndClone(headers);
// Validation Level 1
if(!allowedContentTypes
.includes(sanityHeaders[Constants.HEADER_CONTENT_TYPE])){
throw {
message: "invalid content type",
errors: [sanityHeaders[Constants.HEADER_CONTENT_TYPE]]
};
}
requiredHeaders
.filter((required) => !sanityHeaders[required])
.forEach((required) => {
throw {message: "header '" + required + "' not found"};
});
if(sanityHeaders[Constants.BINARY_HEADERS_02.SPEC_VERSION] !== "0.2"){
throw {
message: "invalid spec version",
errors: [sanityHeaders[Constants.BINARY_HEADERS_02.SPEC_VERSION]]
};
}
// No erros! Its contains the minimum required attributes
this.receiver.check(payload, headers);
};
Receiver.prototype.parse = function(payload, headers) {
this.check(payload, headers);
// Clone and low case all headers names
var sanityHeaders = Commons.sanityAndClone(headers);
var processedHeaders = [];
var cloudevent = new Cloudevent(Spec02);
// dont worry, check() have seen what was required or not
Array.from(Object.keys(setterByHeader))
.filter((header) => sanityHeaders[header])
.forEach((header) => {
var setterName = setterByHeader[header].name;
var parserFun = setterByHeader[header].parser;
// invoke the setter function
cloudevent[setterName](parserFun(sanityHeaders[header]));
// to use ahead, for extensions processing
processedHeaders.push(header);
});
// Parses the payload
var parsedPayload =
parserByType[sanityHeaders[Constants.HEADER_CONTENT_TYPE]]
.parse(payload);
// Every unprocessed header can be an extension
Array.from(Object.keys(sanityHeaders))
.filter((value) => !processedHeaders.includes(value))
.filter((value) =>
value.startsWith(Constants.BINARY_HEADERS_02.EXTENSIONS_PREFIX))
.map((extension) =>
extension.substring(Constants.BINARY_HEADERS_02.EXTENSIONS_PREFIX.length)
).forEach((extension) =>
cloudevent.addExtension(extension,
sanityHeaders[Constants.BINARY_HEADERS_02.EXTENSIONS_PREFIX+extension])
);
// Sets the data
cloudevent.data(parsedPayload);
// Checks the event spec
cloudevent.format();
// return the result
return cloudevent;
return this.receiver.parse(payload, headers);
};
module.exports = Receiver;