Skip to content

Commit 33d4023

Browse files
korosbaywetrkodev
authored
Validate the serialization/deserialization of composed types in typescript (#1290)
* add unit tests to validate the logic in the serialization/deserialization of composed types * revert formating * add more unit tests * chore: removes extraneuous imports * fix unit test --------- Co-authored-by: Vincent Biret <vibiret@microsoft.com> Co-authored-by: rkodev <43806892+rkodev@users.noreply.github.com>
1 parent 1a69fe1 commit 33d4023

7 files changed

+472
-3
lines changed

packages/serialization/json/test/common/JsonParseNode.ts

+27-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77

88
import { assert, describe, it } from "vitest";
99
import { JsonParseNode } from "../../src/index";
10-
import { createTestParserFromDiscriminatorValue, type TestBackedModel, createTestBackedModelFromDiscriminatorValue, type TestParser } from "./testEntity";
10+
import { createTestParserFromDiscriminatorValue, type TestBackedModel, createTestBackedModelFromDiscriminatorValue, type TestParser, TestUnionObject, BarResponse } from "./testEntity";
1111
import { UntypedTestEntity, createUntypedTestEntityFromDiscriminatorValue } from "./untypedTestEntiy";
1212
import { UntypedNode, UntypedObject, isUntypedArray, isUntypedBoolean, isUntypedNode, isUntypedNumber, isUntypedObject } from "@microsoft/kiota-abstractions";
1313

@@ -318,4 +318,30 @@ describe("JsonParseNode", () => {
318318
const result5 = new JsonParseNode("true");
319319
assert.isUndefined(result5.getBooleanValue());
320320
});
321+
322+
it("should parse a union of objects and primitive values when value is primitive", async () => {
323+
const result = new JsonParseNode({
324+
testUnionObject: "Test String Value",
325+
}).getObjectValue(createTestParserFromDiscriminatorValue) as TestParser;
326+
assert.equal(result.testUnionObject, "Test String Value");
327+
});
328+
329+
it("should parse a union of objects and primitive values when value is an object", async () => {
330+
const barResponse = {
331+
propA: "property A test value",
332+
propB: "property B test value",
333+
propC: undefined,
334+
};
335+
const result = new JsonParseNode({
336+
testUnionObject: barResponse as TestUnionObject,
337+
}).getObjectValue(createTestParserFromDiscriminatorValue) as TestParser;
338+
assert.equal(JSON.stringify(result.testUnionObject), JSON.stringify(barResponse));
339+
});
340+
341+
it("should parse a union of objects and primitive values when value is a number", async () => {
342+
const result = new JsonParseNode({
343+
testUnionObject: 1234,
344+
}).getObjectValue(createTestParserFromDiscriminatorValue) as TestParser;
345+
assert.equal(result.testUnionObject, 1234);
346+
});
321347
});

packages/serialization/json/test/common/jsonSerializationWriter.ts

