@cparra/apex-reflection 2.8.0 → 2.9.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.
@@ -1,346 +1,374 @@
1
- import {ClassMirror, InterfaceMirror, reflect} from '../index';
2
-
3
- describe('Enum Reflection', () => {
4
- test('Simple, single line declaration', () => {
5
- const enumBody = 'enum MyEnumName {}';
6
- const result = reflect(enumBody).typeMirror;
7
- expect(result.type_name).toBe('enum');
8
- expect(result.name).toBe('MyEnumName');
9
- });
10
-
11
- test('Multi-line declaration with values', () => {
12
- const enumBody = `
13
- enum MyEnumName {
14
- VALUE_1,
15
- VALUE2
16
- }
17
- `;
18
- const result = reflect(enumBody).typeMirror;
19
- expect(result.type_name).toBe('enum');
20
- expect(result.name).toBe('MyEnumName');
21
- });
22
-
23
- test('With doc comments', () => {
24
- const enumBody = `
25
- /**
26
- * My enum description
27
- */
28
- enum MyEnumName {
29
- VALUE_1,
30
- VALUE2
31
- }
32
- `;
33
- const result = reflect(enumBody).typeMirror;
34
- expect(result.docComment.description).toBe('My enum description');
35
- });
36
- });
37
-
38
- describe('Interface Reflection', () => {
39
- test('Single line interface definition', () => {
40
- const interfaceBody = 'interface MyInterface{}';
41
- const result = reflect(interfaceBody).typeMirror;
42
- expect(result.type_name).toBe('interface');
43
- expect(result.name).toBe('MyInterface');
44
- });
45
-
46
- test('When no access modifier is defined it is private', () => {
47
- const interfaceBody = 'interface MyInterface{}';
48
- const result = reflect(interfaceBody).typeMirror;
49
- expect(result.access_modifier).toBe('private');
50
- });
51
-
52
- test('Can have access modifier', () => {
53
- const interfaceBody = 'public interface MyInterface{}';
54
- const result = reflect(interfaceBody).typeMirror;
55
- expect(result.access_modifier).toBe('public');
56
- });
57
-
58
- test('Can have a sharing modifier', () => {
59
- const interfaceBody = 'public with sharing interface MyInterface{}';
60
- const result = (reflect(interfaceBody).typeMirror) as InterfaceMirror;
61
- expect(result.sharingModifier).toBe('withSharing');
62
- });
63
-
64
- test('Can have methods', () => {
65
- const interfaceBody = `
66
- public with sharing interface MyInterface{
67
- void method1();
68
- }
69
- `;
70
- const result = (reflect(interfaceBody).typeMirror) as InterfaceMirror;
71
- expect(result.methods.length).toBe(1);
72
- expect(result.methods[0].name).toBe('method1');
73
- });
74
-
75
- test('Can have extend other interfaces', () => {
76
- const interfaceBody = `
77
- public with sharing interface MyInterface extends Interface2 {
78
- void method1();
79
- }
80
- `;
81
- const result = (reflect(interfaceBody).typeMirror) as InterfaceMirror;
82
- expect(result.extended_interfaces.length).toBe(1);
83
- expect(result.extended_interfaces[0]).toBe('Interface2');
84
- });
85
-
86
- test('Can have annotations', () => {
87
- const interfaceBody = `
88
- @NamespaceAccessible
89
- public with sharing interface MyInterface{
90
- void method1();
91
- }
92
- `;
93
- const result = (reflect(interfaceBody).typeMirror) as InterfaceMirror;
94
- expect(result.annotations.length).toBe(1);
95
- expect(result.annotations[0].name).toBe('namespaceaccessible');
96
- });
97
- });
98
-
99
- describe('Class reflection', () => {
100
- test('Single line class definition', () => {
101
- const classBody = 'class MyClass{}';
102
- const result = reflect(classBody).typeMirror;
103
- expect(result.type_name).toBe('class');
104
- expect(result.name).toBe('MyClass');
105
- });
106
-
107
- test('When no access modifier is defined it is private', () => {
108
- const classBody = 'class MyClass{}';
109
- const result = reflect(classBody).typeMirror;
110
- expect(result.access_modifier).toBe('private');
111
- });
112
-
113
- test('Can have access modifier', () => {
114
- const interfaceBody = 'public class MyClass{}';
115
- const result = reflect(interfaceBody).typeMirror;
116
- expect(result.access_modifier).toBe('public');
117
- });
118
-
119
- test('Can have a sharing modifier', () => {
120
- const classBody = 'public with sharing class MyClass{}';
121
- const result = (reflect(classBody)).typeMirror as ClassMirror;
122
- expect(result.sharingModifier).toBe('withSharing');
123
- });
124
-
125
- test('Can have a class modifier', () => {
126
- const classBody = 'public with sharing abstract class MyClass{}';
127
- const result = (reflect(classBody)).typeMirror as ClassMirror;
128
- expect(result.classModifier).toBe('abstract');
129
- });
130
-
131
- test('Can extend a class', () => {
132
- const classBody = 'public with sharing class MyClass extends Class2 {}';
133
- const result = (reflect(classBody)).typeMirror as ClassMirror;
134
- expect(result.extended_class).toBe('Class2');
135
- });
136
-
137
- test('Can implement interfaces', () => {
138
- const classBody = 'public with sharing class MyClass implements Interface1, Interface2 {}';
139
- const result = (reflect(classBody)).typeMirror as ClassMirror;
140
- expect(result.implemented_interfaces.length).toBe(2);
141
- expect(result.implemented_interfaces[0]).toBe('Interface1');
142
- expect(result.implemented_interfaces[1]).toBe('Interface2');
143
- });
144
-
145
- test('Can have properties', () => {
146
- const classBody = `
147
- public with sharing class MyClass {
148
- public String Prop1 { get; set; }
149
- public Integer Prop2 { get; set; }
150
- }
151
- `;
152
- const result = (reflect(classBody)).typeMirror as ClassMirror;
153
- expect(result.properties.length).toBe(2);
154
- expect(result.properties[0].typeReference.type).toBe('String');
155
- expect(result.properties[0].name).toBe('Prop1');
156
- expect(result.properties[1].typeReference.type).toBe('Integer');
157
- expect(result.properties[1].name).toBe('Prop2');
158
- });
159
-
160
- test('Can have fields', () => {
161
- const classBody = `
162
- public with sharing class MyClass {
163
- private String var1, var2;
164
- }
165
- `;
166
- const result = (reflect(classBody)).typeMirror as ClassMirror;
167
- expect(result.fields.length).toBe(2);
168
- expect(result.fields[0].typeReference.type).toBe('String');
169
- expect(result.fields[1].typeReference.type).toBe('String');
170
- expect(result.fields[0].name).toBe('var1');
171
- expect(result.fields[1].name).toBe('var2');
172
- });
173
-
174
- test('Can have transient fields', () => {
175
- const classBody = `
176
- public with sharing class MyClass {
177
- transient String var1;
178
- }
179
- `;
180
- const result = (reflect(classBody)).typeMirror as ClassMirror;
181
- expect(result.fields.length).toBe(1);
182
- expect(result.fields[0].typeReference.type).toBe('String');
183
- expect(result.fields[0].name).toBe('var1');
184
- expect(result.fields[0].memberModifiers).toContain('transient');
185
- });
186
-
187
- test('Can have annotations with parameters', () => {
188
- const classBody = `
189
- @IsTest(SeeAllData=true)
190
- /** Some docs */
191
- public with sharing class MyClass {}
192
- `;
193
- const result = (reflect(classBody)).typeMirror as ClassMirror;
194
- expect(result.annotations.length).toBe(1);
195
- expect(result.annotations[0].elementValues.length).toBe(1);
196
- expect(result.annotations[0].elementValues[0].key).toBe('SeeAllData');
197
- expect(result.annotations[0].elementValues[0].value).toBe('true');
198
- });
199
-
200
- test('Can have annotations with parameters after the docs', () => {
201
- const classBody = `
202
- /**
203
- * @description Account related operations.
204
- */
205
- @RestResource(urlMapping='/Account/*')
206
- global with sharing class SampleRestResource {}
207
- `;
208
- const result = (reflect(classBody)).typeMirror as ClassMirror;
209
- expect(result.annotations.length).toBe(1);
210
- expect(result.annotations[0].elementValues.length).toBe(1);
211
- expect(result.annotations[0].elementValues[0].key).toBe('urlMapping');
212
- expect(result.annotations[0].elementValues[0].value).toBe('\'/Account/*\'');
213
- });
214
-
215
- test('Can have constructors', () => {
216
- const classBody = `
217
- public with sharing class MyClass {
218
- public MyClass() {}
219
- public MyClass(String var1) {}
220
- }
221
- `;
222
- const result = (reflect(classBody)).typeMirror as ClassMirror;
223
- expect(result.constructors.length).toBe(2);
224
- expect(result.constructors[0].parameters.length).toBe(0);
225
- expect(result.constructors[0].access_modifier).toBe('public');
226
- expect(result.constructors[1].parameters.length).toBe(1);
227
- expect(result.constructors[1].parameters[0].name).toBe('var1');
228
- expect(result.constructors[1].parameters[0].typeReference.type).toBe('String');
229
- expect(result.constructors[1].access_modifier).toBe('public');
230
- });
231
-
232
- test('Can have methods', () => {
233
- const classBody = `
234
- public with sharing class MyClass {
235
- public static String method1() {
236
- return '';
237
- }
238
-
239
- private void method2(){}
240
- }
241
- `;
242
- const result = (reflect(classBody)).typeMirror as ClassMirror;
243
- expect(result.methods.length).toBe(2);
244
- expect(result.methods[0].memberModifiers.length).toBe(1);
245
- expect(result.methods[0].memberModifiers[0]).toBe('static');
246
- expect(result.methods[0].access_modifier).toBe('public');
247
- expect(result.methods[0].typeReference.type).toBe('String');
248
- expect(result.methods[0].name).toBe('method1');
249
-
250
- expect(result.methods[1].memberModifiers.length).toBe(0);
251
- expect(result.methods[1].access_modifier).toBe('private');
252
- expect(result.methods[1].typeReference.type).toBe('void');
253
- expect(result.methods[1].name).toBe('method2');
254
- });
255
-
256
- test('Can have virtual methods', () => {
257
- const classBody = `
258
- public with sharing class MyClass {
259
- public virtual String method1() {
260
- return null ?? '';
261
- }
262
- }
263
- `;
264
- const result = (reflect(classBody)).typeMirror as ClassMirror;
265
- expect(result.methods[0].memberModifiers[0]).toBe('virtual');
266
- });
267
-
268
- test('Can have abstract methods', () => {
269
- const classBody = `
270
- public with sharing abstract class MyClass {
271
- public abstract String method1() {
272
- return null ?? '';
273
- }
274
- }
275
- `;
276
- const result = (reflect(classBody)).typeMirror as ClassMirror;
277
- expect(result.methods[0].memberModifiers[0]).toBe('abstract');
278
- });
279
-
280
- test('Can have inner enums', () => {
281
- const classBody = `
282
- public with sharing class MyClass {
283
- public enum MyEnum {}
284
- }
285
- `;
286
- const result = (reflect(classBody)).typeMirror as ClassMirror;
287
- expect(result.enums.length).toBe(1);
288
- expect(result.enums[0].access_modifier).toBe('public');
289
- expect(result.enums[0].name).toBe('MyEnum');
290
- });
291
-
292
- test('Can have inner interfaces', () => {
293
- const classBody = `
294
- public with sharing class MyClass {
295
- public interface MyInterface {
296
- void method1();
297
- void method2();
298
- }
299
- }
300
- `;
301
- const result = (reflect(classBody)).typeMirror as ClassMirror;
302
- expect(result.interfaces.length).toBe(1);
303
- expect(result.interfaces[0].name).toBe('MyInterface');
304
- expect(result.interfaces[0].methods.length).toBe(2);
305
- });
306
-
307
- test('Can have inner classes', () => {
308
- const classBody = `
309
- public with sharing class MyClass {
310
- public class MyClass {
311
- public void method1();
312
- public void method2();
313
- }
314
- }
315
- `;
316
- const result = (reflect(classBody)).typeMirror as ClassMirror;
317
- expect(result.classes.length).toBe(1);
318
- expect(result.classes[0].name).toBe('MyClass');
319
- expect(result.classes[0].methods.length).toBe(2);
320
- });
321
-
322
- test('Can have members in groups', () => {
323
- const classBody = `
324
- public with sharing class MyClass {
325
- /**
326
- * @start-group Group Name
327
- * @description Group Description
328
- */
329
- public String Prop1 { get; set; }
330
- public Integer Prop2 { get; set; }
331
- /** @end-group */
332
- }
333
- `;
334
-
335
- const result = (reflect(classBody)).typeMirror as ClassMirror;
336
- expect(result.properties.length).toBe(2);
337
- expect(result.properties[0].typeReference.type).toBe('String');
338
- expect(result.properties[0].name).toBe('Prop1');
339
- expect(result.properties[0].group).toBe('Group Name');
340
- expect(result.properties[0].groupDescription).toBe('Group Description');
341
- expect(result.properties[1].typeReference.type).toBe('Integer');
342
- expect(result.properties[1].name).toBe('Prop2');
343
- expect(result.properties[1].group).toBe('Group Name');
344
- expect(result.properties[1].groupDescription).toBe('Group Description');
345
- });
346
- });
1
+ import {ClassMirror, EnumMirror, InterfaceMirror, reflect} from '../index';
2
+
3
+ describe('Enum Reflection', () => {
4
+ test('Simple, single line declaration', () => {
5
+ const enumBody = 'enum MyEnumName {}';
6
+ const result = reflect(enumBody).typeMirror;
7
+ expect(result.type_name).toBe('enum');
8
+ expect(result.name).toBe('MyEnumName');
9
+ });
10
+
11
+ test('Multi-line declaration with values', () => {
12
+ const enumBody = `
13
+ enum MyEnumName {
14
+ VALUE_1,
15
+ VALUE2
16
+ }
17
+ `;
18
+ const result = reflect(enumBody).typeMirror;
19
+ expect(result.type_name).toBe('enum');
20
+ expect(result.name).toBe('MyEnumName');
21
+ });
22
+
23
+ test('With doc comments', () => {
24
+ const enumBody = `
25
+ /**
26
+ * My enum description
27
+ */
28
+ enum MyEnumName {
29
+ VALUE_1,
30
+ VALUE2
31
+ }
32
+ `;
33
+ const result = reflect(enumBody).typeMirror;
34
+ expect(result.docComment.description).toBe('My enum description');
35
+ });
36
+
37
+ test('Enums can have values', () => {
38
+ const enumBody = `
39
+ enum MyEnumName {
40
+ VALUE_1,
41
+ VALUE2
42
+ }
43
+ `;
44
+ const result = reflect(enumBody).typeMirror as EnumMirror;
45
+ expect(result.values.length).toBe(2);
46
+ expect(result.values[0].name).toBe('VALUE_1');
47
+ expect(result.values[1].name).toBe('VALUE2');
48
+ });
49
+
50
+ test('Enum values can have descriptions', () => {
51
+ const enumBody = `
52
+ enum MyEnumName {
53
+ /**
54
+ * Value 1 description
55
+ */
56
+ VALUE_1,
57
+ VALUE2
58
+ }
59
+ `;
60
+ const result = reflect(enumBody).typeMirror as EnumMirror;
61
+ expect(result.values.length).toBe(2);
62
+ expect(result.values[0].docComment.description).toBe('Value 1 description');
63
+ });
64
+ });
65
+
66
+ describe('Interface Reflection', () => {
67
+ test('Single line interface definition', () => {
68
+ const interfaceBody = 'interface MyInterface{}';
69
+ const result = reflect(interfaceBody).typeMirror;
70
+ expect(result.type_name).toBe('interface');
71
+ expect(result.name).toBe('MyInterface');
72
+ });
73
+
74
+ test('When no access modifier is defined it is private', () => {
75
+ const interfaceBody = 'interface MyInterface{}';
76
+ const result = reflect(interfaceBody).typeMirror;
77
+ expect(result.access_modifier).toBe('private');
78
+ });
79
+
80
+ test('Can have access modifier', () => {
81
+ const interfaceBody = 'public interface MyInterface{}';
82
+ const result = reflect(interfaceBody).typeMirror;
83
+ expect(result.access_modifier).toBe('public');
84
+ });
85
+
86
+ test('Can have a sharing modifier', () => {
87
+ const interfaceBody = 'public with sharing interface MyInterface{}';
88
+ const result = (reflect(interfaceBody).typeMirror) as InterfaceMirror;
89
+ expect(result.sharingModifier).toBe('withSharing');
90
+ });
91
+
92
+ test('Can have methods', () => {
93
+ const interfaceBody = `
94
+ public with sharing interface MyInterface{
95
+ void method1();
96
+ }
97
+ `;
98
+ const result = (reflect(interfaceBody).typeMirror) as InterfaceMirror;
99
+ expect(result.methods.length).toBe(1);
100
+ expect(result.methods[0].name).toBe('method1');
101
+ });
102
+
103
+ test('Can have extend other interfaces', () => {
104
+ const interfaceBody = `
105
+ public with sharing interface MyInterface extends Interface2 {
106
+ void method1();
107
+ }
108
+ `;
109
+ const result = (reflect(interfaceBody).typeMirror) as InterfaceMirror;
110
+ expect(result.extended_interfaces.length).toBe(1);
111
+ expect(result.extended_interfaces[0]).toBe('Interface2');
112
+ });
113
+
114
+ test('Can have annotations', () => {
115
+ const interfaceBody = `
116
+ @NamespaceAccessible
117
+ public with sharing interface MyInterface{
118
+ void method1();
119
+ }
120
+ `;
121
+ const result = (reflect(interfaceBody).typeMirror) as InterfaceMirror;
122
+ expect(result.annotations.length).toBe(1);
123
+ expect(result.annotations[0].name).toBe('namespaceaccessible');
124
+ });
125
+ });
126
+
127
+ describe('Class reflection', () => {
128
+ test('Single line class definition', () => {
129
+ const classBody = 'class MyClass{}';
130
+ const result = reflect(classBody).typeMirror;
131
+ expect(result.type_name).toBe('class');
132
+ expect(result.name).toBe('MyClass');
133
+ });
134
+
135
+ test('When no access modifier is defined it is private', () => {
136
+ const classBody = 'class MyClass{}';
137
+ const result = reflect(classBody).typeMirror;
138
+ expect(result.access_modifier).toBe('private');
139
+ });
140
+
141
+ test('Can have access modifier', () => {
142
+ const interfaceBody = 'public class MyClass{}';
143
+ const result = reflect(interfaceBody).typeMirror;
144
+ expect(result.access_modifier).toBe('public');
145
+ });
146
+
147
+ test('Can have a sharing modifier', () => {
148
+ const classBody = 'public with sharing class MyClass{}';
149
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
150
+ expect(result.sharingModifier).toBe('withSharing');
151
+ });
152
+
153
+ test('Can have a class modifier', () => {
154
+ const classBody = 'public with sharing abstract class MyClass{}';
155
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
156
+ expect(result.classModifier).toBe('abstract');
157
+ });
158
+
159
+ test('Can extend a class', () => {
160
+ const classBody = 'public with sharing class MyClass extends Class2 {}';
161
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
162
+ expect(result.extended_class).toBe('Class2');
163
+ });
164
+
165
+ test('Can implement interfaces', () => {
166
+ const classBody = 'public with sharing class MyClass implements Interface1, Interface2 {}';
167
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
168
+ expect(result.implemented_interfaces.length).toBe(2);
169
+ expect(result.implemented_interfaces[0]).toBe('Interface1');
170
+ expect(result.implemented_interfaces[1]).toBe('Interface2');
171
+ });
172
+
173
+ test('Can have properties', () => {
174
+ const classBody = `
175
+ public with sharing class MyClass {
176
+ public String Prop1 { get; set; }
177
+ public Integer Prop2 { get; set; }
178
+ }
179
+ `;
180
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
181
+ expect(result.properties.length).toBe(2);
182
+ expect(result.properties[0].typeReference.type).toBe('String');
183
+ expect(result.properties[0].name).toBe('Prop1');
184
+ expect(result.properties[1].typeReference.type).toBe('Integer');
185
+ expect(result.properties[1].name).toBe('Prop2');
186
+ });
187
+
188
+ test('Can have fields', () => {
189
+ const classBody = `
190
+ public with sharing class MyClass {
191
+ private String var1, var2;
192
+ }
193
+ `;
194
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
195
+ expect(result.fields.length).toBe(2);
196
+ expect(result.fields[0].typeReference.type).toBe('String');
197
+ expect(result.fields[1].typeReference.type).toBe('String');
198
+ expect(result.fields[0].name).toBe('var1');
199
+ expect(result.fields[1].name).toBe('var2');
200
+ });
201
+
202
+ test('Can have transient fields', () => {
203
+ const classBody = `
204
+ public with sharing class MyClass {
205
+ transient String var1;
206
+ }
207
+ `;
208
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
209
+ expect(result.fields.length).toBe(1);
210
+ expect(result.fields[0].typeReference.type).toBe('String');
211
+ expect(result.fields[0].name).toBe('var1');
212
+ expect(result.fields[0].memberModifiers).toContain('transient');
213
+ });
214
+
215
+ test('Can have annotations with parameters', () => {
216
+ const classBody = `
217
+ @IsTest(SeeAllData=true)
218
+ /** Some docs */
219
+ public with sharing class MyClass {}
220
+ `;
221
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
222
+ expect(result.annotations.length).toBe(1);
223
+ expect(result.annotations[0].elementValues.length).toBe(1);
224
+ expect(result.annotations[0].elementValues[0].key).toBe('SeeAllData');
225
+ expect(result.annotations[0].elementValues[0].value).toBe('true');
226
+ });
227
+
228
+ test('Can have annotations with parameters after the docs', () => {
229
+ const classBody = `
230
+ /**
231
+ * @description Account related operations.
232
+ */
233
+ @RestResource(urlMapping='/Account/*')
234
+ global with sharing class SampleRestResource {}
235
+ `;
236
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
237
+ expect(result.annotations.length).toBe(1);
238
+ expect(result.annotations[0].elementValues.length).toBe(1);
239
+ expect(result.annotations[0].elementValues[0].key).toBe('urlMapping');
240
+ expect(result.annotations[0].elementValues[0].value).toBe('\'/Account/*\'');
241
+ });
242
+
243
+ test('Can have constructors', () => {
244
+ const classBody = `
245
+ public with sharing class MyClass {
246
+ public MyClass() {}
247
+ public MyClass(String var1) {}
248
+ }
249
+ `;
250
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
251
+ expect(result.constructors.length).toBe(2);
252
+ expect(result.constructors[0].parameters.length).toBe(0);
253
+ expect(result.constructors[0].access_modifier).toBe('public');
254
+ expect(result.constructors[1].parameters.length).toBe(1);
255
+ expect(result.constructors[1].parameters[0].name).toBe('var1');
256
+ expect(result.constructors[1].parameters[0].typeReference.type).toBe('String');
257
+ expect(result.constructors[1].access_modifier).toBe('public');
258
+ });
259
+
260
+ test('Can have methods', () => {
261
+ const classBody = `
262
+ public with sharing class MyClass {
263
+ public static String method1() {
264
+ return '';
265
+ }
266
+
267
+ private void method2(){}
268
+ }
269
+ `;
270
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
271
+ expect(result.methods.length).toBe(2);
272
+ expect(result.methods[0].memberModifiers.length).toBe(1);
273
+ expect(result.methods[0].memberModifiers[0]).toBe('static');
274
+ expect(result.methods[0].access_modifier).toBe('public');
275
+ expect(result.methods[0].typeReference.type).toBe('String');
276
+ expect(result.methods[0].name).toBe('method1');
277
+
278
+ expect(result.methods[1].memberModifiers.length).toBe(0);
279
+ expect(result.methods[1].access_modifier).toBe('private');
280
+ expect(result.methods[1].typeReference.type).toBe('void');
281
+ expect(result.methods[1].name).toBe('method2');
282
+ });
283
+
284
+ test('Can have virtual methods', () => {
285
+ const classBody = `
286
+ public with sharing class MyClass {
287
+ public virtual String method1() {
288
+ return null ?? '';
289
+ }
290
+ }
291
+ `;
292
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
293
+ expect(result.methods[0].memberModifiers[0]).toBe('virtual');
294
+ });
295
+
296
+ test('Can have abstract methods', () => {
297
+ const classBody = `
298
+ public with sharing abstract class MyClass {
299
+ public abstract String method1() {
300
+ return null ?? '';
301
+ }
302
+ }
303
+ `;
304
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
305
+ expect(result.methods[0].memberModifiers[0]).toBe('abstract');
306
+ });
307
+
308
+ test('Can have inner enums', () => {
309
+ const classBody = `
310
+ public with sharing class MyClass {
311
+ public enum MyEnum {}
312
+ }
313
+ `;
314
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
315
+ expect(result.enums.length).toBe(1);
316
+ expect(result.enums[0].access_modifier).toBe('public');
317
+ expect(result.enums[0].name).toBe('MyEnum');
318
+ });
319
+
320
+ test('Can have inner interfaces', () => {
321
+ const classBody = `
322
+ public with sharing class MyClass {
323
+ public interface MyInterface {
324
+ void method1();
325
+ void method2();
326
+ }
327
+ }
328
+ `;
329
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
330
+ expect(result.interfaces.length).toBe(1);
331
+ expect(result.interfaces[0].name).toBe('MyInterface');
332
+ expect(result.interfaces[0].methods.length).toBe(2);
333
+ });
334
+
335
+ test('Can have inner classes', () => {
336
+ const classBody = `
337
+ public with sharing class MyClass {
338
+ public class MyClass {
339
+ public void method1();
340
+ public void method2();
341
+ }
342
+ }
343
+ `;
344
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
345
+ expect(result.classes.length).toBe(1);
346
+ expect(result.classes[0].name).toBe('MyClass');
347
+ expect(result.classes[0].methods.length).toBe(2);
348
+ });
349
+
350
+ test('Can have members in groups', () => {
351
+ const classBody = `
352
+ public with sharing class MyClass {
353
+ /**
354
+ * @start-group Group Name
355
+ * @description Group Description
356
+ */
357
+ public String Prop1 { get; set; }
358
+ public Integer Prop2 { get; set; }
359
+ /** @end-group */
360
+ }
361
+ `;
362
+
363
+ const result = (reflect(classBody)).typeMirror as ClassMirror;
364
+ expect(result.properties.length).toBe(2);
365
+ expect(result.properties[0].typeReference.type).toBe('String');
366
+ expect(result.properties[0].name).toBe('Prop1');
367
+ expect(result.properties[0].group).toBe('Group Name');
368
+ expect(result.properties[0].groupDescription).toBe('Group Description');
369
+ expect(result.properties[1].typeReference.type).toBe('Integer');
370
+ expect(result.properties[1].name).toBe('Prop2');
371
+ expect(result.properties[1].group).toBe('Group Name');
372
+ expect(result.properties[1].groupDescription).toBe('Group Description');
373
+ });
374
+ });