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
|
@@ -0,0 +1,542 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const util_1 = require("./util");
|
|
4
|
+
// ----------------------------------------------------------------------
|
|
5
|
+
test('okay to add a new class', () => util_1.expectNoError(`
|
|
6
|
+
export class Foo1 { }
|
|
7
|
+
`, `
|
|
8
|
+
export class Foo1 { }
|
|
9
|
+
export class Foo2 { }
|
|
10
|
+
`));
|
|
11
|
+
// ----------------------------------------------------------------------
|
|
12
|
+
test('okay to add a new function to a class', () => util_1.expectNoError(`
|
|
13
|
+
export class Foo { }
|
|
14
|
+
`, `
|
|
15
|
+
export class Foo {
|
|
16
|
+
public bar(): void { }
|
|
17
|
+
}
|
|
18
|
+
`));
|
|
19
|
+
// ----------------------------------------------------------------------
|
|
20
|
+
test('not okay to add a required argument to a method', () => util_1.expectError(/newly required argument/, `
|
|
21
|
+
export class Foo {
|
|
22
|
+
public bar(arg1: string): void {
|
|
23
|
+
Array.isArray(arg1);
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
`, `
|
|
27
|
+
export class Foo {
|
|
28
|
+
public bar(arg1: string, arg2: string): void {
|
|
29
|
+
Array.isArray(arg1);
|
|
30
|
+
Array.isArray(arg2);
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
`));
|
|
34
|
+
// ----------------------------------------------------------------------
|
|
35
|
+
test('okay to make a required argument optional', () => util_1.expectNoError(`
|
|
36
|
+
export class Foo {
|
|
37
|
+
public bar(arg1: string, arg2: string): void {
|
|
38
|
+
Array.isArray(arg1);
|
|
39
|
+
Array.isArray(arg2);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
`, `
|
|
43
|
+
export class Foo {
|
|
44
|
+
public bar(arg1: string, arg2?: string): void {
|
|
45
|
+
Array.isArray(arg1);
|
|
46
|
+
Array.isArray(arg2);
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
`));
|
|
50
|
+
// ----------------------------------------------------------------------
|
|
51
|
+
test('okay to turn required arguments into varargs', () => util_1.expectNoError(`
|
|
52
|
+
export class Foo {
|
|
53
|
+
public bar(arg1: string, arg2: number, arg3: number): void {
|
|
54
|
+
Array.isArray(arg1);
|
|
55
|
+
Array.isArray(arg2);
|
|
56
|
+
Array.isArray(arg3);
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
`, `
|
|
60
|
+
export class Foo {
|
|
61
|
+
public bar(arg1: string, ...args: number[]): void {
|
|
62
|
+
Array.isArray(arg1);
|
|
63
|
+
Array.isArray(args);
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
`));
|
|
67
|
+
// ----------------------------------------------------------------------
|
|
68
|
+
test('not allowed to change argument type to a different scalar', () => util_1.expectError(/method.*foo.*argument arg1, takes number \(formerly string\): string is not assignable to number/i, `
|
|
69
|
+
export class Foo {
|
|
70
|
+
public bar(arg1: string): void {
|
|
71
|
+
Array.isArray(arg1);
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
`, `
|
|
75
|
+
export class Foo {
|
|
76
|
+
public bar(arg1: number): void {
|
|
77
|
+
Array.isArray(arg1);
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
`));
|
|
81
|
+
// ----------------------------------------------------------------------
|
|
82
|
+
test('cannot add any abstract members to a subclassable class', () => util_1.expectError(/adds requirement for subclasses to implement 'piet'./, `
|
|
83
|
+
/**
|
|
84
|
+
* @subclassable
|
|
85
|
+
*/
|
|
86
|
+
export abstract class Henk {
|
|
87
|
+
abstract readonly name: string;
|
|
88
|
+
}
|
|
89
|
+
`, `
|
|
90
|
+
/**
|
|
91
|
+
* @subclassable
|
|
92
|
+
*/
|
|
93
|
+
export abstract class Henk {
|
|
94
|
+
abstract readonly name: string;
|
|
95
|
+
abstract readonly piet: string;
|
|
96
|
+
}
|
|
97
|
+
`));
|
|
98
|
+
// ----------------------------------------------------------------------
|
|
99
|
+
test('cannot add any members to a subclassable interface, not even optional ones', () => util_1.expectError(/adds requirement for subclasses to implement 'piet'./, `
|
|
100
|
+
/**
|
|
101
|
+
* @subclassable
|
|
102
|
+
*/
|
|
103
|
+
export interface IHenk {
|
|
104
|
+
name: string;
|
|
105
|
+
}
|
|
106
|
+
`, `
|
|
107
|
+
/**
|
|
108
|
+
* @subclassable
|
|
109
|
+
*/
|
|
110
|
+
export interface IHenk {
|
|
111
|
+
name: string;
|
|
112
|
+
piet?: string;
|
|
113
|
+
}
|
|
114
|
+
`));
|
|
115
|
+
// ----------------------------------------------------------------------
|
|
116
|
+
test('cannot make a member less visible', () => util_1.expectError(/changed from 'public' to 'protected'/, `
|
|
117
|
+
export class Henk {
|
|
118
|
+
public name: string = 'henk';
|
|
119
|
+
}
|
|
120
|
+
`, `
|
|
121
|
+
export class Henk {
|
|
122
|
+
protected name: string = 'henk';
|
|
123
|
+
}
|
|
124
|
+
`));
|
|
125
|
+
// ----------------------------------------------------------------------
|
|
126
|
+
describe('implement base types need to be present in updated type system', () => {
|
|
127
|
+
test('for interfaces', () => util_1.expectError(/not assignable to all base types anymore/, `
|
|
128
|
+
export interface IPapa {
|
|
129
|
+
readonly pipe: string;
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
export interface IBebe extends IPapa {
|
|
133
|
+
readonly pacifier: string;
|
|
134
|
+
}
|
|
135
|
+
`, `
|
|
136
|
+
export interface IPapa {
|
|
137
|
+
readonly pipe: string;
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
export interface IBebe {
|
|
141
|
+
readonly pacifier: string;
|
|
142
|
+
}
|
|
143
|
+
`));
|
|
144
|
+
test('for structs', () => util_1.expectError(/not assignable to all base types anymore/, `
|
|
145
|
+
export interface Papa {
|
|
146
|
+
readonly pipe: string;
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
export interface Bebe extends Papa {
|
|
150
|
+
readonly pacifier: string;
|
|
151
|
+
}
|
|
152
|
+
`, `
|
|
153
|
+
export interface Papa {
|
|
154
|
+
readonly pipe: string;
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
export interface Bebe {
|
|
158
|
+
readonly pacifier: string;
|
|
159
|
+
}
|
|
160
|
+
`));
|
|
161
|
+
test('for classes', () => util_1.expectError(/not assignable to all base types anymore/, `
|
|
162
|
+
export interface IPapa {
|
|
163
|
+
readonly pipe: string;
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
export class Bebe implements IPapa {
|
|
167
|
+
readonly pipe: string = 'pff';
|
|
168
|
+
readonly pacifier: string = 'mmm';
|
|
169
|
+
}
|
|
170
|
+
`, `
|
|
171
|
+
export interface IPapa {
|
|
172
|
+
readonly pipe: string;
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
export class Bebe {
|
|
176
|
+
readonly pipe: string = 'pff';
|
|
177
|
+
readonly pacifier: string = 'mmm';
|
|
178
|
+
}
|
|
179
|
+
`));
|
|
180
|
+
test('for base classes', () => util_1.expectError(/not assignable to all base types anymore/, `
|
|
181
|
+
export class Papa {
|
|
182
|
+
readonly pipe: string = 'pff';
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
export class Bebe extends Papa {
|
|
186
|
+
readonly pacifier: string = 'mmm';
|
|
187
|
+
}
|
|
188
|
+
`, `
|
|
189
|
+
export class Papa {
|
|
190
|
+
readonly pipe: string = 'pff';
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
export class Bebe {
|
|
194
|
+
readonly pacifier: string = 'mmm';
|
|
195
|
+
}
|
|
196
|
+
`));
|
|
197
|
+
test('new levels of inheritance are allowed', () => util_1.expectNoError(`
|
|
198
|
+
export class Papa {
|
|
199
|
+
readonly pipe: string = 'pff';
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
export class Bebe extends Papa {
|
|
203
|
+
readonly pacifier: string = 'mmm';
|
|
204
|
+
}
|
|
205
|
+
`, `
|
|
206
|
+
export class Papa {
|
|
207
|
+
readonly pipe: string = 'pff';
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
export class Inbetween extends Papa {
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
export class Bebe extends Inbetween {
|
|
214
|
+
readonly pacifier: string = 'mmm';
|
|
215
|
+
}
|
|
216
|
+
`));
|
|
217
|
+
test('change direct implementation to inherited implementation via interface', () => util_1.expectNoError(`
|
|
218
|
+
export interface IPapa {
|
|
219
|
+
readonly pipe: string;
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
export class Bebe implements IPapa {
|
|
223
|
+
readonly pipe: string = 'pff';
|
|
224
|
+
readonly pacifier: string = 'mmm';
|
|
225
|
+
}
|
|
226
|
+
`, `
|
|
227
|
+
export interface IPapa {
|
|
228
|
+
readonly pipe: string;
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
export interface IInbetween extends IPapa {
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
export class Bebe implements IInbetween {
|
|
235
|
+
readonly pipe: string = 'pff';
|
|
236
|
+
readonly pacifier: string = 'mmm';
|
|
237
|
+
}
|
|
238
|
+
`));
|
|
239
|
+
test('change direct implementation to inherited implementation via base class', () => util_1.expectNoError(`
|
|
240
|
+
export interface IPapa {
|
|
241
|
+
readonly pipe: string;
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
export class Bebe implements IPapa {
|
|
245
|
+
readonly pipe: string = 'pff';
|
|
246
|
+
readonly pacifier: string = 'mmm';
|
|
247
|
+
}
|
|
248
|
+
`, `
|
|
249
|
+
export interface IPapa {
|
|
250
|
+
readonly pipe: string;
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
export class Inbetween implements IPapa {
|
|
254
|
+
readonly pipe: string = 'pff';
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
export class Bebe extends Inbetween {
|
|
258
|
+
readonly pacifier: string = 'mmm';
|
|
259
|
+
}
|
|
260
|
+
`));
|
|
261
|
+
});
|
|
262
|
+
// ----------------------------------------------------------------------
|
|
263
|
+
test.each([
|
|
264
|
+
{
|
|
265
|
+
oldDecl: 'name: string',
|
|
266
|
+
newDecl: 'name?: string',
|
|
267
|
+
error: /type Optional<string> \(formerly string\): output type is now optional/,
|
|
268
|
+
},
|
|
269
|
+
{
|
|
270
|
+
oldDecl: 'name?: string',
|
|
271
|
+
newDecl: 'name: string',
|
|
272
|
+
error: undefined,
|
|
273
|
+
},
|
|
274
|
+
{
|
|
275
|
+
oldDecl: 'name: string',
|
|
276
|
+
newDecl: 'name: string | number',
|
|
277
|
+
error: /string \| number is not assignable to string/,
|
|
278
|
+
},
|
|
279
|
+
{
|
|
280
|
+
oldDecl: 'name: string | number',
|
|
281
|
+
newDecl: 'name: string',
|
|
282
|
+
error: undefined,
|
|
283
|
+
},
|
|
284
|
+
])('change class property ', ({ oldDecl, newDecl, error }) => util_1.expectError(error, `
|
|
285
|
+
export class Henk {
|
|
286
|
+
public readonly ${oldDecl} = 'henk';
|
|
287
|
+
}
|
|
288
|
+
`, `
|
|
289
|
+
export class Henk {
|
|
290
|
+
public readonly ${newDecl} = 'henk';
|
|
291
|
+
}
|
|
292
|
+
`));
|
|
293
|
+
// ----------------------------------------------------------------------
|
|
294
|
+
test.each([
|
|
295
|
+
{
|
|
296
|
+
oldDecl: 'name: string',
|
|
297
|
+
newDecl: 'name?: string',
|
|
298
|
+
error: /changed to Optional<string> \(formerly string\)/,
|
|
299
|
+
},
|
|
300
|
+
{
|
|
301
|
+
oldDecl: 'name?: string',
|
|
302
|
+
newDecl: 'name: string',
|
|
303
|
+
error: /changed to string \(formerly Optional<string>\)/,
|
|
304
|
+
},
|
|
305
|
+
{
|
|
306
|
+
oldDecl: 'name: string',
|
|
307
|
+
newDecl: 'name: string | number',
|
|
308
|
+
error: /changed to string \| number \(formerly string\)/,
|
|
309
|
+
},
|
|
310
|
+
{
|
|
311
|
+
oldDecl: 'name: string | number',
|
|
312
|
+
newDecl: 'name: string',
|
|
313
|
+
error: /changed to string \(formerly string \| number\)/,
|
|
314
|
+
},
|
|
315
|
+
])('cannot change a mutable class property type: %p to %p', ({ oldDecl, newDecl, error }) => util_1.expectError(error, `
|
|
316
|
+
export class Henk {
|
|
317
|
+
public ${oldDecl} = 'henk';
|
|
318
|
+
}
|
|
319
|
+
`, `
|
|
320
|
+
export class Henk {
|
|
321
|
+
public ${newDecl} = 'henk';
|
|
322
|
+
}
|
|
323
|
+
`));
|
|
324
|
+
// ----------------------------------------------------------------------
|
|
325
|
+
test('consider inherited constructor', () => util_1.expectError(/number is not assignable to string/, `
|
|
326
|
+
export class Super {
|
|
327
|
+
constructor(param: number) {
|
|
328
|
+
Array.isArray(param);
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
export class Sub extends Super {
|
|
332
|
+
}
|
|
333
|
+
`, `
|
|
334
|
+
export class Super {
|
|
335
|
+
constructor(param: number) {
|
|
336
|
+
Array.isArray(param);
|
|
337
|
+
}
|
|
338
|
+
}
|
|
339
|
+
export class Sub extends Super {
|
|
340
|
+
constructor(param: string) {
|
|
341
|
+
super(5);
|
|
342
|
+
Array.isArray(param);
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
`));
|
|
346
|
+
// ----------------------------------------------------------------------
|
|
347
|
+
test('consider inherited constructor, the other way', () => util_1.expectError(/newly required argument/, `
|
|
348
|
+
export class Super {
|
|
349
|
+
constructor(param: number) {
|
|
350
|
+
Array.isArray(param);
|
|
351
|
+
}
|
|
352
|
+
}
|
|
353
|
+
export class Sub extends Super {
|
|
354
|
+
constructor() {
|
|
355
|
+
super(5);
|
|
356
|
+
}
|
|
357
|
+
}
|
|
358
|
+
`, `
|
|
359
|
+
export class Super {
|
|
360
|
+
constructor(param: number) {
|
|
361
|
+
Array.isArray(param);
|
|
362
|
+
}
|
|
363
|
+
}
|
|
364
|
+
export class Sub extends Super {
|
|
365
|
+
}
|
|
366
|
+
`));
|
|
367
|
+
// ----------------------------------------------------------------------
|
|
368
|
+
test('method can be moved to supertype', () => util_1.expectNoError(`
|
|
369
|
+
export class Super {
|
|
370
|
+
}
|
|
371
|
+
export class Sub extends Super {
|
|
372
|
+
public foo(param: number) {
|
|
373
|
+
Array.isArray(param);
|
|
374
|
+
}
|
|
375
|
+
}
|
|
376
|
+
`, `
|
|
377
|
+
export class Super {
|
|
378
|
+
public foo(param: number) {
|
|
379
|
+
Array.isArray(param);
|
|
380
|
+
}
|
|
381
|
+
}
|
|
382
|
+
export class Sub extends Super {
|
|
383
|
+
}
|
|
384
|
+
`));
|
|
385
|
+
// ----------------------------------------------------------------------
|
|
386
|
+
test('property can be moved to supertype', () => util_1.expectNoError(`
|
|
387
|
+
export class Super {
|
|
388
|
+
}
|
|
389
|
+
export class Sub extends Super {
|
|
390
|
+
public foo: string = 'foo';
|
|
391
|
+
}
|
|
392
|
+
`, `
|
|
393
|
+
export class Super {
|
|
394
|
+
public foo: string = 'foo';
|
|
395
|
+
}
|
|
396
|
+
export class Sub extends Super {
|
|
397
|
+
}
|
|
398
|
+
`));
|
|
399
|
+
// ----------------------------------------------------------------------
|
|
400
|
+
test('subclassable is forever', () => util_1.expectError(/has gone from @subclassable to non-@subclassable/, `
|
|
401
|
+
/**
|
|
402
|
+
* @subclassable
|
|
403
|
+
*/
|
|
404
|
+
export class Super {
|
|
405
|
+
}
|
|
406
|
+
`, `
|
|
407
|
+
export class Super {
|
|
408
|
+
}
|
|
409
|
+
`));
|
|
410
|
+
// ----------------------------------------------------------------------
|
|
411
|
+
test('change from method to property', () => util_1.expectError(/changed from method to property/, `
|
|
412
|
+
export class Boom {
|
|
413
|
+
public foo() { return 12; }
|
|
414
|
+
}
|
|
415
|
+
`, `
|
|
416
|
+
export class Boom {
|
|
417
|
+
public get foo() { return 12; }
|
|
418
|
+
}
|
|
419
|
+
`));
|
|
420
|
+
// ----------------------------------------------------------------------
|
|
421
|
+
test('change from method with arguments to property', () => util_1.expectError(/changed from method to property/, `
|
|
422
|
+
export class Boom {
|
|
423
|
+
public foo(arg: number) { return 12 * arg; }
|
|
424
|
+
}
|
|
425
|
+
`, `
|
|
426
|
+
export class Boom {
|
|
427
|
+
public get foo() { return 12; }
|
|
428
|
+
}
|
|
429
|
+
`));
|
|
430
|
+
// ----------------------------------------------------------------------
|
|
431
|
+
test('change from property to method', () => util_1.expectError(/changed from property to method/, `
|
|
432
|
+
export class Boom {
|
|
433
|
+
public get foo() { return 12; }
|
|
434
|
+
}
|
|
435
|
+
`, `
|
|
436
|
+
export class Boom {
|
|
437
|
+
public foo() { return 12; }
|
|
438
|
+
}
|
|
439
|
+
`));
|
|
440
|
+
// ----------------------------------------------------------------------
|
|
441
|
+
test.each([
|
|
442
|
+
{
|
|
443
|
+
oldDecl: 'foo(arg: string) { Array.isArray(arg); }',
|
|
444
|
+
newDecl: 'foo(arg: string | number) { Array.isArray(arg); }',
|
|
445
|
+
},
|
|
446
|
+
{
|
|
447
|
+
oldDecl: 'foo(): string { return "x"; }',
|
|
448
|
+
newDecl: 'foo(): string | number { return "x"; }',
|
|
449
|
+
},
|
|
450
|
+
{
|
|
451
|
+
oldDecl: 'readonly foo: string = "x";',
|
|
452
|
+
newDecl: 'readonly foo: string | number = "x";',
|
|
453
|
+
},
|
|
454
|
+
])('cannot change any type in @subclassable class: %p to %p', ({ oldDecl, newDecl }) => util_1.expectError(/type is @subclassable/, `
|
|
455
|
+
/** @subclassable */
|
|
456
|
+
export class Boom {
|
|
457
|
+
public ${oldDecl}
|
|
458
|
+
}
|
|
459
|
+
`, `
|
|
460
|
+
/** @subclassable */
|
|
461
|
+
export class Boom {
|
|
462
|
+
public ${newDecl}
|
|
463
|
+
}
|
|
464
|
+
`));
|
|
465
|
+
// ----------------------------------------------------------------------
|
|
466
|
+
test.each([
|
|
467
|
+
{
|
|
468
|
+
oldDecl: 'foo(arg: string): void;',
|
|
469
|
+
newDecl: 'foo(arg: string | number): void;',
|
|
470
|
+
},
|
|
471
|
+
{ oldDecl: 'foo(): string;', newDecl: 'foo(): string | number;' },
|
|
472
|
+
{
|
|
473
|
+
oldDecl: 'readonly foo: string;',
|
|
474
|
+
newDecl: 'readonly foo: string | number;',
|
|
475
|
+
},
|
|
476
|
+
])('cannot change any type in @subclassable interface: %p to %p', ({ oldDecl, newDecl }) => util_1.expectError(/type is @subclassable/, `
|
|
477
|
+
/** @subclassable */
|
|
478
|
+
export interface IBoom {
|
|
479
|
+
${oldDecl}
|
|
480
|
+
}
|
|
481
|
+
`, `
|
|
482
|
+
/** @subclassable */
|
|
483
|
+
export interface IBoom {
|
|
484
|
+
${newDecl}
|
|
485
|
+
}
|
|
486
|
+
`));
|
|
487
|
+
// ----------------------------------------------------------------------
|
|
488
|
+
test.each([
|
|
489
|
+
// No usage => can add field
|
|
490
|
+
['', true],
|
|
491
|
+
// Return type => can add field
|
|
492
|
+
['foo(): TheStruct;', true],
|
|
493
|
+
['readonly foo: TheStruct;', true],
|
|
494
|
+
// Input type => can NOT add field
|
|
495
|
+
['foo: TheStruct;', false],
|
|
496
|
+
['foo(arg: TheStruct): void', false],
|
|
497
|
+
])('add required field to structs: refencing via %p -> allowed %p', (usageDecl, allowed) => util_1.expectError(allowed ? undefined : /newly required property 'fieldTwo' added/, `
|
|
498
|
+
export interface TheStruct {
|
|
499
|
+
readonly fieldOne: string;
|
|
500
|
+
}
|
|
501
|
+
|
|
502
|
+
export interface IConsumer {
|
|
503
|
+
${usageDecl}
|
|
504
|
+
}
|
|
505
|
+
`, `
|
|
506
|
+
export interface TheStruct {
|
|
507
|
+
readonly fieldOne: string;
|
|
508
|
+
readonly fieldTwo: string;
|
|
509
|
+
}
|
|
510
|
+
|
|
511
|
+
export interface IConsumer {
|
|
512
|
+
${usageDecl}
|
|
513
|
+
}
|
|
514
|
+
`));
|
|
515
|
+
// ----------------------------------------------------------------------
|
|
516
|
+
test.each([
|
|
517
|
+
// No usage => can add field
|
|
518
|
+
['', true],
|
|
519
|
+
// Return type => can NOT remove information
|
|
520
|
+
['foo(): TheStruct;', false],
|
|
521
|
+
['readonly foo: TheStruct;', false],
|
|
522
|
+
['foo: TheStruct;', false],
|
|
523
|
+
// Input type => can make optional
|
|
524
|
+
['foo(arg: TheStruct): void', true],
|
|
525
|
+
])('make required field optional: refencing via %p -> allowed %p', (usageDecl, allowed) => util_1.expectError(allowed ? undefined : /formerly required property 'fieldOne' is optional/, `
|
|
526
|
+
export interface TheStruct {
|
|
527
|
+
readonly fieldOne: string;
|
|
528
|
+
}
|
|
529
|
+
|
|
530
|
+
export interface IConsumer {
|
|
531
|
+
${usageDecl}
|
|
532
|
+
}
|
|
533
|
+
`, `
|
|
534
|
+
export interface TheStruct {
|
|
535
|
+
readonly fieldOne?: string;
|
|
536
|
+
}
|
|
537
|
+
|
|
538
|
+
export interface IConsumer {
|
|
539
|
+
${usageDecl}
|
|
540
|
+
}
|
|
541
|
+
`));
|
|
542
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"classes.test.js","sourceRoot":"","sources":["classes.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { expectError, expectNoError } from './util';\n\n// ----------------------------------------------------------------------\n\ntest('okay to add a new class', () =>\n  expectNoError(\n    `\n    export class Foo1 { }\n  `,\n    `\n    export class Foo1 { }\n    export class Foo2 { }\n  `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('okay to add a new function to a class', () =>\n  expectNoError(\n    `\n    export class Foo { }\n  `,\n    `\n    export class Foo {\n      public bar(): void { }\n    }\n  `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('not okay to add a required argument to a method', () =>\n  expectError(\n    /newly required argument/,\n    `\n    export class Foo {\n      public bar(arg1: string): void {\n        Array.isArray(arg1);\n      }\n    }\n  `,\n    `\n    export class Foo {\n      public bar(arg1: string, arg2: string): void {\n        Array.isArray(arg1);\n        Array.isArray(arg2);\n      }\n    }\n  `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('okay to make a required argument optional', () =>\n  expectNoError(\n    `\n    export class Foo {\n      public bar(arg1: string, arg2: string): void {\n        Array.isArray(arg1);\n        Array.isArray(arg2);\n      }\n    }\n  `,\n    `\n    export class Foo {\n      public bar(arg1: string, arg2?: string): void {\n        Array.isArray(arg1);\n        Array.isArray(arg2);\n      }\n    }\n  `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('okay to turn required arguments into varargs', () =>\n  expectNoError(\n    `\n    export class Foo {\n      public bar(arg1: string, arg2: number, arg3: number): void {\n        Array.isArray(arg1);\n        Array.isArray(arg2);\n        Array.isArray(arg3);\n      }\n    }\n  `,\n    `\n    export class Foo {\n      public bar(arg1: string, ...args: number[]): void {\n        Array.isArray(arg1);\n        Array.isArray(args);\n      }\n    }\n  `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('not allowed to change argument type to a different scalar', () =>\n  expectError(\n    /method.*foo.*argument arg1, takes number \\(formerly string\\): string is not assignable to number/i,\n    `\n    export class Foo {\n      public bar(arg1: string): void {\n        Array.isArray(arg1);\n      }\n    }\n  `,\n    `\n    export class Foo {\n      public bar(arg1: number): void {\n        Array.isArray(arg1);\n      }\n    }\n  `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('cannot add any abstract members to a subclassable class', () =>\n  expectError(\n    /adds requirement for subclasses to implement 'piet'./,\n    `\n    /**\n     * @subclassable\n     */\n    export abstract class Henk {\n      abstract readonly name: string;\n    }\n  `,\n    `\n    /**\n     * @subclassable\n     */\n    export abstract class Henk {\n      abstract readonly name: string;\n      abstract readonly piet: string;\n    }\n  `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('cannot add any members to a subclassable interface, not even optional ones', () =>\n  expectError(\n    /adds requirement for subclasses to implement 'piet'./,\n    `\n    /**\n     * @subclassable\n     */\n    export interface IHenk {\n      name: string;\n    }\n  `,\n    `\n    /**\n     * @subclassable\n     */\n    export interface IHenk {\n      name: string;\n      piet?: string;\n    }\n  `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('cannot make a member less visible', () =>\n  expectError(\n    /changed from 'public' to 'protected'/,\n    `\n    export class Henk {\n      public name: string = 'henk';\n    }\n  `,\n    `\n    export class Henk {\n      protected name: string = 'henk';\n    }\n  `,\n  ));\n\n// ----------------------------------------------------------------------\n\ndescribe('implement base types need to be present in updated type system', () => {\n  test('for interfaces', () =>\n    expectError(\n      /not assignable to all base types anymore/,\n      `\n      export interface IPapa {\n        readonly pipe: string;\n      }\n\n      export interface IBebe extends IPapa {\n        readonly pacifier: string;\n      }\n    `,\n      `\n      export interface IPapa {\n        readonly pipe: string;\n      }\n\n      export interface IBebe {\n        readonly pacifier: string;\n      }\n    `,\n    ));\n\n  test('for structs', () =>\n    expectError(\n      /not assignable to all base types anymore/,\n      `\n      export interface Papa {\n        readonly pipe: string;\n      }\n\n      export interface Bebe extends Papa {\n        readonly pacifier: string;\n      }\n    `,\n      `\n      export interface Papa {\n        readonly pipe: string;\n      }\n\n      export interface Bebe {\n        readonly pacifier: string;\n      }\n    `,\n    ));\n\n  test('for classes', () =>\n    expectError(\n      /not assignable to all base types anymore/,\n      `\n      export interface IPapa {\n        readonly pipe: string;\n      }\n\n      export class Bebe implements IPapa {\n        readonly pipe: string = 'pff';\n        readonly pacifier: string = 'mmm';\n      }\n    `,\n      `\n      export interface IPapa {\n        readonly pipe: string;\n      }\n\n      export class Bebe {\n        readonly pipe: string = 'pff';\n        readonly pacifier: string = 'mmm';\n      }\n    `,\n    ));\n\n  test('for base classes', () =>\n    expectError(\n      /not assignable to all base types anymore/,\n      `\n      export class Papa {\n        readonly pipe: string = 'pff';\n      }\n\n      export class Bebe extends Papa {\n        readonly pacifier: string = 'mmm';\n      }\n    `,\n      `\n      export class Papa {\n        readonly pipe: string = 'pff';\n      }\n\n      export class Bebe {\n        readonly pacifier: string = 'mmm';\n      }\n    `,\n    ));\n\n  test('new levels of inheritance are allowed', () =>\n    expectNoError(\n      `\n      export class Papa {\n        readonly pipe: string = 'pff';\n      }\n\n      export class Bebe extends Papa {\n        readonly pacifier: string = 'mmm';\n      }\n    `,\n      `\n      export class Papa {\n        readonly pipe: string = 'pff';\n      }\n\n      export class Inbetween extends Papa {\n      }\n\n      export class Bebe extends Inbetween {\n        readonly pacifier: string = 'mmm';\n      }\n    `,\n    ));\n\n  test('change direct implementation to inherited implementation via interface', () =>\n    expectNoError(\n      `\n      export interface IPapa {\n        readonly pipe: string;\n      }\n\n      export class Bebe implements IPapa {\n        readonly pipe: string = 'pff';\n        readonly pacifier: string = 'mmm';\n      }\n    `,\n      `\n      export interface IPapa {\n        readonly pipe: string;\n      }\n\n      export interface IInbetween extends IPapa {\n      }\n\n      export class Bebe implements IInbetween {\n        readonly pipe: string = 'pff';\n        readonly pacifier: string = 'mmm';\n      }\n    `,\n    ));\n\n  test('change direct implementation to inherited implementation via base class', () =>\n    expectNoError(\n      `\n      export interface IPapa {\n        readonly pipe: string;\n      }\n\n      export class Bebe implements IPapa {\n        readonly pipe: string = 'pff';\n        readonly pacifier: string = 'mmm';\n      }\n    `,\n      `\n      export interface IPapa {\n        readonly pipe: string;\n      }\n\n      export class Inbetween implements IPapa {\n        readonly pipe: string = 'pff';\n      }\n\n      export class Bebe extends Inbetween {\n        readonly pacifier: string = 'mmm';\n      }\n    `,\n    ));\n});\n\n// ----------------------------------------------------------------------\n\ntest.each([\n  {\n    oldDecl: 'name: string',\n    newDecl: 'name?: string',\n    error:\n      /type Optional<string> \\(formerly string\\): output type is now optional/,\n  },\n  {\n    oldDecl: 'name?: string',\n    newDecl: 'name: string',\n    error: undefined, // Strengthening is okay\n  },\n  {\n    oldDecl: 'name: string',\n    newDecl: 'name: string | number',\n    error: /string \\| number is not assignable to string/,\n  },\n  {\n    oldDecl: 'name: string | number',\n    newDecl: 'name: string',\n    error: undefined, // Strengthening is okay\n  },\n])('change class property ', ({ oldDecl, newDecl, error }) =>\n  expectError(\n    error,\n    `\n    export class Henk {\n      public readonly ${oldDecl} = 'henk';\n    }\n  `,\n    `\n    export class Henk {\n      public readonly ${newDecl} = 'henk';\n    }\n  `,\n  ),\n);\n\n// ----------------------------------------------------------------------\n\ntest.each([\n  {\n    oldDecl: 'name: string',\n    newDecl: 'name?: string',\n    error: /changed to Optional<string> \\(formerly string\\)/,\n  },\n  {\n    oldDecl: 'name?: string',\n    newDecl: 'name: string',\n    error: /changed to string \\(formerly Optional<string>\\)/,\n  },\n  {\n    oldDecl: 'name: string',\n    newDecl: 'name: string | number',\n    error: /changed to string \\| number \\(formerly string\\)/,\n  },\n  {\n    oldDecl: 'name: string | number',\n    newDecl: 'name: string',\n    error: /changed to string \\(formerly string \\| number\\)/,\n  },\n])(\n  'cannot change a mutable class property type: %p to %p',\n  ({ oldDecl, newDecl, error }) =>\n    expectError(\n      error,\n      `\n      export class Henk {\n        public ${oldDecl} = 'henk';\n      }\n    `,\n      `\n      export class Henk {\n        public ${newDecl} = 'henk';\n      }\n    `,\n    ),\n);\n\n// ----------------------------------------------------------------------\n\ntest('consider inherited constructor', () =>\n  expectError(\n    /number is not assignable to string/,\n    `\n    export class Super {\n      constructor(param: number) {\n        Array.isArray(param);\n      }\n    }\n    export class Sub extends Super {\n    }\n  `,\n    `\n    export class Super {\n      constructor(param: number) {\n        Array.isArray(param);\n      }\n    }\n    export class Sub extends Super {\n      constructor(param: string) {\n        super(5);\n        Array.isArray(param);\n      }\n    }\n  `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('consider inherited constructor, the other way', () =>\n  expectError(\n    /newly required argument/,\n    `\n    export class Super {\n      constructor(param: number) {\n        Array.isArray(param);\n      }\n    }\n    export class Sub extends Super {\n      constructor() {\n        super(5);\n      }\n    }\n  `,\n    `\n    export class Super {\n      constructor(param: number) {\n        Array.isArray(param);\n      }\n    }\n    export class Sub extends Super {\n    }\n  `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('method can be moved to supertype', () =>\n  expectNoError(\n    `\n    export class Super {\n    }\n    export class Sub extends Super {\n      public foo(param: number) {\n        Array.isArray(param);\n      }\n    }\n  `,\n    `\n    export class Super {\n      public foo(param: number) {\n        Array.isArray(param);\n      }\n    }\n    export class Sub extends Super {\n    }\n  `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('property can be moved to supertype', () =>\n  expectNoError(\n    `\n    export class Super {\n    }\n    export class Sub extends Super {\n      public foo: string = 'foo';\n    }\n  `,\n    `\n    export class Super {\n      public foo: string = 'foo';\n    }\n    export class Sub extends Super {\n    }\n  `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('subclassable is forever', () =>\n  expectError(\n    /has gone from @subclassable to non-@subclassable/,\n    `\n    /**\n     * @subclassable\n     */\n    export class Super {\n    }\n  `,\n    `\n    export class Super {\n    }\n  `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('change from method to property', () =>\n  expectError(\n    /changed from method to property/,\n    `\n    export class Boom {\n      public foo() { return 12; }\n    }\n    `,\n    `\n    export class Boom {\n      public get foo() { return 12; }\n    }\n    `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('change from method with arguments to property', () =>\n  expectError(\n    /changed from method to property/,\n    `\n    export class Boom {\n      public foo(arg: number) { return 12 * arg; }\n    }\n    `,\n    `\n    export class Boom {\n      public get foo() { return 12; }\n    }\n    `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest('change from property to method', () =>\n  expectError(\n    /changed from property to method/,\n    `\n    export class Boom {\n      public get foo() { return 12; }\n    }\n    `,\n    `\n    export class Boom {\n      public foo() { return 12; }\n    }\n    `,\n  ));\n\n// ----------------------------------------------------------------------\n\ntest.each([\n  {\n    oldDecl: 'foo(arg: string) { Array.isArray(arg); }',\n    newDecl: 'foo(arg: string | number) { Array.isArray(arg); }',\n  },\n  {\n    oldDecl: 'foo(): string { return \"x\"; }',\n    newDecl: 'foo(): string | number { return \"x\"; }',\n  },\n  {\n    oldDecl: 'readonly foo: string = \"x\";',\n    newDecl: 'readonly foo: string | number = \"x\";',\n  },\n])(\n  'cannot change any type in @subclassable class: %p to %p',\n  ({ oldDecl, newDecl }) =>\n    expectError(\n      /type is @subclassable/,\n      `\n      /** @subclassable */\n      export class Boom {\n        public ${oldDecl}\n      }\n      `,\n      `\n      /** @subclassable */\n      export class Boom {\n        public ${newDecl}\n      }\n      `,\n    ),\n);\n\n// ----------------------------------------------------------------------\n\ntest.each([\n  {\n    oldDecl: 'foo(arg: string): void;',\n    newDecl: 'foo(arg: string | number): void;',\n  },\n  { oldDecl: 'foo(): string;', newDecl: 'foo(): string | number;' },\n  {\n    oldDecl: 'readonly foo: string;',\n    newDecl: 'readonly foo: string | number;',\n  },\n])(\n  'cannot change any type in @subclassable interface: %p to %p',\n  ({ oldDecl, newDecl }) =>\n    expectError(\n      /type is @subclassable/,\n      `\n      /** @subclassable */\n      export interface IBoom {\n        ${oldDecl}\n      }\n      `,\n      `\n      /** @subclassable */\n      export interface IBoom {\n        ${newDecl}\n      }\n      `,\n    ),\n);\n\n// ----------------------------------------------------------------------\n\ntest.each([\n  // No usage => can add field\n  ['', true],\n  // Return type => can add field\n  ['foo(): TheStruct;', true],\n  ['readonly foo: TheStruct;', true],\n  // Input type => can NOT add field\n  ['foo: TheStruct;', false],\n  ['foo(arg: TheStruct): void', false],\n])(\n  'add required field to structs: refencing via %p -> allowed %p',\n  (usageDecl, allowed) =>\n    expectError(\n      allowed ? undefined : /newly required property 'fieldTwo' added/,\n      `\n      export interface TheStruct {\n        readonly fieldOne: string;\n      }\n\n      export interface IConsumer {\n        ${usageDecl}\n      }\n      `,\n      `\n      export interface TheStruct {\n        readonly fieldOne: string;\n        readonly fieldTwo: string;\n      }\n\n      export interface IConsumer {\n        ${usageDecl}\n      }\n      `,\n    ),\n);\n\n// ----------------------------------------------------------------------\n\ntest.each([\n  // No usage => can add field\n  ['', true],\n  // Return type => can NOT remove information\n  ['foo(): TheStruct;', false],\n  ['readonly foo: TheStruct;', false],\n  ['foo: TheStruct;', false],\n  // Input type => can make optional\n  ['foo(arg: TheStruct): void', true],\n])(\n  'make required field optional: refencing via %p -> allowed %p',\n  (usageDecl, allowed) =>\n    expectError(\n      allowed ? undefined : /formerly required property 'fieldOne' is optional/,\n      `\n      export interface TheStruct {\n        readonly fieldOne: string;\n      }\n\n      export interface IConsumer {\n        ${usageDecl}\n      }\n      `,\n      `\n      export interface TheStruct {\n        readonly fieldOne?: string;\n      }\n\n      export interface IConsumer {\n        ${usageDecl}\n      }\n      `,\n    ),\n);\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|