+47
Original file line numberDiff line numberDiff line change
@@ -264,4 +264,51 @@ describe("JsonParseNode", () => {
264264
const contentAsStr = decoder.decode(serializedContent);
265265
assert.equal(contentAsStr, '{"id":"1","title":"title","location":{"address":{"city":"Redmond","postalCode":"98052","state":"Washington","street":"NE 36th St"},"coordinates":{"latitude":47.678581,"longitude":-122.131577},"displayName":"Microsoft Building 25","floorCount":50,"hasReception":true,"contact":null},"keywords":[{"created":"2023-07-26T10:41:26Z","label":"Keyword1","termGuid":"10e9cc83-b5a4-4c8d-8dab-4ada1252dd70","wssId":6442450941},{"created":"2023-07-26T10:51:26Z","label":"Keyword2","termGuid":"2cae6c6a-9bb8-4a78-afff-81b88e735fef","wssId":6442450942}],"extra":{"value":{"createdDateTime":{"value":"2024-01-15T00:00:00+00:00"}}}}');
266266
});
267+
268+
it("it should serialize a union of object and primitive when the value is a string", async () => {
269+
const inputObject: TestParser = {
270+
testUnionObject: "Test String value",
271+
};
272+
const writer = new JsonSerializationWriter();
273+
writer.writeObjectValue("", inputObject, serializeTestParser);
274+
const serializedContent = writer.getSerializedContent();
275+
const decoder = new TextDecoder();
276+
const contentAsStr = decoder.decode(serializedContent);
277+
const result = JSON.parse(contentAsStr);
278+
assert.isTrue("testUnionObject" in result);
279+
assert.equal(result["testUnionObject"], "Test String value");
280+
});
281+
282+
it("it should serialize a union of object and primitive when the value is a number", async () => {
283+
const inputObject: TestParser = {
284+
testUnionObject: 1234,
285+
};
286+
const writer = new JsonSerializationWriter();
287+
writer.writeObjectValue("", inputObject, serializeTestParser);
288+
const serializedContent = writer.getSerializedContent();
289+
const decoder = new TextDecoder();
290+
const contentAsStr = decoder.decode(serializedContent);
291+
const result = JSON.parse(contentAsStr);
292+
assert.isTrue("testUnionObject" in result);
293+
assert.equal(result["testUnionObject"], 1234);
294+
});
295+
296+
it("it should serialize a union of object and primitive when the value is an object", async () => {
297+
const barResponse = {
298+
propA: "property A test value",
299+
propB: "property B test value",
300+
propC: undefined,
301+
};
302+
const inputObject: TestParser = {
303+
testUnionObject: barResponse,
304+
};
305+
const writer = new JsonSerializationWriter();
306+
writer.writeObjectValue("", inputObject, serializeTestParser);
307+
const serializedContent = writer.getSerializedContent();
308+
const decoder = new TextDecoder();
309+
const contentAsStr = decoder.decode(serializedContent);
310+
const result = JSON.parse(contentAsStr);
311+
assert.isTrue("testUnionObject" in result);
312+
assert.equal(JSON.stringify(result["testUnionObject"]), JSON.stringify(barResponse));
313+
});
267314
});

packages/serialization/json/test/common/testEntity.ts

+30
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ export interface TestParser {
2222
id?: string | null | undefined;
2323
testNumber?: number | null | undefined;
2424
testGuid?: Guid | null | undefined;
25+
testUnionObject?: TestUnionObject | null | undefined;
2526
}
2627
export interface TestBackedModel extends TestParser, BackedModel {
2728
backingStoreEnabled?: boolean | undefined;
@@ -35,6 +36,7 @@ export interface BarResponse extends Parsable {
3536
propB?: string | undefined;
3637
propC?: Date | undefined;
3738
}
39+
export type TestUnionObject = FooResponse | BarResponse | string | number;
3840

3941
export function createTestParserFromDiscriminatorValue(parseNode: ParseNode | undefined) {
4042
if (!parseNode) throw new Error("parseNode cannot be undefined");
@@ -85,6 +87,9 @@ export function deserializeTestParser(testParser: TestParser | undefined = {}):
8587
testGuid: (n) => {
8688
testParser.testGuid = n.getGuidValue();
8789
},
90+
testUnionObject: (n) => {
91+
testParser.testUnionObject = n.getStringValue() ?? n.getNumberValue() ?? n.getObjectValue(createTestUnionObjectFromDiscriminatorValue);
92+
},
8893
};
8994
}
9095

@@ -137,6 +142,13 @@ export function serializeTestParser(writer: SerializationWriter, entity: TestPar
137142
writer.writeObjectValue("testObject", entity.testObject, serializeTestObject);
138143
writer.writeCollectionOfObjectValues("foos", entity.foos, serializeFoo);
139144
writer.writeAdditionalData(entity.additionalData);
145+
if (typeof entity.testUnionObject === "string") {
146+
writer.writeStringValue("testUnionObject", entity.testUnionObject);
147+
} else if (typeof entity.testUnionObject === "number") {
148+
writer.writeNumberValue("testUnionObject", entity.testUnionObject);
149+
} else {
150+
writer.writeObjectValue("testUnionObject", entity.testUnionObject as any, serializeTestUnionObject);
151+
}
140152
}
141153

