Custom Response Models
Introduction
Section titled âIntroductionâIn this section, weâll focus on creating custom response models and demonstrate how to use them in your API operations. Weâll also incorporate predefined response models from the TypeSpec HTTP library.
Introduction to Custom Response Models
Section titled âIntroduction to Custom Response ModelsâCustom response models allow you to define structured responses for your API operations. They help ensure consistency and clarity in your API responses. TypeSpec defines response models for common HTTP responses in the HTTP library, which we can incorporate into our custom response models.
Common HTTP Status Codes and TypeSpec Response Models
Section titled âCommon HTTP Status Codes and TypeSpec Response ModelsâHere are some common HTTP status codes and their equivalent TypeSpec response models from the TypeSpec HTTP library:
| HTTP Status Code | Meaning | TypeSpec Response Model | 
|---|---|---|
| 200 OK | The request was successful, and the server returned the requested resource. | OkResponse | 
| 201 Created | The request was successful, and a new resource was created. | CreatedResponse | 
| 204 No Content | The request was successful, but there is no content to return. | NoContentResponse | 
| 400 Bad Request | The server could not understand the request due to invalid syntax. | BadRequestResponse | 
| 401 Unauthorized | The client must authenticate itself to get the requested response. | UnauthorizedResponse | 
| 403 Forbidden | The client does not have access rights to the content. | ForbiddenResponse | 
| 404 Not Found | The server cannot find the requested resource. | NotFoundResponse | 
Benefits of Using Custom Response Models
Section titled âBenefits of Using Custom Response Modelsâ- Reducing Duplication: By defining common response structures once, you can reuse them across multiple operations.
- Improving Readability: Custom response models make your API definitions clearer and easier to understand.
- Minimizing Errors: Consistent response models help reduce the likelihood of errors in your API responses.
Creating Custom Response Models
Section titled âCreating Custom Response ModelsâLetâs start by defining and extending some custom response models. These models will incorporate existing response models from the TypeSpec HTTP library to ensure consistency.
Example: Defining and Extending Custom Response Models
Section titled âExample: Defining and Extending Custom Response Modelsâmodel PetListResponse {  ...OkResponse;  ...Body<Pet[]>;}
model PetResponse {  ...OkResponse;  ...Body<Pet>;}
model PetCreatedResponse {  ...CreatedResponse;  ...Body<Pet>;}
model PetErrorResponse {  ...BadRequestResponse;  ...Body<ValidationError>;}
model PetNotFoundResponse {  ...NotFoundResponse;  ...Body<NotFoundError>;}
model PetUnauthorizedResponse {  ...UnauthorizedResponse;  ...Body<UnauthorizedError>;}
model PetSuccessResponse {  ...OkResponse;  ...Body<string>;}
model PetNoContentResponse {  ...NoContentResponse;}In this example:
- PetListResponseextends- OkResponseand includes a body with an array of- Petobjects.
- PetResponseextends- OkResponseand includes a body with a single- Petobject.
- PetCreatedResponseextends- CreatedResponseand includes a body with a newly created- Petobject.
- PetErrorResponseextends- BadRequestResponseand includes a body with a- ValidationErrorobject.
- PetNotFoundResponseextends- NotFoundResponseand includes a body with a- NotFoundErrorobject.
- PetUnauthorizedResponseextends- UnauthorizedResponseand includes a body with an- UnauthorizedErrorobject.
- PetSuccessResponseextends- OkResponseand includes a body with a success message.
- PetNoContentResponseextends- NoContentResponsefor situations where the request succeeded but there is no content to return.
Note: Base response models like OkResponse, CreatedResponse, BadRequestResponse, NotFoundResponse, and UnauthorizedResponse are imported from the TypeSpec HTTP data types library, which weâre importing in our project as @typespec/http.
Using Custom Response Models in Operations
Section titled âUsing Custom Response Models in OperationsâNow that we have defined our custom response models, letâs use them in our API operations.
Example: Applying Custom Response Models to Operations
Section titled âExample: Applying Custom Response Models to Operationsâimport "@typespec/http";import "@typespec/versioning";
using Http;using Versioning;
@service(#{ title: "Pet Store" })@server("https://example.com", "Single server endpoint")@versioned(Versions)namespace PetStore;
enum Versions {  v1: "1.0",  v2: "2.0",}
model Pet {  id: int32;
  @minLength(1)  name: string;
  @minValue(0)  @maxValue(100)  age: int32;
  kind: petType;}
enum petType {  dog: "dog",  cat: "cat",  fish: "fish",  bird: "bird",  reptile: "reptile",}
@added(Versions.v2)model Toy {  id: int32;  name: string;}
model CommonParameters {  @header  requestID: string;
  @query  locale?: string;
  @header  clientVersion?: string;}
// highlight-startmodel PetListResponse {  ...OkResponse;  ...Body<Pet[]>;}
model PetResponse {  ...OkResponse;  ...Body<Pet>;}
model PetCreatedResponse {  ...CreatedResponse;  ...Body<Pet>;}
model PetAcceptedResponse {  ...AcceptedResponse;  ...Body<Pet>;}
model PetErrorResponse {  ...BadRequestResponse;  ...Body<ValidationError>;}
model PetNotFoundResponse {  ...NotFoundResponse;  ...Body<NotFoundError>;}
model PetUnauthorizedResponse {  ...UnauthorizedResponse;  ...Body<UnauthorizedError>;}
model PetSuccessResponse {  ...OkResponse;  ...Body<string>;}
model PetNoContentResponse {  ...NoContentResponse;}// highlight-end
@route("/pets")namespace Pets {  @get  // highlight-next-line  op listPets(...CommonParameters): PetListResponse;
  @get  // highlight-start  op getPet(@path petId: int32, @header ifMatch?: string): PetResponse | PetNotFoundResponse;  // highlight-end  @useAuth(BearerAuth)  @post  // highlight-start  op createPet(@body pet: Pet):    | PetCreatedResponse    | PetAcceptedResponse    | PetErrorResponse    | PetUnauthorizedResponse;  // highlight-end
  @useAuth(BearerAuth)  @put  // highlight-start  op updatePet(@path petId: int32, @body pet: Pet):    | PetResponse    | PetErrorResponse    | PetUnauthorizedResponse    | PetNotFoundResponse    | InternalServerErrorResponse;  // highlight-end
  @useAuth(BearerAuth)  @delete  // highlight-start  op deletePet(@path petId: int32): PetNoContentResponse | PetUnauthorizedResponse;  // highlight-end
  @route("{petId}/toys")  namespace Toys {    @added(Versions.v2)    @get    op listToys(@path petId: int32, ...CommonParameters): {      @body toys: Toy[];    };
    @added(Versions.v2)    @post    @useAuth(BearerAuth)    op createToy(@path petId: int32, @body toy: Toy, ...CommonParameters): {      @statusCode statusCode: 201;      @body newToy: Toy;    };
    @added(Versions.v2)    @put    @useAuth(BearerAuth)    op updateToy(@path petId: int32, @path toyId: int32, @body toy: Toy, ...CommonParameters): {      @body updatedToy: Toy;    };
    @added(Versions.v2)    @delete    @useAuth(BearerAuth)    op deleteToy(@path petId: int32, @path toyId: int32, ...CommonParameters): {      @statusCode statusCode: 204;    };  }}
@errormodel NotFoundError {  code: "NOT_FOUND";  message: string;}
@errormodel ValidationError {  code: "VALIDATION_ERROR";  message: string;  details: string[];}
@errormodel UnauthorizedError {  code: "UNAUTHORIZED";  message: string;}
@errormodel InternalServerError {  code: "INTERNAL_SERVER_ERROR";  message: string;}
model InternalServerErrorResponse {  @statusCode statusCode: 500;  @body error: InternalServerError;}In this example:
- The listPetsoperation uses thePetListResponsecustom response model.
- The getPetoperation uses thePetResponseandPetNotFoundResponsecustom response models.
- The createPetoperation uses thePetCreatedResponse,PetAcceptedResponse,PetErrorResponse, andPetUnauthorizedResponsecustom response models.
- The updatePetoperation uses thePetResponse,PetErrorResponse,PetUnauthorizedResponse,PetNotFoundResponse, andInternalServerErrorResponsecustom response models.
- The deletePetoperation uses thePetNoContentResponseandPetUnauthorizedResponsecustom response models.
Note that we could also define custom response models for the Toys operations, similar to the Pets operations. But for brevity, weâre omitting them in this example.
Conclusion
Section titled âConclusionâIn this section, we focused on creating custom response models in your REST API. By defining and extending custom response models, we can reduce duplication, improve readability, and minimize errors in our API responses. We also incorporated existing response models from the TypeSpec HTTP library to ensure consistency and clarity.