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/LICENSE +202 -0
- package/NOTICE +2 -0
- package/README.md +111 -0
- package/bin/jsii-diff +2 -0
- package/bin/jsii-diff.d.ts +1 -0
- package/bin/jsii-diff.js +230 -0
- package/lib/diagnostics.d.ts +17 -0
- package/lib/diagnostics.js +52 -0
- package/lib/index.d.ts +11 -0
- package/lib/index.js +20 -0
- package/lib/stability.d.ts +3 -0
- package/lib/stability.js +52 -0
- package/lib/type-analysis.d.ts +39 -0
- package/lib/type-analysis.js +172 -0
- package/lib/type-comparison.d.ts +195 -0
- package/lib/type-comparison.js +508 -0
- package/lib/types.d.ts +46 -0
- package/lib/types.js +136 -0
- package/lib/util.d.ts +19 -0
- package/lib/util.js +117 -0
- package/lib/validations.d.ts +157 -0
- package/lib/validations.js +487 -0
- package/lib/version.d.ts +2 -0
- package/lib/version.js +7 -0
- package/package.json +57 -0
- package/test/classes.test.d.ts +1 -0
- package/test/classes.test.js +542 -0
- package/test/diagnostics.test.d.ts +1 -0
- package/test/diagnostics.test.js +110 -0
- package/test/enums.test.d.ts +1 -0
- package/test/enums.test.js +41 -0
- package/test/structs.test.d.ts +1 -0
- package/test/structs.test.js +242 -0
- package/test/type-unions.test.d.ts +1 -0
- package/test/type-unions.test.js +101 -0
- package/test/util.d.ts +4 -0
- package/test/util.js +38 -0
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;
|