--- title: Create a Web API app on Mac or Linux | Microsoft Docs author: rick-anderson description: Build a web API with ASP.NET Core MVC and Visual Studio Code keywords: ASP.NET Core, WebAPI, Web API, REST, Mac, Linux ms.author: riande manager: wpickett ms.date: 03/14/2017 ms.topic: article ms.assetid: 830b4bf5-dd14-423e-9f59-764a6f13a8f6 ms.technology: aspnet ms.prod: asp.net-core uid: tutorials/web-api-vsc --- # Create a Web API app on Mac or Linux with Visual Studio Code By [Mike Wasson](https://github.com/mikewasson) and [Rick Anderson](https://twitter.com/RickAndMSFT) In this tutorial, you’ll build a simple web API for managing a list of "to-do" items. You won’t build any UI in this tutorial. ASP.NET Core has built-in support for MVC building Web APIs. ## Overview Here is the API that you’ll create: |API | Description | Request body | Response body | |--- | ---- | ---- | ---- | |GET /api/todo | Get all to-do items | None | Array of to-do items| |GET /api/todo/{id} | Get an item by ID | None | To-do item| |POST /api/todo | Add a new item | To-do item | To-do item | |PUT /api/todo/{id} | Update an existing item   | To-do item | None | |DELETE /api/todo/{id}     | Delete an item     | None | None|
The following diagram shows the basic design of the app. ![The client is represented by a box on the left and submits a request and receives a response from the application, a box drawn on the right. Within the application box, three boxes represent the controller, the model, and the data access layer. The request comes into the application's controller, and read/write operations occur between the controller and the data access layer. The model is serialized and returned to the client in the response.](first-web-api/_static/architecture.png) * The client is whatever consumes the web API (browser, mobile app, and so forth). We aren’t writing a client in this tutorial. We'll use [Postman](https://www.getpostman.com/) to test the app. * A *model* is an object that represents the data in your application. In this case, the only model is a to-do item. Models are represented as simple C# classes (POCOs). * A *controller* is an object that handles HTTP requests and creates the HTTP response. This app will have a single controller. * To keep the tutorial simple, the app doesn’t use a persistent database. Instead, it stores to-do items in an in-memory database. ## Set up your development environment Download and install [.NET Core](https://microsoft.com/net/core) and [Visual Studio Code](https://code.visualstudio.com) with the [C# extension](https://marketplace.visualstudio.com/items?itemName=ms-vscode.csharp). ## Create the project From a console, run the following commands: ```console mkdir TodoApi cd TodoApi dotnet new webapi ``` Open the *TodoApi* folder in Visual Studio Code (VS Code) and select the *Startup.cs* file. - Select **Yes** to the **Warn** message "Required assets to build and debug are missing from 'TodoApi'. Add them?" - Select **Restore** to the **Info** message "There are unresolved dependencies". ![VS Code with Warn Required assets to build and debug are missing from 'TodoApi'. Add them? Don't ask Again, Not Now, Yes and also Info - there are unresolved dependencies - Restore - Close](web-api-vsc/_static/vsc_restore.png) Press **Debug** (F5) to build and run the program. In a browser navigate to http://localhost:5000/api/values . The following is displayed: `["value1","value2"]` ## Add support for Entity Framework Core Edit the *TodoApi.csproj* file to install the [Entity Framework Core InMemory](https://docs.microsoft.com/en-us/ef/core/providers/in-memory/) database provider. This database provider allows Entity Framework Core to be used with an in-memory database. [!code-xml[Main](web-api-vsc/sample/TodoApi/TodoApi.csproj?highlight=12)] Run `dotnet restore` to download and install the EF Core InMemory DB provider. You can run `dotnet restore` from the terminal or enter `⌘⇧P` (macOS) or `Ctrl+Shift+P` (Linux) in VS Code and then type **.NET**. Select **.NET: Restore Packages**. ## Add a model class A model is an object that represents the data in your application. In this case, the only model is a to-do item. Add a folder named *Models*. You can put model classes anywhere in your project, but the *Models* folder is used by convention. Add a `TodoItem` class with the following code: [!code-csharp[Main](first-web-api/sample/TodoApi/Models/TodoItem.cs)] * The `[Key]` data annotation denotes the property, `Key`, is a unique identifier. * `[DatabaseGenerated` specifies the database will generate the key (rather than the application). * `DatabaseGeneratedOption.Identity` specifies the database should generate integer keys when a row is inserted. ## Create the database context The *database context* is the main class that coordinates Entity Framework functionality for a given data model. You create this class by deriving from the `Microsoft.EntityFrameworkCore.DbContext` class. Add a `TodoContext` class in the *Models* folder: [!code-csharp[Main](first-web-api/sample/TodoApi/Models/TodoContext.cs)] ## Add a repository class A *repository* is an object that encapsulates the data layer. The *repository* contains logic for retrieving and mapping data to an entity model. Create the repository code in the *Models* folder. Defining a repository interface named `ITodoRepository`: [!code-csharp[Main](first-web-api/sample/TodoApi/Models/ITodoRepository.cs)] This interface defines basic CRUD operations. Add a `TodoRepository` class that implements `ITodoRepository`: [!code-csharp[Main](first-web-api/sample/TodoApi/Models/TodoRepository.cs)] Press **Debug** (F5) to build the app to verify you don't have any compiler errors. ## Register the repository and EF in-memory database By defining a repository interface, we can decouple the repository class from the MVC controller that uses it. Instead of instantiating a `TodoRepository` inside the controller we will inject an `ITodoRepository` using the built-in support in ASP.NET Core for [dependency injection](xref:fundamentals/dependency-injection). This approach makes it easier to unit test your controllers. Unit tests should inject a mock or stub version of `ITodoRepository`. That way, the test narrowly targets the controller logic and not the data access layer. In order to inject the repository into the controller, we need to register it with the DI container. Open the *Startup.cs* file. The code below also registers the in-memory database. In the `ConfigureServices` method, add the highlighted code: [!code-csharp[Main](first-web-api/sample/TodoApi/Startup.cs?name=snippet_AddSingleton&highlight=1,2,5,9)] ## Add a controller In the *Controllers* folder, create a class named `TodoController`. Add the following (and add closing braces): [!code-csharp[Main](first-web-api/sample/TodoApi/Controllers/TodoController.cs?name=snippet_todo1)] This defines an empty controller class. In the next sections, we'll add methods to implement the API. ## Getting to-do items To get to-do items, add the following methods to the `TodoController` class. [!code-csharp[Main](first-web-api/sample/TodoApi/Controllers/TodoController.cs?name=snippet_GetAll)] These methods implement the two GET methods: * `GET /api/todo` * `GET /api/todo/{id}` Here is an example HTTP response for the `GetAll` method: ``` HTTP/1.1 200 OK Content-Type: application/json; charset=utf-8 Server: Microsoft-IIS/10.0 Date: Thu, 18 Jun 2015 20:51:10 GMT Content-Length: 82 [{"Key":"4f67d7c5-a2a9-4aae-b030-16003dd829ae","Name":"Item1","IsComplete":false}] ``` Later in the tutorial I'll show how you can view the HTTP response using [Postman](https://www.getpostman.com/). ### Routing and URL paths The `[HttpGet]` attribute specifies an HTTP GET method. The URL path for each method is constructed as follows: * Take the template string in the controller’s route attribute, `[Route("api/[controller]")]` * Replace "[Controller]" with the name of the controller, which is the controller class name minus the "Controller" suffix. For this sample, the controller class name is **Todo**Controller and the root name is "todo". ASP.NET Core [routing](xref:mvc/controllers/routing) is not case sensitive. * If the `[HttpGet]` attribute has a route template (such as `[HttpGet("/products")]`, append that to the path. This sample doesn't use a template. See [Attribute routing with Http[Verb] attributes](https://review.docs.microsoft.com/en-us/aspnet/core/mvc/controllers/routing#attribute-routing-with-httpverb-attributes) for more information. In the `GetById` method: ```csharp [HttpGet("{id}", Name = "GetTodo")] public IActionResult GetById(string id) ``` `"{id}"` is a placeholder variable for the ID of the `todo` item. When `GetById` is invoked, it assigns the value of "{id}" in the URL to the method's `id` parameter. `Name = "GetTodo"` creates a named route and allows you to link to this route in an HTTP Response. I'll explain it with an example later. See [Routing to Controller Actions](xref:mvc/controllers/routing) for detailed information. ### Return values The `GetAll` method returns an `IEnumerable`. MVC automatically serializes the object to [JSON](http://www.json.org/) and writes the JSON into the body of the response message. The response code for this method is 200, assuming there are no unhandled exceptions. (Unhandled exceptions are translated into 5xx errors.) In contrast, the `GetById` method returns the more general `IActionResult` type, which represents a wide range of return types. `GetById` has two different return types: * If no item matches the requested ID, the method returns a 404 error. This is done by returning `NotFound`. * Otherwise, the method returns 200 with a JSON response body. This is done by returning an `ObjectResult` ### Launch the app In VS Code, press F5 to launch the app. Navigate to http://localhost:5000/api/todo (The `Todo` controller we just created). ## Implement the other CRUD operations We'll add `Create`, `Update`, and `Delete` methods to the controller. These are variations on a theme, so I'll just show the code and highlight the main differences. Build the project after adding or changing code. ### Create [!code-csharp[Main](first-web-api/sample/TodoApi/Controllers/TodoController.cs?name=snippet_Create)] This is an HTTP POST method, indicated by the [`[HttpPost]`](https://docs.asp.net/projects/api/en/latest/autoapi/Microsoft/AspNetCore/Mvc/HttpPostAttribute/index.html) attribute. The [`[FromBody]`](https://docs.asp.net/projects/api/en/latest/autoapi/Microsoft/AspNetCore/Mvc/FromBodyAttribute/index.html) attribute tells MVC to get the value of the to-do item from the body of the HTTP request. The `CreatedAtRoute` method returns a 201 response, which is the standard response for an HTTP POST method that creates a new resource on the server. `CreatedAtRoute` also adds a Location header to the response. The Location header specifies the URI of the newly created to-do item. See [10.2.2 201 Created](http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html). ### Use Postman to send a Create request ![Postman console](first-web-api/_static/pmc.png) * Set the HTTP method to `POST` * Select the **Body** radio button * Select the **raw** radio button * Set the type to JSON * In the key-value editor, enter a Todo item such as ```JSON { "name":"walk dog", "isComplete":true }``` * Select **Send** Select the Headers tab in the lower pane and copy the **Location** header: ![Headers tab of the Postman console](first-web-api/_static/pmget.png) You can use the Location header URI to access the resource you just created. Recall the `GetById` method created the `"GetTodo"` named route: ```csharp [HttpGet("{id}", Name = "GetTodo")] public IActionResult GetById(string id) ``` ### Update [!code-csharp[Main](first-web-api/sample/TodoApi/Controllers/TodoController.cs?name=snippet_Update)] `Update` is similar to `Create`, but uses HTTP PUT. The response is [204 (No Content)](http://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html). According to the HTTP spec, a PUT request requires the client to send the entire updated entity, not just the deltas. To support partial updates, use HTTP PATCH. ![Postman console showing 204 (No Content) response](first-web-api/_static/pmcput.png) ### Delete [!code-csharp[Main](first-web-api/sample/TodoApi/Controllers/TodoController.cs?name=snippet_Delete)] The response is [204 (No Content)](http://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html). ![Postman console showing 204 (No Content) response](first-web-api/_static/pmd.png) ## Next steps * [Routing to Controller Actions](xref:mvc/controllers/routing) * For information about deploying your API, see [Publishing and Deployment](../publishing/index.md). * [View or download sample code](https://github.com/aspnet/Docs/tree/master/aspnetcore/tutorials/first-web-api/sample) * [Postman](https://www.getpostman.com/)