cdk8s-cli
Version:
This is the command line tool for Cloud Development Kit (CDK) for Kubernetes (cdk8s).
98 lines • 13.9 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
exports.SafeReviver = void 0;
/**
* JSON/YAML reviver that:
*
* - Throws when an illegal key is detected.
* - Replaces illegal values with a special marker.
*/
class SafeReviver {
constructor(props) {
var _a, _b;
this.allowlistedKeys = (_a = props === null || props === void 0 ? void 0 : props.allowlistedKeys) !== null && _a !== void 0 ? _a : [];
this.sanitizers = (_b = props === null || props === void 0 ? void 0 : props.sanitizers) !== null && _b !== void 0 ? _b : [];
}
sanitizeValue(path, value) {
for (const sanitizer of this.sanitizers) {
const { applied, sanitized } = sanitizer(path, value);
if (applied) {
return sanitized;
}
}
return value;
}
/**
* Sanitizes a JSON object in-place.
*/
sanitize(obj) {
if (obj == null)
return;
this._sanitizeObj([], obj);
}
_sanitizeObj(path, partialObj) {
for (const [key, value] of Object.entries(partialObj)) {
if (typeof (key) !== 'string') {
throw new Error(`Expected key (${key}) to be of type 'string', but got '${typeof (key)}'`);
}
if (!this.allowlistedKeys.includes(key) && !SafeReviver.LEGAL_CHARS.test(key)) {
// keys cannot be stripped so we have to throw - thats ok, we don't want to parse such docs at all
throw new Error(`Key '${key}' contains non standard characters (Must match regex '${SafeReviver.LEGAL_CHARS}')`);
}
const childPath = path.concat([key]);
if (typeof (value) === 'string') {
partialObj[key] = this.sanitizeValue(childPath, value);
}
else if (typeof (value) === 'object' && value !== null) {
this._sanitizeObj(childPath, value); // recursive call
}
}
}
}
exports.SafeReviver = SafeReviver;
// . | used in resource fqn which servers as a key (e.g io.k8s.apimachinery.pkg.apis.meta.v1.APIGroup)
// / | used in $ref to point to a definition (e.g #/definitions/io.k8s.apimachinery.pkg.apis.meta.v1.GroupVersionForDiscovery)
// - | used in annotation keys (e.g x-kubernetes-group-version-kind)
// # | used in $ref to point to a definition (e.g #/definitions/io.k8s.apimachinery.pkg.apis.meta.v1.GroupVersionForDiscovery)
// , | used in values that represent a list (e.g merge,retainKeys)
// + | used in values representing MIME types (e.g application/json-patch+json)
// : | used in e.g. Prometheus events (e.g run:completed)
// * | used in e.g. AWS policies (e.g s3:ObjectCreated:*)
SafeReviver.LEGAL_CHARS = /^(\w|\.|\/|-|#|,)*$/;
SafeReviver.LEGAL_CHARS_IN_ENUM = /^( |\w|\.|\/|-|#|,|\+|:|\*|!|=|~)*$/;
// the string we use as the stripped value
SafeReviver.STRIPPED_VALUE = '__stripped_by_cdk8s__';
// remove characters from descriptions that might terminate the comment
SafeReviver.DESCRIPTION_SANITIZER = (path, value) => {
if (path.length > 0 && path[path.length - 1] === 'description') {
return { applied: true, sanitized: value.replace(/\*\//g, '_/') };
}
else {
return { applied: false };
}
};
// strip most illegal values
// the reason we don't throw is because sometimes these type of values exist in
// the original text for good reason, like for example a `jsonPath` key in a CRD manifest, and we don't
// want to fail the entire thing.
SafeReviver.LEGAL_CHAR_SANITIZER = (path, value) => {
// case 1: we are in an array of enums
if (path.length > 2 && path[path.length - 2] === 'enum' && /^\d+$/.test(path[path.length - 1])) {
if (!SafeReviver.LEGAL_CHARS_IN_ENUM.test(value)) {
return { applied: true, sanitized: SafeReviver.STRIPPED_VALUE };
}
else {
return { applied: false };
}
// case 2: default
}
else {
if (!SafeReviver.LEGAL_CHARS.test(value)) {
return { applied: true, sanitized: SafeReviver.STRIPPED_VALUE };
}
else {
return { applied: false };
}
}
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"reviver.js","sourceRoot":"","sources":["../src/reviver.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * A function that takes a value and the path to it within the parent object,\n * and either transforms it or leaves it alone.\n */\nexport type Sanitizer = (path: string[], value: string) => { applied: boolean; sanitized?: string };\n\n/**\n * Properties for 'SafeReviver'.\n */\nexport interface SafeReviverProps {\n  readonly allowlistedKeys?: string[];\n  readonly sanitizers?: Array<Sanitizer>;\n}\n\n/**\n * JSON/YAML reviver that:\n *\n * - Throws when an illegal key is detected.\n * - Replaces illegal values with a special marker.\n */\nexport class SafeReviver {\n\n  // . | used in resource fqn which servers as a key (e.g io.k8s.apimachinery.pkg.apis.meta.v1.APIGroup)\n  // / | used in $ref to point to a definition (e.g #/definitions/io.k8s.apimachinery.pkg.apis.meta.v1.GroupVersionForDiscovery)\n  // - | used in annotation keys (e.g x-kubernetes-group-version-kind)\n  // # | used in $ref to point to a definition (e.g #/definitions/io.k8s.apimachinery.pkg.apis.meta.v1.GroupVersionForDiscovery)\n  // , | used in values that represent a list (e.g merge,retainKeys)\n  // + | used in values representing MIME types (e.g application/json-patch+json)\n  // : | used in e.g. Prometheus events (e.g run:completed)\n  // * | used in e.g. AWS policies (e.g s3:ObjectCreated:*)\n  public static readonly LEGAL_CHARS = /^(\\w|\\.|\\/|-|#|,)*$/;\n  public static readonly LEGAL_CHARS_IN_ENUM = /^( |\\w|\\.|\\/|-|#|,|\\+|:|\\*|!|=|~)*$/;\n\n  // the string we use as the stripped value\n  public static readonly STRIPPED_VALUE = '__stripped_by_cdk8s__';\n\n  // remove characters from descriptions that might terminate the comment\n  public static readonly DESCRIPTION_SANITIZER: Sanitizer = (path: string[], value: string) => {\n    if (path.length > 0 && path[path.length - 1] === 'description') {\n      return { applied: true, sanitized: value.replace(/\\*\\//g, '_/') };\n    } else {\n      return { applied: false };\n    }\n  };\n\n  // strip most illegal values\n  // the reason we don't throw is because sometimes these type of values exist in\n  // the original text for good reason, like for example a `jsonPath` key in a CRD manifest, and we don't\n  // want to fail the entire thing.\n  public static readonly LEGAL_CHAR_SANITIZER: Sanitizer = (path: string[], value: string) => {\n    // case 1: we are in an array of enums\n    if (path.length > 2 && path[path.length - 2] === 'enum' && /^\\d+$/.test(path[path.length - 1])) {\n\n      if (!SafeReviver.LEGAL_CHARS_IN_ENUM.test(value)) {\n        return { applied: true, sanitized: SafeReviver.STRIPPED_VALUE };\n      } else {\n        return { applied: false };\n      }\n\n    // case 2: default\n    } else {\n\n      if (!SafeReviver.LEGAL_CHARS.test(value)) {\n        return { applied: true, sanitized: SafeReviver.STRIPPED_VALUE };\n      } else {\n        return { applied: false };\n      }\n\n    }\n  };\n\n  private readonly allowlistedKeys: string[];\n  private readonly sanitizers: Array<Sanitizer>;\n\n  constructor(props?: SafeReviverProps) {\n    this.allowlistedKeys = props?.allowlistedKeys ?? [];\n    this.sanitizers = props?.sanitizers ?? [];\n  }\n\n  public sanitizeValue(path: string[], value: string): string | undefined {\n    for (const sanitizer of this.sanitizers) {\n      const { applied, sanitized } = sanitizer(path, value);\n      if (applied) {\n        return sanitized;\n      }\n    }\n\n    return value;\n  }\n\n  /**\n   * Sanitizes a JSON object in-place.\n   */\n  public sanitize(obj: any) {\n    if (obj == null) return;\n    this._sanitizeObj([], obj);\n  }\n\n  private _sanitizeObj(path: string[], partialObj: any) {\n    for (const [key, value] of Object.entries(partialObj)) {\n      if (typeof(key) !== 'string') {\n        throw new Error(`Expected key (${key}) to be of type 'string', but got '${typeof(key)}'`);\n      }\n\n      if (!this.allowlistedKeys.includes(key) && !SafeReviver.LEGAL_CHARS.test(key)) {\n        // keys cannot be stripped so we have to throw - thats ok, we don't want to parse such docs at all\n        throw new Error(`Key '${key}' contains non standard characters (Must match regex '${SafeReviver.LEGAL_CHARS}')`);\n      }\n\n      const childPath = path.concat([key]);\n\n      if (typeof(value) === 'string') {\n        partialObj[key] = this.sanitizeValue(childPath, value);\n      } else if (typeof(value) === 'object' && value !== null) {\n        this._sanitizeObj(childPath, value); // recursive call\n      }\n    }\n  }\n}\n"]}
;