UNPKG

@decaf-ts/db-decorators

Version:

Agnostic database decorators and repository

151 lines 14.9 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.ConflictError = exports.NotFoundError = exports.SerializationError = exports.InternalError = exports.ValidationError = exports.BadRequestError = exports.BaseError = void 0; /** * @description Base error class for the repository module * @summary Abstract base error class that all other error types extend from. Provides common error handling functionality and standardized HTTP code mapping. * @param {string} name - The name of the error * @param {string|Error} msg - The error message or Error object to wrap * @param {number} code - The HTTP status code associated with this error * @return {void} * @class BaseError * @example * // This is an abstract class and should not be instantiated directly * // Instead, use one of the concrete error classes: * throw new ValidationError('Invalid data provided'); * @mermaid * sequenceDiagram * participant C as Caller * participant E as BaseError * C->>E: new BaseError(name,msg,code) * E-->>C: Error instance with message and code * @category Errors */ class BaseError extends Error { constructor(name, msg, code) { if (msg instanceof BaseError) return msg; const message = `[${name}] ${msg instanceof Error ? msg.message : msg}`; super(message); this.code = code; if (msg instanceof Error) this.stack = msg.stack; } } exports.BaseError = BaseError; /** * @description Error (40) thrown when a bad request is received * @summary Represents a failure in the input data, typically thrown when a client sends invalid or incomplete data * @param {string|Error} msg - The error message or Error object * @return {BadRequestError} A new BadRequestError instance * @class BadRequestError * @category Errors */ class BadRequestError extends BaseError { constructor(msg, name = BadRequestError.name, code = 400) { super(name, msg, code); } } exports.BadRequestError = BadRequestError; /** * @description Error (422) thrown when validation fails * @summary Represents a failure in the Model details, typically thrown when data validation fails * @param {string|Error} msg - The error message or Error object * @return {ValidationError} A new ValidationError instance * @class ValidationError * @example * // Throw a validation error when data is invalid * if (!isValid(data)) { * throw new ValidationError('Invalid data format'); * } * @category Errors */ class ValidationError extends BadRequestError { constructor(msg) { super(msg, ValidationError.name, 422); } } exports.ValidationError = ValidationError; /** * @description Error (500) thrown for internal system failures * @summary Represents an internal failure (should mean an error in code) with HTTP 500 status code * @param {string|Error} msg - The error message or Error object * @return {InternalError} A new InternalError instance * @class InternalError * @example * // Throw an internal error when an unexpected condition occurs * try { * // Some operation * } catch (error) { * throw new InternalError('Unexpected internal error occurred'); * } * @category Errors */ class InternalError extends BaseError { constructor(msg, name = InternalError.name, code = 500) { super(name, msg, code); } } exports.InternalError = InternalError; /** * @description Error (500) thrown when serialization or deserialization fails * @summary Represents a failure in the Model de/serialization, typically when converting between data formats * @param {string|Error} msg - The error message or Error object * @return {SerializationError} A new SerializationError instance * @class SerializationError * @example * // Throw a serialization error when JSON parsing fails * try { * const data = JSON.parse(invalidJson); * } catch (error) { * throw new SerializationError('Failed to parse JSON data'); * } * @category Errors */ class SerializationError extends InternalError { constructor(msg) { super(msg, SerializationError.name, 500); } } exports.SerializationError = SerializationError; /** * @description Error thrown when a requested resource is not found * @summary Represents a failure in finding a model, resulting in a 404 HTTP status code * @param {string|Error} msg - The error message or Error object * @return {NotFoundError} A new NotFoundError instance * @class NotFoundError * @example * // Throw a not found error when a record doesn't exist * const user = await repository.findById(id); * if (!user) { * throw new NotFoundError(`User with ID ${id} not found`); * } * @category Errors */ class NotFoundError extends BadRequestError { constructor(msg) { super(msg, NotFoundError.name, 404); } } exports.NotFoundError = NotFoundError; /** * @description Error thrown when a conflict occurs in the storage * @summary Represents a conflict in the storage, typically when trying to create a duplicate resource * @param {string|Error} msg - The error message or Error object * @return {ConflictError} A new ConflictError instance * @class ConflictError * @example * // Throw a conflict error when trying to create a duplicate record * const existingUser = await repository.findByEmail(email); * if (existingUser) { * throw new ConflictError(`User with email ${email} already exists`); * } * @category Errors */ class ConflictError extends BadRequestError { constructor(msg) { super(msg, ConflictError.name, 409); } } exports.ConflictError = ConflictError; //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXJyb3JzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3JlcG9zaXRvcnkvZXJyb3JzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBbUJHO0FBQ0gsTUFBc0IsU0FBVSxTQUFRLEtBQUs7SUFFM0MsWUFBc0IsSUFBWSxFQUFFLEdBQW1CLEVBQUUsSUFBWTtRQUNuRSxJQUFJLEdBQUcsWUFBWSxTQUFTO1lBQUUsT0FBTyxHQUFHLENBQUM7UUFDekMsTUFBTSxPQUFPLEdBQUcsSUFBSSxJQUFJLEtBQUssR0FBRyxZQUFZLEtBQUssQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsR0FBRyxFQUFFLENBQUM7UUFDeEUsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQ2YsSUFBSSxDQUFDLElBQUksR0FBRyxJQUFJLENBQUM7UUFDakIsSUFBSSxHQUFHLFlBQVksS0FBSztZQUFFLElBQUksQ0FBQyxLQUFLLEdBQUcsR0FBRyxDQUFDLEtBQUssQ0FBQztJQUNuRCxDQUFDO0NBQ0Y7QUFURCw4QkFTQztBQUVEOzs7Ozs7O0dBT0c7QUFDSCxNQUFhLGVBQWdCLFNBQVEsU0FBUztJQUM1QyxZQUFZLEdBQW1CLEVBQUUsSUFBSSxHQUFHLGVBQWUsQ0FBQyxJQUFJLEVBQUUsSUFBSSxHQUFHLEdBQUc7UUFDdEUsS0FBSyxDQUFDLElBQUksRUFBRSxHQUFHLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDekIsQ0FBQztDQUNGO0FBSkQsMENBSUM7QUFFRDs7Ozs7Ozs7Ozs7O0dBWUc7QUFDSCxNQUFhLGVBQWdCLFNBQVEsZUFBZTtJQUNsRCxZQUFZLEdBQW1CO1FBQzdCLEtBQUssQ0FBQyxHQUFHLEVBQUUsZUFBZSxDQUFDLElBQUksRUFBRSxHQUFHLENBQUMsQ0FBQztJQUN4QyxDQUFDO0NBQ0Y7QUFKRCwwQ0FJQztBQUNEOzs7Ozs7Ozs7Ozs7OztHQWNHO0FBQ0gsTUFBYSxhQUFjLFNBQVEsU0FBUztJQUMxQyxZQUFZLEdBQW1CLEVBQUUsSUFBSSxHQUFHLGFBQWEsQ0FBQyxJQUFJLEVBQUUsSUFBSSxHQUFHLEdBQUc7UUFDcEUsS0FBSyxDQUFDLElBQUksRUFBRSxHQUFHLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDekIsQ0FBQztDQUNGO0FBSkQsc0NBSUM7QUFDRDs7Ozs7Ozs7Ozs7Ozs7R0FjRztBQUNILE1BQWEsa0JBQW1CLFNBQVEsYUFBYTtJQUNuRCxZQUFZLEdBQW1CO1FBQzdCLEtBQUssQ0FBQyxHQUFHLEVBQUUsa0JBQWtCLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxDQUFDO0lBQzNDLENBQUM7Q0FDRjtBQUpELGdEQUlDO0FBRUQ7Ozs7Ozs7Ozs7Ozs7R0FhRztBQUNILE1BQWEsYUFBYyxTQUFRLGVBQWU7SUFDaEQsWUFBWSxHQUFtQjtRQUM3QixLQUFLLENBQUMsR0FBRyxFQUFFLGFBQWEsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLENBQUM7SUFDdEMsQ0FBQztDQUNGO0FBSkQsc0NBSUM7QUFDRDs7Ozs7Ozs7Ozs7OztHQWFHO0FBQ0gsTUFBYSxhQUFjLFNBQVEsZUFBZTtJQUNoRCxZQUFZLEdBQW1CO1FBQzdCLEtBQUssQ0FBQyxHQUFHLEVBQUUsYUFBYSxDQUFDLElBQUksRUFBRSxHQUFHLENBQUMsQ0FBQztJQUN0QyxDQUFDO0NBQ0Y7QUFKRCxzQ0FJQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQGRlc2NyaXB0aW9uIEJhc2UgZXJyb3IgY2xhc3MgZm9yIHRoZSByZXBvc2l0b3J5IG1vZHVsZVxuICogQHN1bW1hcnkgQWJzdHJhY3QgYmFzZSBlcnJvciBjbGFzcyB0aGF0IGFsbCBvdGhlciBlcnJvciB0eXBlcyBleHRlbmQgZnJvbS4gUHJvdmlkZXMgY29tbW9uIGVycm9yIGhhbmRsaW5nIGZ1bmN0aW9uYWxpdHkgYW5kIHN0YW5kYXJkaXplZCBIVFRQIGNvZGUgbWFwcGluZy5cbiAqIEBwYXJhbSB7c3RyaW5nfSBuYW1lIC0gVGhlIG5hbWUgb2YgdGhlIGVycm9yXG4gKiBAcGFyYW0ge3N0cmluZ3xFcnJvcn0gbXNnIC0gVGhlIGVycm9yIG1lc3NhZ2Ugb3IgRXJyb3Igb2JqZWN0IHRvIHdyYXBcbiAqIEBwYXJhbSB7bnVtYmVyfSBjb2RlIC0gVGhlIEhUVFAgc3RhdHVzIGNvZGUgYXNzb2NpYXRlZCB3aXRoIHRoaXMgZXJyb3JcbiAqIEByZXR1cm4ge3ZvaWR9XG4gKiBAY2xhc3MgQmFzZUVycm9yXG4gKiBAZXhhbXBsZVxuICogLy8gVGhpcyBpcyBhbiBhYnN0cmFjdCBjbGFzcyBhbmQgc2hvdWxkIG5vdCBiZSBpbnN0YW50aWF0ZWQgZGlyZWN0bHlcbiAqIC8vIEluc3RlYWQsIHVzZSBvbmUgb2YgdGhlIGNvbmNyZXRlIGVycm9yIGNsYXNzZXM6XG4gKiB0aHJvdyBuZXcgVmFsaWRhdGlvbkVycm9yKCdJbnZhbGlkIGRhdGEgcHJvdmlkZWQnKTtcbiAqIEBtZXJtYWlkXG4gKiBzZXF1ZW5jZURpYWdyYW1cbiAqICAgcGFydGljaXBhbnQgQyBhcyBDYWxsZXJcbiAqICAgcGFydGljaXBhbnQgRSBhcyBCYXNlRXJyb3JcbiAqICAgQy0+PkU6IG5ldyBCYXNlRXJyb3IobmFtZSxtc2csY29kZSlcbiAqICAgRS0tPj5DOiBFcnJvciBpbnN0YW5jZSB3aXRoIG1lc3NhZ2UgYW5kIGNvZGVcbiAqIEBjYXRlZ29yeSBFcnJvcnNcbiAqL1xuZXhwb3J0IGFic3RyYWN0IGNsYXNzIEJhc2VFcnJvciBleHRlbmRzIEVycm9yIHtcbiAgcmVhZG9ubHkgY29kZSE6IG51bWJlcjtcbiAgcHJvdGVjdGVkIGNvbnN0cnVjdG9yKG5hbWU6IHN0cmluZywgbXNnOiBzdHJpbmcgfCBFcnJvciwgY29kZTogbnVtYmVyKSB7XG4gICAgaWYgKG1zZyBpbnN0YW5jZW9mIEJhc2VFcnJvcikgcmV0dXJuIG1zZztcbiAgICBjb25zdCBtZXNzYWdlID0gYFske25hbWV9XSAke21zZyBpbnN0YW5jZW9mIEVycm9yID8gbXNnLm1lc3NhZ2UgOiBtc2d9YDtcbiAgICBzdXBlcihtZXNzYWdlKTtcbiAgICB0aGlzLmNvZGUgPSBjb2RlO1xuICAgIGlmIChtc2cgaW5zdGFuY2VvZiBFcnJvcikgdGhpcy5zdGFjayA9IG1zZy5zdGFjaztcbiAgfVxufVxuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBFcnJvciAoNDApIHRocm93biB3aGVuIGEgYmFkIHJlcXVlc3QgaXMgcmVjZWl2ZWRcbiAqIEBzdW1tYXJ5IFJlcHJlc2VudHMgYSBmYWlsdXJlIGluIHRoZSBpbnB1dCBkYXRhLCB0eXBpY2FsbHkgdGhyb3duIHdoZW4gYSBjbGllbnQgc2VuZHMgaW52YWxpZCBvciBpbmNvbXBsZXRlIGRhdGFcbiAqIEBwYXJhbSB7c3RyaW5nfEVycm9yfSBtc2cgLSBUaGUgZXJyb3IgbWVzc2FnZSBvciBFcnJvciBvYmplY3RcbiAqIEByZXR1cm4ge0JhZFJlcXVlc3RFcnJvcn0gQSBuZXcgQmFkUmVxdWVzdEVycm9yIGluc3RhbmNlXG4gKiBAY2xhc3MgQmFkUmVxdWVzdEVycm9yXG4gKiBAY2F0ZWdvcnkgRXJyb3JzXG4gKi9cbmV4cG9ydCBjbGFzcyBCYWRSZXF1ZXN0RXJyb3IgZXh0ZW5kcyBCYXNlRXJyb3Ige1xuICBjb25zdHJ1Y3Rvcihtc2c6IHN0cmluZyB8IEVycm9yLCBuYW1lID0gQmFkUmVxdWVzdEVycm9yLm5hbWUsIGNvZGUgPSA0MDApIHtcbiAgICBzdXBlcihuYW1lLCBtc2csIGNvZGUpO1xuICB9XG59XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIEVycm9yICg0MjIpIHRocm93biB3aGVuIHZhbGlkYXRpb24gZmFpbHNcbiAqIEBzdW1tYXJ5IFJlcHJlc2VudHMgYSBmYWlsdXJlIGluIHRoZSBNb2RlbCBkZXRhaWxzLCB0eXBpY2FsbHkgdGhyb3duIHdoZW4gZGF0YSB2YWxpZGF0aW9uIGZhaWxzXG4gKiBAcGFyYW0ge3N0cmluZ3xFcnJvcn0gbXNnIC0gVGhlIGVycm9yIG1lc3NhZ2Ugb3IgRXJyb3Igb2JqZWN0XG4gKiBAcmV0dXJuIHtWYWxpZGF0aW9uRXJyb3J9IEEgbmV3IFZhbGlkYXRpb25FcnJvciBpbnN0YW5jZVxuICogQGNsYXNzIFZhbGlkYXRpb25FcnJvclxuICogQGV4YW1wbGVcbiAqIC8vIFRocm93IGEgdmFsaWRhdGlvbiBlcnJvciB3aGVuIGRhdGEgaXMgaW52YWxpZFxuICogaWYgKCFpc1ZhbGlkKGRhdGEpKSB7XG4gKiAgIHRocm93IG5ldyBWYWxpZGF0aW9uRXJyb3IoJ0ludmFsaWQgZGF0YSBmb3JtYXQnKTtcbiAqIH1cbiAqIEBjYXRlZ29yeSBFcnJvcnNcbiAqL1xuZXhwb3J0IGNsYXNzIFZhbGlkYXRpb25FcnJvciBleHRlbmRzIEJhZFJlcXVlc3RFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1zZzogc3RyaW5nIHwgRXJyb3IpIHtcbiAgICBzdXBlcihtc2csIFZhbGlkYXRpb25FcnJvci5uYW1lLCA0MjIpO1xuICB9XG59XG4vKipcbiAqIEBkZXNjcmlwdGlvbiBFcnJvciAoNTAwKSB0aHJvd24gZm9yIGludGVybmFsIHN5c3RlbSBmYWlsdXJlc1xuICogQHN1bW1hcnkgUmVwcmVzZW50cyBhbiBpbnRlcm5hbCBmYWlsdXJlIChzaG91bGQgbWVhbiBhbiBlcnJvciBpbiBjb2RlKSB3aXRoIEhUVFAgNTAwIHN0YXR1cyBjb2RlXG4gKiBAcGFyYW0ge3N0cmluZ3xFcnJvcn0gbXNnIC0gVGhlIGVycm9yIG1lc3NhZ2Ugb3IgRXJyb3Igb2JqZWN0XG4gKiBAcmV0dXJuIHtJbnRlcm5hbEVycm9yfSBBIG5ldyBJbnRlcm5hbEVycm9yIGluc3RhbmNlXG4gKiBAY2xhc3MgSW50ZXJuYWxFcnJvclxuICogQGV4YW1wbGVcbiAqIC8vIFRocm93IGFuIGludGVybmFsIGVycm9yIHdoZW4gYW4gdW5leHBlY3RlZCBjb25kaXRpb24gb2NjdXJzXG4gKiB0cnkge1xuICogICAvLyBTb21lIG9wZXJhdGlvblxuICogfSBjYXRjaCAoZXJyb3IpIHtcbiAqICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoJ1VuZXhwZWN0ZWQgaW50ZXJuYWwgZXJyb3Igb2NjdXJyZWQnKTtcbiAqIH1cbiAqIEBjYXRlZ29yeSBFcnJvcnNcbiAqL1xuZXhwb3J0IGNsYXNzIEludGVybmFsRXJyb3IgZXh0ZW5kcyBCYXNlRXJyb3Ige1xuICBjb25zdHJ1Y3Rvcihtc2c6IHN0cmluZyB8IEVycm9yLCBuYW1lID0gSW50ZXJuYWxFcnJvci5uYW1lLCBjb2RlID0gNTAwKSB7XG4gICAgc3VwZXIobmFtZSwgbXNnLCBjb2RlKTtcbiAgfVxufVxuLyoqXG4gKiBAZGVzY3JpcHRpb24gRXJyb3IgKDUwMCkgdGhyb3duIHdoZW4gc2VyaWFsaXphdGlvbiBvciBkZXNlcmlhbGl6YXRpb24gZmFpbHNcbiAqIEBzdW1tYXJ5IFJlcHJlc2VudHMgYSBmYWlsdXJlIGluIHRoZSBNb2RlbCBkZS9zZXJpYWxpemF0aW9uLCB0eXBpY2FsbHkgd2hlbiBjb252ZXJ0aW5nIGJldHdlZW4gZGF0YSBmb3JtYXRzXG4gKiBAcGFyYW0ge3N0cmluZ3xFcnJvcn0gbXNnIC0gVGhlIGVycm9yIG1lc3NhZ2Ugb3IgRXJyb3Igb2JqZWN0XG4gKiBAcmV0dXJuIHtTZXJpYWxpemF0aW9uRXJyb3J9IEEgbmV3IFNlcmlhbGl6YXRpb25FcnJvciBpbnN0YW5jZVxuICogQGNsYXNzIFNlcmlhbGl6YXRpb25FcnJvclxuICogQGV4YW1wbGVcbiAqIC8vIFRocm93IGEgc2VyaWFsaXphdGlvbiBlcnJvciB3aGVuIEpTT04gcGFyc2luZyBmYWlsc1xuICogdHJ5IHtcbiAqICAgY29uc3QgZGF0YSA9IEpTT04ucGFyc2UoaW52YWxpZEpzb24pO1xuICogfSBjYXRjaCAoZXJyb3IpIHtcbiAqICAgdGhyb3cgbmV3IFNlcmlhbGl6YXRpb25FcnJvcignRmFpbGVkIHRvIHBhcnNlIEpTT04gZGF0YScpO1xuICogfVxuICogQGNhdGVnb3J5IEVycm9yc1xuICovXG5leHBvcnQgY2xhc3MgU2VyaWFsaXphdGlvbkVycm9yIGV4dGVuZHMgSW50ZXJuYWxFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1zZzogc3RyaW5nIHwgRXJyb3IpIHtcbiAgICBzdXBlcihtc2csIFNlcmlhbGl6YXRpb25FcnJvci5uYW1lLCA1MDApO1xuICB9XG59XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIEVycm9yIHRocm93biB3aGVuIGEgcmVxdWVzdGVkIHJlc291cmNlIGlzIG5vdCBmb3VuZFxuICogQHN1bW1hcnkgUmVwcmVzZW50cyBhIGZhaWx1cmUgaW4gZmluZGluZyBhIG1vZGVsLCByZXN1bHRpbmcgaW4gYSA0MDQgSFRUUCBzdGF0dXMgY29kZVxuICogQHBhcmFtIHtzdHJpbmd8RXJyb3J9IG1zZyAtIFRoZSBlcnJvciBtZXNzYWdlIG9yIEVycm9yIG9iamVjdFxuICogQHJldHVybiB7Tm90Rm91bmRFcnJvcn0gQSBuZXcgTm90Rm91bmRFcnJvciBpbnN0YW5jZVxuICogQGNsYXNzIE5vdEZvdW5kRXJyb3JcbiAqIEBleGFtcGxlXG4gKiAvLyBUaHJvdyBhIG5vdCBmb3VuZCBlcnJvciB3aGVuIGEgcmVjb3JkIGRvZXNuJ3QgZXhpc3RcbiAqIGNvbnN0IHVzZXIgPSBhd2FpdCByZXBvc2l0b3J5LmZpbmRCeUlkKGlkKTtcbiAqIGlmICghdXNlcikge1xuICogICB0aHJvdyBuZXcgTm90Rm91bmRFcnJvcihgVXNlciB3aXRoIElEICR7aWR9IG5vdCBmb3VuZGApO1xuICogfVxuICogQGNhdGVnb3J5IEVycm9yc1xuICovXG5leHBvcnQgY2xhc3MgTm90Rm91bmRFcnJvciBleHRlbmRzIEJhZFJlcXVlc3RFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1zZzogc3RyaW5nIHwgRXJyb3IpIHtcbiAgICBzdXBlcihtc2csIE5vdEZvdW5kRXJyb3IubmFtZSwgNDA0KTtcbiAgfVxufVxuLyoqXG4gKiBAZGVzY3JpcHRpb24gRXJyb3IgdGhyb3duIHdoZW4gYSBjb25mbGljdCBvY2N1cnMgaW4gdGhlIHN0b3JhZ2VcbiAqIEBzdW1tYXJ5IFJlcHJlc2VudHMgYSBjb25mbGljdCBpbiB0aGUgc3RvcmFnZSwgdHlwaWNhbGx5IHdoZW4gdHJ5aW5nIHRvIGNyZWF0ZSBhIGR1cGxpY2F0ZSByZXNvdXJjZVxuICogQHBhcmFtIHtzdHJpbmd8RXJyb3J9IG1zZyAtIFRoZSBlcnJvciBtZXNzYWdlIG9yIEVycm9yIG9iamVjdFxuICogQHJldHVybiB7Q29uZmxpY3RFcnJvcn0gQSBuZXcgQ29uZmxpY3RFcnJvciBpbnN0YW5jZVxuICogQGNsYXNzIENvbmZsaWN0RXJyb3JcbiAqIEBleGFtcGxlXG4gKiAvLyBUaHJvdyBhIGNvbmZsaWN0IGVycm9yIHdoZW4gdHJ5aW5nIHRvIGNyZWF0ZSBhIGR1cGxpY2F0ZSByZWNvcmRcbiAqIGNvbnN0IGV4aXN0aW5nVXNlciA9IGF3YWl0IHJlcG9zaXRvcnkuZmluZEJ5RW1haWwoZW1haWwpO1xuICogaWYgKGV4aXN0aW5nVXNlcikge1xuICogICB0aHJvdyBuZXcgQ29uZmxpY3RFcnJvcihgVXNlciB3aXRoIGVtYWlsICR7ZW1haWx9IGFscmVhZHkgZXhpc3RzYCk7XG4gKiB9XG4gKiBAY2F0ZWdvcnkgRXJyb3JzXG4gKi9cbmV4cG9ydCBjbGFzcyBDb25mbGljdEVycm9yIGV4dGVuZHMgQmFkUmVxdWVzdEVycm9yIHtcbiAgY29uc3RydWN0b3IobXNnOiBzdHJpbmcgfCBFcnJvcikge1xuICAgIHN1cGVyKG1zZywgQ29uZmxpY3RFcnJvci5uYW1lLCA0MDkpO1xuICB9XG59XG4iXX0=