gitlab-ci-local
Version:
Tired of pushing to test your .gitlab-ci.yml?
114 lines • 17.2 kB
JavaScript
// adapted from https://github.com/apideck-libraries/better-ajv-errors (MIT)
// https://github.com/apideck-libraries/better-ajv-errors/tree/026206038919c1fb73b4e8ef258a2e4a01813c4a
import pointer from "jsonpointer";
const QUOTES_REGEX = /"/g;
const NOT_REGEX = /NOT/g;
const SLASH_REGEX = /\//g;
const AJV_ERROR_KEYWORD_WEIGHT_MAP = { enum: 1, type: 0 };
const pointerToDotNotation = (pointer) => {
return pointer.replace(SLASH_REGEX, ".");
};
const cleanAjvMessage = (message) => {
return message.replace(QUOTES_REGEX, "'").replace(NOT_REGEX, "not");
};
const getLastSegment = (path) => {
const segments = path.split("/");
return segments.pop();
};
const safeJsonPointer = ({ object, pnter, fallback }) => {
try {
return pointer.get(object, pnter) ?? fallback;
}
catch (err) {
return fallback;
}
};
const filterSingleErrorPerProperty = (errors) => {
const errorsPerProperty = {};
errors.forEach(error => {
const prop = error.instancePath + (error.params?.additionalProperty ?? error.params?.missingProperty ?? "");
const existingError = errorsPerProperty[prop];
if (!existingError) {
errorsPerProperty[prop] = error;
return errorsPerProperty;
}
const weight = AJV_ERROR_KEYWORD_WEIGHT_MAP[error.keyword] ?? 0;
const existingWeight = AJV_ERROR_KEYWORD_WEIGHT_MAP[existingError.keyword] ?? 0;
if (weight > existingWeight) {
errorsPerProperty[prop] = error;
}
});
return Object.values(errorsPerProperty);
};
export const betterAjvErrors = ({ errors, data, basePath = "", }) => {
if (!Array.isArray(errors) || !errors?.length) {
return [];
}
const definedErrors = filterSingleErrorPerProperty(errors);
return definedErrors.map((error) => {
const path = basePath ? pointerToDotNotation(basePath + error.instancePath) : pointerToDotNotation(error.instancePath).substring(1);
const prop = getLastSegment(error.instancePath);
const schemaPath = error.schemaPath;
const propertyMessage = prop ? `property '${prop}'` : path;
const defaultMessage = `${propertyMessage} ${(cleanAjvMessage(error.message))}`;
let validationError;
switch (error.keyword) {
case "additionalProperties": {
const additionalProp = error.params.additionalProperty;
validationError = {
message: `'${additionalProp}' property is not expected to be here`,
path,
schemaPath,
};
break;
}
case "enum": {
const allowedValues = error.params.allowedValues.map((value) => value.toString());
const prop = getLastSegment(error.instancePath);
const value = safeJsonPointer({ object: data, pnter: error.instancePath, fallback: "" });
validationError = {
message: `'${prop}' property must be one of [${allowedValues.join(", ")}] (found ${value})`,
path,
schemaPath,
};
break;
}
case "type": {
const prop = getLastSegment(error.instancePath);
const type = error.params.type;
validationError = {
message: `'${prop}' property type must be ${type}`,
path,
schemaPath,
};
break;
}
case "required": {
validationError = {
message: `${path} must have required property '${error.params.missingProperty}'`,
path,
schemaPath,
};
break;
}
case "const": {
return {
message: `'${prop}' property must be equal to the allowed value`,
path,
schemaPath,
};
}
default:
validationError = { message: defaultMessage, path, schemaPath };
}
// Remove empty properties
const errorEntries = Object.entries(validationError);
for (const [key, value] of errorEntries) {
if (value === null || value === undefined || value === "") {
delete validationError[key];
}
}
return validationError;
});
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"schema-error.js","sourceRoot":"","sources":["schema-error.ts"],"names":[],"mappings":"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","sourcesContent":["// adapted from https://github.com/apideck-libraries/better-ajv-errors (MIT)\n// https://github.com/apideck-libraries/better-ajv-errors/tree/026206038919c1fb73b4e8ef258a2e4a01813c4a\n\nimport {DefinedError, ErrorObject} from \"ajv\";\nimport pointer from \"jsonpointer\";\n\nexport interface ValidationError {\n    message: string;\n    path: string;\n    schemaPath: string;\n}\n\nconst QUOTES_REGEX = /\"/g;\nconst NOT_REGEX = /NOT/g;\nconst SLASH_REGEX = /\\//g;\n\nconst AJV_ERROR_KEYWORD_WEIGHT_MAP: Partial<Record<DefinedError[\"keyword\"], number>> = {enum: 1, type: 0};\n\nconst pointerToDotNotation = (pointer: string): string => {\n    return pointer.replace(SLASH_REGEX, \".\");\n};\n\nconst cleanAjvMessage = (message: string): string => {\n    return message.replace(QUOTES_REGEX, \"'\").replace(NOT_REGEX, \"not\");\n};\n\nconst getLastSegment = (path: string): string => {\n    const segments = path.split(\"/\");\n    return segments.pop() as string;\n};\n\nconst safeJsonPointer = <T>({object, pnter, fallback}: {object: any; pnter: string; fallback: T}): T => {\n    try {\n        return pointer.get(object, pnter) ?? fallback;\n    } catch (err) {\n        return fallback;\n    }\n};\n\nconst filterSingleErrorPerProperty = (errors: DefinedError[]): DefinedError[] => {\n    const errorsPerProperty: Record<string, DefinedError> = {};\n    errors.forEach(error => {\n        const prop =\n            error.instancePath + ((error.params as any)?.additionalProperty ?? (error.params as any)?.missingProperty ?? \"\");\n        const existingError = errorsPerProperty[prop];\n        if (!existingError) {\n            errorsPerProperty[prop] = error;\n            return errorsPerProperty;\n        }\n\n        const weight = AJV_ERROR_KEYWORD_WEIGHT_MAP[error.keyword] ?? 0;\n        const existingWeight = AJV_ERROR_KEYWORD_WEIGHT_MAP[existingError.keyword] ?? 0;\n\n        if (weight > existingWeight) {\n            errorsPerProperty[prop] = error;\n        }\n    });\n\n    return Object.values(errorsPerProperty);\n};\n\ninterface BetterAjvErrorsOptions {\n    errors: ErrorObject[] | null | undefined;\n    data: any;\n    basePath?: string;\n}\n\nexport const betterAjvErrors = ({\n    errors,\n    data,\n    basePath = \"\",\n}: BetterAjvErrorsOptions): ValidationError[] => {\n    if (!Array.isArray(errors) || !errors?.length) {\n        return [];\n    }\n\n    const definedErrors = filterSingleErrorPerProperty(errors as DefinedError[]);\n\n    return definedErrors.map((error) => {\n        const path = basePath ? pointerToDotNotation(basePath + error.instancePath) : pointerToDotNotation(error.instancePath).substring(1);\n        const prop = getLastSegment(error.instancePath);\n        const schemaPath = error.schemaPath;\n        const propertyMessage = prop ? `property '${prop}'` : path;\n        const defaultMessage = `${propertyMessage} ${(cleanAjvMessage(error.message as string))}`;\n\n        let validationError: ValidationError;\n\n        switch (error.keyword) {\n            case \"additionalProperties\": {\n                const additionalProp = error.params.additionalProperty;\n                validationError = {\n                    message: `'${additionalProp}' property is not expected to be here`,\n                    path,\n                    schemaPath,\n                };\n                break;\n            }\n            case \"enum\": {\n                const allowedValues = error.params.allowedValues.map((value) => value.toString());\n                const prop = getLastSegment(error.instancePath);\n                const value = safeJsonPointer({object: data, pnter: error.instancePath, fallback: \"\"});\n                validationError = {\n                    message: `'${prop}' property must be one of [${allowedValues.join(\", \")}] (found ${value})`,\n                    path,\n                    schemaPath,\n                };\n                break;\n            }\n            case \"type\": {\n                const prop = getLastSegment(error.instancePath);\n                const type = error.params.type;\n                validationError = {\n                    message: `'${prop}' property type must be ${type}`,\n                    path,\n                    schemaPath,\n                };\n                break;\n            }\n            case \"required\": {\n                validationError = {\n                    message: `${path} must have required property '${error.params.missingProperty}'`,\n                    path,\n                    schemaPath,\n                };\n                break;\n            }\n            case \"const\": {\n                return {\n                    message: `'${prop}' property must be equal to the allowed value`,\n                    path,\n                    schemaPath,\n                };\n            }\n\n            default:\n                validationError = {message: defaultMessage, path, schemaPath};\n        }\n\n        // Remove empty properties\n        const errorEntries = Object.entries(validationError);\n        for (const [key, value] of errorEntries as [keyof ValidationError, unknown][]) {\n            if (value === null || value === undefined || value === \"\") {\n                delete validationError[key];\n            }\n        }\n\n        return validationError;\n    });\n};\n"]}