142154
export function serializeFoo(writer: SerializationWriter, entity: FooResponse | undefined = {}): void {
@@ -153,3 +165,21 @@ export function serializeBar(writer: SerializationWriter, entity: BarResponse |
153165
export function serializeTestBackModel(writer: SerializationWriter, entity: TestBackedModel | undefined = {}): void {
154166
serializeTestParser(writer, entity);
155167
}
168+
169+
// Factory Method
170+
export function createTestUnionObjectFromDiscriminatorValue(parseNode: ParseNode | undefined): (instance?: Parsable) => Record<string, (node: ParseNode) => void> {
171+
return deserializeIntoTestUnionObject;
172+
}
173+
174+
// Deserialization methods
175+
export function deserializeIntoTestUnionObject(fooBar: Partial<TestUnionObject> | undefined = {}): Record<string, (node: ParseNode) => void> {
176+
return {
177+
...deserializeFooParser(fooBar as FooResponse),
178+
...deserializeBarParser(fooBar as BarResponse),
179+
};
180+
}
181+
182+
export function serializeTestUnionObject(writer: SerializationWriter, fooBar: Partial<TestUnionObject> | undefined = {}): void {
183+
serializeFoo(writer, fooBar as FooResponse);
184+
serializeBar(writer, fooBar as BarResponse);
185+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
export function deepEqual(obj1: any, obj2: any): boolean {
2+
if (obj1 === obj2) {
3+
return true;
4+
}
5+
6+
if (obj1 == null || obj2 == null || typeof obj1 !== "object" || typeof obj2 !== "object") {
7+
return false;
8+
}
9+
10+
const keys1 = Object.keys(obj1);
11+
const keys2 = Object.keys(obj2);
12+
13+
if (keys1.length !== keys2.length) {
14+
return false;
15+
}
16+
17+
for (const key of keys1) {
18+
if (!keys2.includes(key) || !deepEqual(obj1[key], obj2[key])) {
19+
return false;
20+
}
21+
}
22+
23+
return true;
24+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,216 @@
1+
import { type AdditionalDataHolder, type BaseRequestBuilder, type Parsable, type ParsableFactory, type ParseNode, type RequestConfiguration, type RequestInformation, type RequestsMetadata, type SerializationWriter } from "@microsoft/kiota-abstractions";
2+
3+
export interface Cat extends Parsable, Pet {
4+
/**
5+
* The favoriteToy property
6+
*/
7+
favoriteToy?: string;
8+
}
9+
/**
10+
* Creates a new instance of the appropriate class based on discriminator value
11+
* @param parseNode The parse node to use to read the discriminator value and create the object
12+
* @returns {Cat}
13+
*/
14+
export function createCatFromDiscriminatorValue(parseNode: ParseNode | undefined): (instance?: Parsable) => Record<string, (node: ParseNode) => void> {
15+
return deserializeIntoCat;
16+
}
17+
/**
18+
* Creates a new instance of the appropriate class based on discriminator value
19+
* @param parseNode The parse node to use to read the discriminator value and create the object
20+
* @returns {Dog}
21+
*/
22+
export function createDogFromDiscriminatorValue(parseNode: ParseNode | undefined): (instance?: Parsable) => Record<string, (node: ParseNode) => void> {
23+
return deserializeIntoDog;
24+
}
25+
/**
26+
* Creates a new instance of the appropriate class based on discriminator value
27+
* @param parseNode The parse node to use to read the discriminator value and create the object
28+
* @returns {Pet}
29+
*/
30+
export function createPetFromDiscriminatorValue(parseNode: ParseNode | undefined): (instance?: Parsable) => Record<string, (node: ParseNode) => void> {
31+
return deserializeIntoPet;
32+
}
33+
/**
34+
* The deserialization information for the current model
35+
* @returns {Record<string, (node: ParseNode) => void>}
36+
*/
37+
export function deserializeIntoCat(cat: Partial<Cat> | undefined = {}): Record<string, (node: ParseNode) => void> {
38+
return {
39+
...deserializeIntoPet(cat),
40+
favoriteToy: (n) => {
41+
cat.favoriteToy = n.getStringValue();
42+
},
43+
};
44+
}
45+
/**
46+
* The deserialization information for the current model
47+
* @returns {Record<string, (node: ParseNode) => void>}
48+
*/
49+
export function deserializeIntoDog(dog: Partial<Dog> | undefined = {}): Record<string, (node: ParseNode) => void> {
50+
return {
51+
...deserializeIntoPet(dog),
52+
breed: (n) => {
53+
dog.breed = n.getStringValue();
54+
},
55+
};
56+
}
57+
/**
58+
* The deserialization information for the current model
59+
* @returns {Record<string, (node: ParseNode) => void>}
60+
*/
61+
export function deserializeIntoPet(pet: Partial<Pet> | undefined = {}): Record<string, (node: ParseNode) => void> {
62+
return {
63+
age: (n) => {
64+
pet.age = n.getNumberValue();
65+
},
66+
name: (n) => {
67+
pet.name = n.getStringValue();
68+
},
69+
};
70+
}
71+
export interface Dog extends Parsable, Pet {
72+
/**
73+
* The breed property
74+
*/
75+
breed?: string;
76+
}
77+
export interface Pet extends AdditionalDataHolder, Parsable {
78+
/**
79+
* Stores additional data not described in the OpenAPI description found when deserializing. Can be used for serialization as well.
80+
*/
81+
additionalData?: Record<string, unknown>;
82+
/**
83+
* The age property
84+
*/
85+
age?: number;
86+
/**
87+
* The name property
88+
*/
89+
name?: string;
90+
}
91+
/**
92+
* Serializes information the current object
93+
* @param writer Serialization writer to use to serialize this model
94+
*/
95+
export function serializeCat(writer: SerializationWriter, cat: Partial<Cat> | undefined = {}): void {
96+
serializePet(writer, cat);
97+
writer.writeStringValue("favoriteToy", cat.favoriteToy);
98+
}
99+
/**
100+
* Serializes information the current object
101+
* @param writer Serialization writer to use to serialize this model
102+
*/
103+
export function serializeDog(writer: SerializationWriter, dog: Partial<Dog> | undefined = {}): void {
104+
serializePet(writer, dog);
105+
writer.writeStringValue("breed", dog.breed);
106+
}
107+
/**
108+
* Serializes information the current object
109+
* @param writer Serialization writer to use to serialize this model
110+
*/
111+
export function serializePet(writer: SerializationWriter, pet: Partial<Pet> | undefined = {}): void {
112+
writer.writeNumberValue("age", pet.age);
113+
writer.writeStringValue("name", pet.name);
114+
writer.writeAdditionalData(pet.additionalData);
115+
}
116+
/* tslint:enable */
117+
/* eslint-enable */
118+
/**
119+
* Creates a new instance of the appropriate class based on discriminator value
120+
* @param parseNode The parse node to use to read the discriminator value and create the object
121+
* @returns {Cat | Dog}
122+
*/
123+
export function createPetGetResponse_dataFromDiscriminatorValue(parseNode: ParseNode | undefined): (instance?: Parsable) => Record<string, (node: ParseNode) => void> {
124+
return deserializeIntoPetGetResponse_data;
125+
}
126+
/**
127+
* Creates a new instance of the appropriate class based on discriminator value
128+
* @param parseNode The parse node to use to read the discriminator value and create the object
129+
* @returns {PetGetResponse}
130+
*/
131+
export function createPetGetResponseFromDiscriminatorValue(parseNode: ParseNode | undefined): (instance?: Parsable) => Record<string, (node: ParseNode) => void> {
132+
return deserializeIntoPetGetResponse;
133+
}
134+
/**
135+
* The deserialization information for the current model
136+
* @returns {Record<string, (node: ParseNode) => void>}
137+
*/
138+
export function deserializeIntoPetGetResponse(petGetResponse: Partial<PetGetResponse> | undefined = {}): Record<string, (node: ParseNode) => void> {
139+
return {
140+
data: (n) => {
141+
petGetResponse.data = n.getNumberValue() ?? n.getStringValue() ?? n.getObjectValue<Cat | Dog>(createPetGetResponse_dataFromDiscriminatorValue);
142+
},
143+
request_id: (n) => {
144+
petGetResponse.request_id = n.getStringValue();
145+
},
146+
};
147+
}
148+
/**
149+
* The deserialization information for the current model
150+
* @returns {Record<string, (node: ParseNode) => void>}
151+
*/
152+
export function deserializeIntoPetGetResponse_data(petGetResponse_data: Partial<Cat | Dog> | undefined = {}): Record<string, (node: ParseNode) => void> {
153+
return {
154+
...deserializeIntoCat(petGetResponse_data as Cat),
155+
...deserializeIntoDog(petGetResponse_data as Dog),
156+
};
157+
}
158+
export interface PetGetResponse extends AdditionalDataHolder, Parsable {
159+
/**
160+
* Stores additional data not described in the OpenAPI description found when deserializing. Can be used for serialization as well.
161+
*/
162+
additionalData?: Record<string, unknown>;
163+
/**
164+
* The data property
165+
*/
166+
data?: Cat | Dog | number | string;
167+
/**
168+
* The request_id property
169+
*/
170+
request_id?: string;
171+
}
172+
export type PetGetResponse_data = Cat | Dog | number | string;
173+
/**
174+
* Builds and executes requests for operations under /pet
175+
*/
176+
export interface PetRequestBuilder extends BaseRequestBuilder<PetRequestBuilder> {
177+
/**
178+
* Get pet information
179+
* @param requestConfiguration Configuration for the request such as headers, query parameters, and middleware options.
180+
* @returns {Promise<PetGetResponse>}
181+
*/
182+
get(requestConfiguration?: RequestConfiguration<object> | undefined): Promise<PetGetResponse | undefined>;
183+
/**
184+
* Get pet information
185+
* @param requestConfiguration Configuration for the request such as headers, query parameters, and middleware options.
186+
* @returns {RequestInformation}
187+
*/
188+
toGetRequestInformation(requestConfiguration?: RequestConfiguration<object> | undefined): RequestInformation;
189+
}
190+
/**
191+
* Serializes information the current object
192+
* @param writer Serialization writer to use to serialize this model
193+
*/
194+
export function serializePetGetResponse(writer: SerializationWriter, petGetResponse: Partial<PetGetResponse> | undefined = {}): void {
195+
switch (true) {
196+
case typeof petGetResponse.data === "number":
197+
writer.writeNumberValue("data", petGetResponse.data);
198+
break;
199+
case typeof petGetResponse.data === "string":
200+
writer.writeStringValue("data", petGetResponse.data);
201+
break;
202+
default:
203+
writer.writeObjectValue<Cat | Dog>("data", petGetResponse.data, serializePetGetResponse_data);
204+
break;
205+
}
206+
writer.writeStringValue("request_id", petGetResponse.request_id);
207+
writer.writeAdditionalData(petGetResponse.additionalData);
208+
}
209+
/**
210+
* Serializes information the current object
211+
* @param writer Serialization writer to use to serialize this model
212+
*/
213+
export function serializePetGetResponse_data(writer: SerializationWriter, petGetResponse_data: Partial<Cat | Dog> | undefined = {}): void {
214+
serializeCat(writer, petGetResponse_data as Cat);
215+
serializeDog(writer, petGetResponse_data as Dog);
216+
}

0 commit comments

Comments
 (0)