|
| 1 | +# Generated Client Design |
| 2 | + |
| 3 | +## Introduction |
| 4 | + |
| 5 | +This document tracks design rules and decisions for .NET clients generated by the DPG generator. It may later be extended to include other generator designs. |
| 6 | + |
| 7 | +<!-- |
| 8 | +## Clients |
| 9 | +
|
| 10 | +TBD |
| 11 | +
|
| 12 | +## Client Protocol Operations |
| 13 | +
|
| 14 | +TBD |
| 15 | +
|
| 16 | +## Client Convenience Operations |
| 17 | +--> |
| 18 | + |
| 19 | +## Model Generation |
| 20 | + |
| 21 | +Models are defined as partial classes with one or more constructors, properties, and methods that read from/write to wire formats. |
| 22 | + |
| 23 | +##### Discussion Notes |
| 24 | +The .NET team discourages using C# Records in public APIs, so we will not use them for model types. |
| 25 | + |
| 26 | +### Model Namespace |
| 27 | + |
| 28 | +Generated models will be defined in a `.Models` namespace. |
| 29 | +##### Discussion Notes |
| 30 | + |
| 31 | +Tracking finalization of namespace decision [autorest.csharp #2514](https://github.com/Azure/autorest.csharp/issues/2514) |
| 32 | + |
| 33 | +### Type mappings from Cadl to .NET |
| 34 | + |
| 35 | +The following table shows a mapping from Cadl built-in types to the corresponding types used in .NET models. |
| 36 | + |
| 37 | +Cadl Type | .NET Type | OpenAPI Type | GitHub Issue | Notes |
| 38 | +------------------- | -------- | -- | -- | ------------- |
| 39 | +string | string | string | [autorest.csharp #2337](https://github.com/Azure/autorest.csharp/issues/2337) | [PrimitivePropertyModel.cs](https://github.com/annelo-msft/azure-sdk-for-net/blob/cadl-model-primitive-properties/sdk/template/Azure.Template/src/Generated/Models/PrimitivePropertyModel.cs) [PrimitivePropertyModel.Serialization.cs](https://github.com/annelo-msft/azure-sdk-for-net/blob/cadl-model-primitive-properties/sdk/template/Azure.Template/src/Generated/Models/PrimitivePropertyModel.Serialization.cs) |
| 40 | +bytes | BinaryData | type: string, format: byte | [autorest.csharp #2337](https://github.com/Azure/autorest.csharp/issues/2337) | |
| 41 | +int32 | int | type: integer, format: int32 | [autorest.csharp #2337](https://github.com/Azure/autorest.csharp/issues/2337) | |
| 42 | +int64 | long | type: integer, format: int64 | [autorest.csharp #2337](https://github.com/Azure/autorest.csharp/issues/2337) | |
| 43 | +safeint | long | n/a | [autorest.csharp #2337](https://github.com/Azure/autorest.csharp/issues/2337) | |
| 44 | +float32 | float | type: number, format: float | [autorest.csharp #2337](https://github.com/Azure/autorest.csharp/issues/2337) | |
| 45 | +float64 | double | type: number, format: double | [autorest.csharp #2337](https://github.com/Azure/autorest.csharp/issues/2337) | |
| 46 | +zonedDateTime | DateTimeOffset | type: string, format: date-time | [autorest.csharp #2337](https://github.com/Azure/autorest.csharp/issues/2337) | Serialized differently based on body/header |
| 47 | +duration | TimeSpan | type: string, format: duration | [autorest.csharp #2337](https://github.com/Azure/autorest.csharp/issues/2337) | |
| 48 | +boolean | bool | boolean | [autorest.csharp #2337](https://github.com/Azure/autorest.csharp/issues/2337) | |
| 49 | +<!-- TBD | TBD | type: string, format: binary | [autorest.csharp #2500](https://github.com/Azure/autorest.csharp/issues/2500)| --> |
| 50 | + |
| 51 | +### Model Shape |
| 52 | + |
| 53 | +Model APIs will differ depending on whether the model is an input model, an output model, or a round-trip model (both input and output). |
| 54 | + |
| 55 | +The following describes the model shapes for different cases of properties. |
| 56 | + |
| 57 | + Item | Input | Output | Round-Trip | GitHub Issue |
| 58 | +-- | -------| ------------ | -------- | -- | |
| 59 | +Constructor Accessibility | public | internal | public | |
| 60 | +Required Property | get-only | get-only | get/set | [autorest.csharp #2463](https://github.com/Azure/autorest.csharp/issues/2463) |
| 61 | +Optional Property | get/set | get-only | get/set | [autorest.csharp #2339](https://github.com/Azure/autorest.csharp/issues/2339) |
| 62 | +Collection Property | `IList<T>` get-only | `IReadOnlyList<T>` get-only | `IList<T>` get-only | [autorest.csharp #2471](https://github.com/Azure/autorest.csharp/issues/2471) |
| 63 | + |
| 64 | +### Model Constructors |
| 65 | + |
| 66 | +Models have one or more constructors as follows. |
| 67 | + |
| 68 | +#### Main Constructor |
| 69 | + |
| 70 | +- Accessibility is specified in model shape table above |
| 71 | +- Takes required parameters and does not take optional parameters |
| 72 | +- Validates required reference type parameters for null using `Argument.AssertNotNull` |
| 73 | +- Takes list properties as `IEnumerable<T>` parameters |
| 74 | +- Initializes lists from `IEnumerable` parameter using System.Linq `.ToList()` |
| 75 | + |
| 76 | +#### Serialization Constructor |
| 77 | + |
| 78 | +- Internal accessibility |
| 79 | +- Generated for Output and Round-trip types |
| 80 | +- Takes required and optional parameters |
| 81 | +- Takes list properties as `IReadOnlyList<T>` parameters for Output models and `IList<T>` parameters for Round-trip models |
| 82 | +- List properties are initialized by assignment |
| 83 | + |
| 84 | +See examples: |
| 85 | + |
| 86 | +- Reference and value type properties: |
| 87 | + - [RoundTripModel.cs](https://github.com/annelo-msft/azure-sdk-for-net/blob/cadl-models-roundtrip-basic/sdk/template/Azure.Template/src/Generated/Models/RoundTripModel.cs) |
| 88 | + - [RoundTripModel.Serialization.cs](https://github.com/annelo-msft/azure-sdk-for-net/blob/cadl-models-roundtrip-basic/sdk/template/Azure.Template/src/Generated/Models/RoundTripModel.Serialization.cs) |
| 89 | +- Collection properties: |
| 90 | + - [RoundTripModel.cs](https://github.com/annelo-msft/azure-sdk-for-net/blob/cadl-models-collections-basic/sdk/template/Azure.Template/src/Generated/Models/RoundTripModel.cs) |
| 91 | + - Collection properties [RoundTripModel.Serialization.cs](https://github.com/annelo-msft/azure-sdk-for-net/blob/cadl-models-collections-basic/sdk/template/Azure.Template/src/Generated/Models/RoundTripModel.Serialization.cs) |
| 92 | + |
| 93 | +### Collections |
| 94 | + |
| 95 | +There are a number of outstanding open issues regarding how to represent collections being tracked: |
| 96 | + |
| 97 | +- [autorest.csharp #2515](https://github.com/Azure/autorest.csharp/issues/2515) |
| 98 | +- [autorest.csharp #2513](https://github.com/Azure/autorest.csharp/issues/2513) |
| 99 | + |
| 100 | +### Enums |
| 101 | + |
| 102 | +Enums may be generated as a CLR enum or a C# "Strongly-typed string" as follows. |
| 103 | + |
| 104 | +- Enums are generated in the same namespace as models. |
| 105 | +- A Cadl `enum` not found in a corresponding `@knownValues` decorator is generated as a C# `enum` |
| 106 | +- Generated enum has corresponding internal extensions file providing conversions to/from string. |
| 107 | +- A Cadl string model with a `@knownValues` decorator is generated as a C# "Strongly-typed string" (sometimes called "extensible enum") and the corresponding Cadl `enum` is not generated as a C# `enum` |
| 108 | + |
| 109 | +GitHub issue: [autorest.csharp #2477](https://github.com/Azure/autorest.csharp/issues/2477) |
| 110 | + |
| 111 | +See examples: |
| 112 | + |
| 113 | +- CLR enum: |
| 114 | + - [DayOfTheWeek.cs](https://github.com/annelo-msft/azure-sdk-for-net/blob/cadl-models-enum-properties/sdk/template/Azure.Template/src/Generated/Models/DayOfTheWeek.cs) |
| 115 | + - [DayOfTheWeek.Serialization.cs](https://github.com/annelo-msft/azure-sdk-for-net/blob/cadl-models-enum-properties/sdk/template/Azure.Template/src/Generated/Models/DayOfTheWeek.Serialization.cs) |
| 116 | +- Stronly typed string (extensible enum): |
| 117 | + - [TranslationLanguage.cs](https://github.com/annelo-msft/azure-sdk-for-net/blob/cadl-models-enum-properties/sdk/template/Azure.Template/src/Generated/Models/TranslationLanguage.cs) |
| 118 | + - No corresponding .Serialization.cs file. |
| 119 | + |
| 120 | +### Nested Models |
| 121 | + |
| 122 | +Models that are properties of other models are called nested models. A model that appears only as a non-root of a model graph is defined as partial class according to the same rules as root models. |
| 123 | + |
| 124 | +GitHub issue: [autorest.csharp #2489](https://github.com/Azure/autorest.csharp/issues/2489) |
| 125 | + |
| 126 | +See examples: |
| 127 | + |
| 128 | +- [RoundTripModel.cs](https://github.com/annelo-msft/azure-sdk-for-net/blob/cadl-models-nested-models/sdk/template/Azure.Template/src/Generated/Models/RoundTripModel.cs) |
| 129 | +- [RoundTripModel.Serialization.cs](https://github.com/annelo-msft/azure-sdk-for-net/blob/cadl-models-nested-models/sdk/template/Azure.Template/src/Generated/Models/RoundTripModel.Serialization.cs) |
| 130 | + |
| 131 | +Requirements for model graphs with circular dependencies will be defined in a later iteration. GitHub issue [autorest.csharp #2506](https://github.com/Azure/autorest.csharp/issues/2506) |
| 132 | + |
| 133 | +#### Nested Model Shape |
| 134 | + |
| 135 | +Models that appear as properties of other models have their shape determined as follows: |
| 136 | + |
| 137 | +- Models that only appear as properties of input models are input models |
| 138 | +- Models that only appear as properties of output models are output models |
| 139 | +- Models that only appear as properties of round-trip models are round-trip models |
| 140 | +- Models that appear as properties of more than one model shape are round-trip models |
| 141 | + |
| 142 | +### Model Reference Documentation |
| 143 | + |
| 144 | +### Miscellaneous |
| 145 | + |
| 146 | +- Models should be generated with the `#nullable disable` because Azure SDK libraries have not yet exposed nullable reference types outside of Azure.Core. The reason we have not done this is because the API compatibility tools we use cannot detect API breaking changes to nullable reference types. |
0 commit comments