jsii-diff 1.35.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.
package/lib/types.js ADDED
@@ -0,0 +1,136 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.describeInterfaceType = exports.describeType = exports.apiElementIdentifier = exports.Mismatches = void 0;
4
+ const reflect = require("jsii-reflect");
5
+ class Mismatches {
6
+ constructor(opts) {
7
+ this.mismatches = new Array();
8
+ this.defaultStability = opts.defaultStability;
9
+ }
10
+ report(options) {
11
+ var _a;
12
+ const fqn = apiElementIdentifier(options.violator);
13
+ const key = `${options.ruleKey}:${fqn}`;
14
+ this.mismatches.push({
15
+ violationKey: key,
16
+ message: `${describeApiElement(options.violator)} ${fqn}: ${options.message}`,
17
+ stability: (_a = options.violator.docs.stability) !== null && _a !== void 0 ? _a : this.defaultStability,
18
+ });
19
+ }
20
+ *messages() {
21
+ for (const mis of this.mismatches) {
22
+ yield mis.message;
23
+ }
24
+ }
25
+ get count() {
26
+ return this.mismatches.length;
27
+ }
28
+ filter(pred) {
29
+ const ret = new Mismatches({ defaultStability: this.defaultStability });
30
+ ret.mismatches.push(...this.mismatches.filter(pred));
31
+ return ret;
32
+ }
33
+ withMotivation(motivation) {
34
+ return {
35
+ report: (options) => this.report({
36
+ ...options,
37
+ message: `${options.message}: ${motivation}`,
38
+ }),
39
+ withMotivation: (innerMotivation) => this.withMotivation(`${innerMotivation}: ${motivation}`),
40
+ };
41
+ }
42
+ }
43
+ exports.Mismatches = Mismatches;
44
+ function apiElementIdentifier(apiElement) {
45
+ return dispatch(apiElement, {
46
+ method(x) {
47
+ return `${x.parentType.fqn}.${x.name}`;
48
+ },
49
+ init(x) {
50
+ return `${x.parentType.fqn}.${x.name}`;
51
+ },
52
+ property(x) {
53
+ return `${x.parentType.fqn}.${x.name}`;
54
+ },
55
+ enumMember(x) {
56
+ return `${x.enumType.fqn}.${x.name}`;
57
+ },
58
+ enumType(x) {
59
+ return `${x.fqn}`;
60
+ },
61
+ klass(x) {
62
+ return `${x.fqn}`;
63
+ },
64
+ iface(x) {
65
+ return `${x.fqn}`;
66
+ },
67
+ });
68
+ }
69
+ exports.apiElementIdentifier = apiElementIdentifier;
70
+ function describeApiElement(apiElement) {
71
+ return dispatch(apiElement, {
72
+ method() {
73
+ return 'METHOD';
74
+ },
75
+ init() {
76
+ return 'INITIALIZER';
77
+ },
78
+ property() {
79
+ return 'PROP';
80
+ },
81
+ enumMember() {
82
+ return 'ENUM VALUE';
83
+ },
84
+ enumType() {
85
+ return 'ENUM';
86
+ },
87
+ klass() {
88
+ return 'CLASS';
89
+ },
90
+ iface() {
91
+ return 'IFACE';
92
+ },
93
+ });
94
+ }
95
+ function dispatch(apiElement, fns) {
96
+ if (apiElement instanceof reflect.Method) {
97
+ return fns.method(apiElement);
98
+ }
99
+ if (apiElement instanceof reflect.Property) {
100
+ return fns.property(apiElement);
101
+ }
102
+ if (apiElement instanceof reflect.EnumMember) {
103
+ return fns.enumMember(apiElement);
104
+ }
105
+ if (apiElement instanceof reflect.ClassType) {
106
+ return fns.klass(apiElement);
107
+ }
108
+ if (apiElement instanceof reflect.InterfaceType) {
109
+ return fns.iface(apiElement);
110
+ }
111
+ if (apiElement instanceof reflect.Initializer) {
112
+ return fns.init(apiElement);
113
+ }
114
+ if (apiElement instanceof reflect.EnumType) {
115
+ return fns.enumType(apiElement);
116
+ }
117
+ throw new Error(`Unrecognized violator: ${apiElement.toString()}`);
118
+ }
119
+ function describeType(type) {
120
+ if (type.isClassType()) {
121
+ return 'CLASS';
122
+ }
123
+ if (type.isInterfaceType()) {
124
+ return 'IFACE';
125
+ }
126
+ if (type.isEnumType()) {
127
+ return 'ENUM';
128
+ }
129
+ return 'TYPE';
130
+ }
131
+ exports.describeType = describeType;
132
+ function describeInterfaceType(dataType) {
133
+ return dataType ? 'struct' : 'regular interface';
134
+ }
135
+ exports.describeInterfaceType = describeInterfaceType;
136
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["types.ts"],"names":[],"mappings":"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","sourcesContent":["import { Stability } from '@jsii/spec';\nimport * as reflect from 'jsii-reflect';\n\nexport interface ComparisonOptions {\n  /**\n   * Whether to treat API elements as experimental if unmarked.\n   *\n   * @default Treat as stable\n   */\n  defaultExperimental?: boolean;\n}\n\nexport interface ComparisonContext extends ComparisonOptions {\n  /**\n   * Where to report errors\n   */\n  mismatches: Mismatches;\n}\n\nexport interface ApiMismatch {\n  message: string;\n  violationKey: string;\n  stability: Stability;\n}\n\nexport type ApiElement = reflect.Type | reflect.TypeMember | reflect.EnumMember;\n\nexport interface ReportOptions {\n  ruleKey: string;\n  violator: ApiElement;\n  message: string;\n}\n\nexport interface IReport {\n  report(options: ReportOptions): void;\n  withMotivation(reason: string): IReport;\n}\n\nexport class Mismatches implements IReport {\n  public readonly mismatches = new Array<ApiMismatch>();\n  private readonly defaultStability: Stability;\n\n  public constructor(opts: { defaultStability: Stability }) {\n    this.defaultStability = opts.defaultStability;\n  }\n\n  public report(options: ReportOptions) {\n    const fqn = apiElementIdentifier(options.violator);\n    const key = `${options.ruleKey}:${fqn}`;\n\n    this.mismatches.push({\n      violationKey: key,\n      message: `${describeApiElement(options.violator)} ${fqn}: ${\n        options.message\n      }`,\n      stability: options.violator.docs.stability ?? this.defaultStability,\n    });\n  }\n\n  public *messages() {\n    for (const mis of this.mismatches) {\n      yield mis.message;\n    }\n  }\n\n  public get count() {\n    return this.mismatches.length;\n  }\n\n  public filter(pred: (x: ApiMismatch) => boolean): Mismatches {\n    const ret = new Mismatches({ defaultStability: this.defaultStability });\n    ret.mismatches.push(...this.mismatches.filter(pred));\n    return ret;\n  }\n\n  public withMotivation(motivation: string): IReport {\n    return {\n      report: (options) =>\n        this.report({\n          ...options,\n          message: `${options.message}: ${motivation}`,\n        }),\n      withMotivation: (innerMotivation) =>\n        this.withMotivation(`${innerMotivation}: ${motivation}`),\n    };\n  }\n}\n\nexport function apiElementIdentifier(apiElement: ApiElement) {\n  return dispatch(apiElement, {\n    method(x) {\n      return `${x.parentType.fqn}.${x.name}`;\n    },\n    init(x) {\n      return `${x.parentType.fqn}.${x.name}`;\n    },\n    property(x) {\n      return `${x.parentType.fqn}.${x.name}`;\n    },\n    enumMember(x) {\n      return `${x.enumType.fqn}.${x.name}`;\n    },\n    enumType(x) {\n      return `${x.fqn}`;\n    },\n    klass(x) {\n      return `${x.fqn}`;\n    },\n    iface(x) {\n      return `${x.fqn}`;\n    },\n  });\n}\n\nfunction describeApiElement(apiElement: ApiElement) {\n  return dispatch(apiElement, {\n    method() {\n      return 'METHOD';\n    },\n    init() {\n      return 'INITIALIZER';\n    },\n    property() {\n      return 'PROP';\n    },\n    enumMember() {\n      return 'ENUM VALUE';\n    },\n    enumType() {\n      return 'ENUM';\n    },\n    klass() {\n      return 'CLASS';\n    },\n    iface() {\n      return 'IFACE';\n    },\n  });\n}\n\nfunction dispatch<T>(\n  apiElement: ApiElement,\n  fns: {\n    method(m: reflect.Method): T;\n    init(m: reflect.Initializer): T;\n    property(m: reflect.Property): T;\n    enumMember(m: reflect.EnumMember): T;\n    enumType(m: reflect.EnumType): T;\n    klass(m: reflect.ClassType): T;\n    iface(m: reflect.InterfaceType): T;\n  },\n) {\n  if (apiElement instanceof reflect.Method) {\n    return fns.method(apiElement);\n  }\n  if (apiElement instanceof reflect.Property) {\n    return fns.property(apiElement);\n  }\n  if (apiElement instanceof reflect.EnumMember) {\n    return fns.enumMember(apiElement);\n  }\n  if (apiElement instanceof reflect.ClassType) {\n    return fns.klass(apiElement);\n  }\n  if (apiElement instanceof reflect.InterfaceType) {\n    return fns.iface(apiElement);\n  }\n  if (apiElement instanceof reflect.Initializer) {\n    return fns.init(apiElement);\n  }\n  if (apiElement instanceof reflect.EnumType) {\n    return fns.enumType(apiElement);\n  }\n\n  throw new Error(`Unrecognized violator: ${apiElement.toString()}`);\n}\n\nexport function describeType(type: reflect.Type) {\n  if (type.isClassType()) {\n    return 'CLASS';\n  }\n  if (type.isInterfaceType()) {\n    return 'IFACE';\n  }\n  if (type.isEnumType()) {\n    return 'ENUM';\n  }\n  return 'TYPE';\n}\n\nexport function describeInterfaceType(dataType: boolean) {\n  return dataType ? 'struct' : 'regular interface';\n}\n"]}
package/lib/util.d.ts ADDED
@@ -0,0 +1,19 @@
1
+ export declare function inTempDir<T>(block: () => T | Promise<T>): Promise<T>;
2
+ export declare type DownloadFailure = 'no_such_package';
3
+ export declare type NpmDownloadResult<T> = {
4
+ success: true;
5
+ result: T;
6
+ } | {
7
+ success: false;
8
+ reason: DownloadFailure;
9
+ };
10
+ export declare function showDownloadFailure(f: DownloadFailure): "NPM package does not exist" | undefined;
11
+ export declare function downloadNpmPackage<T>(pkg: string, block: (dir: string) => Promise<T>): Promise<NpmDownloadResult<T>>;
12
+ export declare function flatMap<T, U>(xs: T[], fn: (x: T) => U[]): U[];
13
+ /**
14
+ * Don't recurse infinitely by guarding a block with `do()`.
15
+ */
16
+ export declare class RecursionBreaker<A> {
17
+ private readonly elements;
18
+ do(key: A, block: () => void): void;
19
+ }
package/lib/util.js ADDED
@@ -0,0 +1,117 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.RecursionBreaker = exports.flatMap = exports.downloadNpmPackage = exports.showDownloadFailure = exports.inTempDir = void 0;
4
+ const childProcess = require("child_process");
5
+ const fs = require("fs-extra");
6
+ const log4js = require("log4js");
7
+ const os = require("os");
8
+ const path = require("path");
9
+ const util = require("util");
10
+ const LOG = log4js.getLogger('jsii-diff');
11
+ const exec = util.promisify(childProcess.exec);
12
+ async function inTempDir(block) {
13
+ const origDir = process.cwd();
14
+ const tmpDir = await fs.mkdtemp(path.join(os.tmpdir(), 'jsii'));
15
+ process.chdir(tmpDir);
16
+ try {
17
+ return await block();
18
+ }
19
+ finally {
20
+ process.chdir(origDir);
21
+ await fs.remove(tmpDir);
22
+ }
23
+ }
24
+ exports.inTempDir = inTempDir;
25
+ function showDownloadFailure(f) {
26
+ switch (f) {
27
+ case 'no_such_package':
28
+ return 'NPM package does not exist';
29
+ default:
30
+ return undefined;
31
+ }
32
+ }
33
+ exports.showDownloadFailure = showDownloadFailure;
34
+ async function downloadNpmPackage(pkg, block) {
35
+ return inTempDir(async () => {
36
+ LOG.info(`Fetching NPM package ${pkg}`);
37
+ try {
38
+ // Need to install package and dependencies in order for jsii-reflect
39
+ // to not bork when it can find the dependencies.
40
+ await exec(`npm install --silent --prefix . ${pkg}`);
41
+ }
42
+ catch (e) {
43
+ // If this fails, might be because the package doesn't exist
44
+ if (!isSubprocesFailedError(e)) {
45
+ throw e;
46
+ }
47
+ if (await npmPackageExists(pkg)) {
48
+ throw new Error(`NPM fetch failed: ${e}. Please try again.`);
49
+ }
50
+ LOG.warn(`NPM package ${pkg} does not exist.`);
51
+ return {
52
+ success: false,
53
+ reason: 'no_such_package',
54
+ };
55
+ }
56
+ const pkgDir = trimVersionString(pkg);
57
+ return {
58
+ success: true,
59
+ result: await block(path.join(process.cwd(), 'node_modules', pkgDir)),
60
+ };
61
+ });
62
+ }
63
+ exports.downloadNpmPackage = downloadNpmPackage;
64
+ function isSubprocesFailedError(e) {
65
+ return e.code !== undefined && e.cmd !== undefined;
66
+ }
67
+ async function npmPackageExists(pkg) {
68
+ try {
69
+ LOG.info(`Checking existence of ${pkg}`);
70
+ await exec(`npm show --silent ${pkg}`);
71
+ return true;
72
+ }
73
+ catch (e) {
74
+ if (!isSubprocesFailedError(e)) {
75
+ throw e;
76
+ }
77
+ return false;
78
+ }
79
+ }
80
+ /**
81
+ * Trim an optional version string from an NPM package name
82
+ */
83
+ function trimVersionString(pkg) {
84
+ // The arbitrary char before the @ prevents matching a @ at the start of the
85
+ // string.
86
+ return pkg.replace(/(.)@.*$/, '$1');
87
+ }
88
+ function flatMap(xs, fn) {
89
+ const ret = new Array();
90
+ for (const x of xs) {
91
+ ret.push(...fn(x));
92
+ }
93
+ return ret;
94
+ }
95
+ exports.flatMap = flatMap;
96
+ /**
97
+ * Don't recurse infinitely by guarding a block with `do()`.
98
+ */
99
+ class RecursionBreaker {
100
+ constructor() {
101
+ this.elements = new Set();
102
+ }
103
+ do(key, block) {
104
+ if (this.elements.has(key)) {
105
+ return;
106
+ }
107
+ this.elements.add(key);
108
+ try {
109
+ block();
110
+ }
111
+ finally {
112
+ this.elements.delete(key);
113
+ }
114
+ }
115
+ }
116
+ exports.RecursionBreaker = RecursionBreaker;
117
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"util.js","sourceRoot":"","sources":["util.ts"],"names":[],"mappings":"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","sourcesContent":["import * as childProcess from 'child_process';\nimport * as fs from 'fs-extra';\nimport * as log4js from 'log4js';\nimport * as os from 'os';\nimport * as path from 'path';\nimport * as util from 'util';\n\nconst LOG = log4js.getLogger('jsii-diff');\n\nconst exec = util.promisify(childProcess.exec);\n\nexport async function inTempDir<T>(block: () => T | Promise<T>): Promise<T> {\n  const origDir = process.cwd();\n  const tmpDir = await fs.mkdtemp(path.join(os.tmpdir(), 'jsii'));\n  process.chdir(tmpDir);\n  try {\n    return await block();\n  } finally {\n    process.chdir(origDir);\n    await fs.remove(tmpDir);\n  }\n}\n\nexport type DownloadFailure = 'no_such_package';\n\nexport type NpmDownloadResult<T> =\n  | { success: true; result: T }\n  | { success: false; reason: DownloadFailure };\n\nexport function showDownloadFailure(f: DownloadFailure) {\n  switch (f) {\n    case 'no_such_package':\n      return 'NPM package does not exist';\n    default:\n      return undefined;\n  }\n}\n\nexport async function downloadNpmPackage<T>(\n  pkg: string,\n  block: (dir: string) => Promise<T>,\n): Promise<NpmDownloadResult<T>> {\n  return inTempDir(async () => {\n    LOG.info(`Fetching NPM package ${pkg}`);\n\n    try {\n      // Need to install package and dependencies in order for jsii-reflect\n      // to not bork when it can find the dependencies.\n      await exec(`npm install --silent --prefix . ${pkg}`);\n    } catch (e) {\n      // If this fails, might be because the package doesn't exist\n      if (!isSubprocesFailedError(e)) {\n        throw e;\n      }\n      if (await npmPackageExists(pkg)) {\n        throw new Error(`NPM fetch failed: ${e}. Please try again.`);\n      }\n      LOG.warn(`NPM package ${pkg} does not exist.`);\n      return {\n        success: false,\n        reason: 'no_such_package',\n      } as NpmDownloadResult<T>;\n    }\n\n    const pkgDir = trimVersionString(pkg);\n    return {\n      success: true,\n      result: await block(path.join(process.cwd(), 'node_modules', pkgDir)),\n    } as NpmDownloadResult<T>;\n  });\n}\n\nfunction isSubprocesFailedError(e: any) {\n  return e.code !== undefined && e.cmd !== undefined;\n}\n\nasync function npmPackageExists(pkg: string): Promise<boolean> {\n  try {\n    LOG.info(`Checking existence of ${pkg}`);\n    await exec(`npm show --silent ${pkg}`);\n    return true;\n  } catch (e) {\n    if (!isSubprocesFailedError(e)) {\n      throw e;\n    }\n    return false;\n  }\n}\n\n/**\n * Trim an optional version string from an NPM package name\n */\nfunction trimVersionString(pkg: string) {\n  // The arbitrary char before the @ prevents matching a @ at the start of the\n  // string.\n  return pkg.replace(/(.)@.*$/, '$1');\n}\n\nexport function flatMap<T, U>(xs: T[], fn: (x: T) => U[]): U[] {\n  const ret = new Array<U>();\n  for (const x of xs) {\n    ret.push(...fn(x));\n  }\n  return ret;\n}\n\n/**\n * Don't recurse infinitely by guarding a block with `do()`.\n */\nexport class RecursionBreaker<A> {\n  private readonly elements = new Set<A>();\n\n  public do(key: A, block: () => void) {\n    if (this.elements.has(key)) {\n      return;\n    }\n\n    this.elements.add(key);\n    try {\n      block();\n    } finally {\n      this.elements.delete(key);\n    }\n  }\n}\n"]}
@@ -0,0 +1,157 @@
1
+ import * as reflect from 'jsii-reflect';
2
+ import { IReport } from './types';
3
+ /**
4
+ * The updated type is still nominally assignable to all original base types
5
+ *
6
+ * Make sure the following remains compilable:
7
+ *
8
+ * ```
9
+ * BASE instance = new CLASS();
10
+ * ```
11
+ *
12
+ * Where CLASS ≤: BASE.
13
+ */
14
+ export declare function validateBaseTypeAssignability<T extends reflect.ReferenceType>(original: T, updated: T, mismatches: IReport): void;
15
+ /**
16
+ * The updated type has not been newly made abstract
17
+ *
18
+ * Make sure the following remains compilable:
19
+ *
20
+ * ```
21
+ * new CLASS();
22
+ * ```
23
+ */
24
+ export declare function validateNotMadeAbstract(original: reflect.ClassType, updated: reflect.ClassType, mismatches: IReport): void;
25
+ /**
26
+ * The updated type has not had its @subclassable attribute removed
27
+ *
28
+ * This would lift a restriction we can't afford.
29
+ */
30
+ export declare function validateSubclassableNotRemoved<T extends reflect.ReferenceType>(original: T, updated: T, mismatches: IReport): void;
31
+ /**
32
+ * Check that the `static`-ness of a member hasn't changed
33
+ */
34
+ export declare function validateStaticSame<T extends reflect.Method | reflect.Property>(original: T, updated: T, mismatches: IReport): void;
35
+ /**
36
+ * Check that the `async`-ness of a method hasn't changed
37
+ */
38
+ export declare function validateAsyncSame(original: reflect.Method, updated: reflect.Method, mismatches: IReport): void;
39
+ /**
40
+ * Once variadic, can never be made non-variadic anymore (because I could always have been passing N+1 arguments)
41
+ */
42
+ export declare function validateNotMadeNonVariadic<T extends reflect.Method | reflect.Initializer>(original: T, updated: T, mismatches: IReport): void;
43
+ /**
44
+ * Check that no new abstract members were added to a subclassable type
45
+ *
46
+ * You cannot have added abstract members to the class/interface, as they are
47
+ * an added burden on potential implementors.
48
+ */
49
+ export declare function validateNoNewAbstractMembers<T extends reflect.ReferenceType>(original: T, updated: T, mismatches: IReport): void;
50
+ /**
51
+ * Validate that a method return type is the same or strengthened
52
+ *
53
+ * Make sure the following remains compilable:
54
+ *
55
+ * ```
56
+ * T value = object.method();
57
+ * ```
58
+ *
59
+ * Where RETURN_TYPE(method) ≤: T.
60
+ */
61
+ export declare function validateReturnTypeNotWeakened(original: reflect.Method, updated: reflect.Method, mismatches: IReport): void;
62
+ /**
63
+ * Validate that a method return type is the exact same
64
+ *
65
+ * Necessary for subclassable types in C#.
66
+ */
67
+ export declare function validateReturnTypeSame(original: reflect.Method, updated: reflect.Method, mismatches: IReport): void;
68
+ /**
69
+ * Validate that a property type is the same or strengthened
70
+ *
71
+ * Make sure the following remains compilable:
72
+ *
73
+ * ```
74
+ * T value = object.prop;
75
+ * ```
76
+ *
77
+ * Where RETURN_TYPE(prop) ≤: T.
78
+ */
79
+ export declare function validatePropertyTypeNotWeakened(original: reflect.Property, updated: reflect.Property, mismatches: IReport): void;
80
+ /**
81
+ * Validate that a property type is the exact same
82
+ *
83
+ * Necessary for subclassable types in C#.
84
+ */
85
+ export declare function validatePropertyTypeSame(original: reflect.Property, updated: reflect.Property, mismatches: IReport): void;
86
+ /**
87
+ * Validate that a method return type is the same or weakened
88
+ *
89
+ * Make sure the following remains compilable if U is changed:
90
+ *
91
+ * ```
92
+ * function method(arg: U) { ... }
93
+ *
94
+ * object.method(<T>value);
95
+ * ```
96
+ *
97
+ * Where T ≤: U.
98
+ */
99
+ export declare function validateParameterTypeWeakened(method: reflect.Method | reflect.Initializer, original: reflect.Parameter, updated: reflect.Parameter, mismatches: IReport): void;
100
+ /**
101
+ * Validate that a method parameter type is the exact same
102
+ *
103
+ * Necessary for subclassable types in C#.
104
+ */
105
+ export declare function validateParameterTypeSame(method: reflect.Method | reflect.Initializer, original: reflect.Parameter, updated: reflect.Parameter, mismatches: IReport): void;
106
+ /**
107
+ * Validate that each param in the old callable is still available in the new callable, and apply custom validation to the pairs
108
+ *
109
+ * Make sure the following remains compilable:
110
+ *
111
+ * ```
112
+ * object.method(a1, a2, ..., aN);
113
+ * ```
114
+ *
115
+ * (All types still assignable)
116
+ */
117
+ export declare function validateExistingParams<T extends reflect.Initializer | reflect.Method>(original: T, updated: T, mismatches: IReport, validateParam: (oldParam: reflect.Parameter, newParam: reflect.Parameter) => void): void;
118
+ /**
119
+ * Validate that no new required params got added to the end of the method
120
+ *
121
+ * Make sure the following remains compilable:
122
+ *
123
+ * ```
124
+ * object.method(a1, a2, ..., aN);
125
+ * ```
126
+ *
127
+ * (Not too few arguments)
128
+ */
129
+ export declare function validateNoNewRequiredParams<T extends reflect.Initializer | reflect.Method>(original: T, updated: T, mismatches: IReport): void;
130
+ export declare function validateMethodCompatible<T extends reflect.Method | reflect.Initializer>(original: T, updated: T, mismatches: IReport): void;
131
+ /**
132
+ * Check if a class/interface has been marked as @subclassable
133
+ */
134
+ export declare function subclassableType(x: reflect.Documentable): boolean;
135
+ /**
136
+ * Validate that a previously mutable property is not made immutable
137
+ *
138
+ * Make sure the following remains compilable:
139
+ *
140
+ * ```
141
+ * object.prop = value;
142
+ * ```
143
+ */
144
+ export declare function validateNotMadeImmutable(original: reflect.Property, updated: reflect.Property, mismatches: IReport): void;
145
+ export declare function memberPairs<T extends reflect.TypeMember, U extends reflect.ReferenceType>(origClass: U, xs: T[], updatedClass: U, mismatches: IReport): IterableIterator<[T, reflect.TypeMember]>;
146
+ /**
147
+ * Validate that each enum member in the old enum enum, and apply custom validation to the enums
148
+ *
149
+ * Make sure the following remains compilable:
150
+ *
151
+ * ```
152
+ * T x = ENUM.member;
153
+ * ```
154
+ *
155
+ * (For every member of enum)
156
+ */
157
+ export declare function validateExistingMembers(original: reflect.EnumType, updated: reflect.EnumType, mismatches: IReport, validateMember: (oldParam: reflect.EnumMember, newParam: reflect.EnumMember) => void): void;