Sunday, 5 April 2020

Microservices Using ASP.NET Core

Microservices

The term microservices portrays a software development style that has grown from contemporary trends to set up practices that are meant to increase the speed and efficiency of developing and managing software solutions at scale. Microservices is more about applying a certain number of principles and architectural patterns as architecture. Each microservice lives independently, but on the other hand, also all rely on each other. All microservices in a project get deployed in production at their own pace, on-premise on the cloud, independently, living side by side.

Microservices Architecture

The following picture from Microsoft Docs shows the microservices architecture style.
Microservice Using ASP.NET Core 
There are various components in a microservices architecture apart from microservices themselves.
Management. Maintains the nodes for the service.
Identity Provider. Manages the identity information and provides authentication services within a distributed network.
Service Discovery. Keeps track of services and service addresses and endpoints.
API Gateway. Serves as client’s entry point. Single point of contact from the client which in turn returns responses from underlying microservices and sometimes an aggregated response from multiple underlying microservices.
CDN. A content delivery network to serve static resources for e.g. pages and web content in a distributed network
Static Content The static resources like pages and web content
Microservices are deployed independently with their own database per service so the underlying microservices look as shown in the following picture.
Microservice Using ASP.NET Core 

Monolithic vs Microservices Architecture

Monolithic applications are more of a single complete package having all the related needed components and services encapsulated in one package.
Following is the diagrammatic representation of monolithic architecture being package completely or being service based.
Microservice Using ASP.NET Core 
Microservice is an approach to create small services each running in their own space and can communicate via messaging. These are independent services directly calling their own database.
Following is the diagrammatic representation of microservices architecture.
Microservice Using ASP.NET Core 
In monolithic architecture, the database remains the same for all the functionalities even if an approach of service-oriented architecture is followed, whereas in microservices each service will have its own database.

Docker Containers and Docker installation

Containers like Dockers and others slice the operating system resources, for e.g. the network stack, processes namespace, file system hierarchy and the storage stack. Dockers are more like virtualizing the operating system. Learn more about dockers here. Open this URL and click on Download from Docker hub. Once downloaded, login to the Docker and follow instructions to install Docker for Windows.

Microservice using ASP.NET Core

This section will demonstrate how to create a Product microservice using ASP.NET Core step by step with the help of pictures. The service would be built using ASP.NET Core 2.1 and Visual Studio 2017. Asp.NET Core comes integrated with VS 2017. This service will have its own DBcontext and database with the isolated repository so that the service could be deployed independently.
Microservice Using ASP.NET Core 

Creating an ASP.NET Core Application Solution

  1. Open the Visual Studio and add a new project.

    Microservice Using ASP.NET Core
  1. Choose the application as ASP.NET Core Web Application and give it a meaningful name.

    Microservice Using ASP.NET Core
  1. Next, choose API as the type of the project and make sure that “Enable Docker Support” option is selected with OS type as Linux.

    Microservice Using ASP.NET Core
  1. The solution will look as shown below.

    Microservice Using ASP.NET Core

Adding Models

  1. Add a new folder named “Model” to the project.

    Microservice Using ASP.NET Core
  1. In the Models folder, add a class named Product.

    Microservice Using ASP.NET Core
  1. Add a few properties like Id, Name, Description, Price to the product class. The product should also be of some kind and for that, a category model is defined and a CategoryId property is added to the product model.

    Microservice Using ASP.NET Core
  1. Similarly, add Category model.

    Microservice Using ASP.NET Core

Enabling EF Core

Though .NET Core API project has inbuilt support for EF Core and all the related dependencies are downloaded at the time of project creation and compilation that could be found under SDK section in the project as shown below.
Microservice Using ASP.NET Core 
Microsoft.EntityFrameworkCore.SqlServer (2.1.1) should be the package inside the downloaded SDK’s. If it is not present, it could be explicitly added to the project via Nuget Packages.

Adding EF Core DbContext

A database context is needed so that the models could interact with the database.
  1. Add a new folder named DBContexts to the project.

    Microservice Using ASP.NET Core
  1. Add a new class named ProductContext which includes the DbSet properties for Products and Categories. OnModelCreating is a method via which the master data could be seeded to the database. So, add the OnModelCreating method and add some sample categories that will be added to the database initially into the category table when the database is created.

    Microservice Using ASP.NET Core

    ProductContext code
    1. using Microsoft.EntityFrameworkCore;  
    2. using ProductMicroservice.Models;  
    3.   
    4. namespace ProductMicroservice.DBContexts  
    5. {  
    6.   public class ProductContext : DbContext  
    7.   {  
    8.     public ProductContext(DbContextOptions<ProductContext> options) : base(options)  
    9.     {  
    10.     }  
    11.     public DbSet<Product> Products { getset; }  
    12.     public DbSet<Category> Categories { getset; }  
    13.   
    14.     protected override void OnModelCreating(ModelBuilder modelBuilder)  
    15.     {  
    16.       modelBuilder.Entity<Category>().HasData(  
    17.           new Category  
    18.           {  
    19.             Id = 1,  
    20.             Name = "Electronics",  
    21.             Description = "Electronic Items",  
    22.           },  
    23.           new Category  
    24.           {  
    25.             Id = 2,  
    26.             Name = "Clothes",  
    27.             Description = "Dresses",  
    28.           },  
    29.           new Category  
    30.           {  
    31.             Id = 3,  
    32.             Name = "Grocery",  
    33.             Description = "Grocery Items",  
    34.           }  
    35.       );  
    36.     }  
    37.   
    38.   }  
    39. }  
  1. Add a connection string in the appsettings.json file.

    Microservice Using ASP.NET Core
Open the Startup.cs file to add the SQL server db provider for EF Core. Add the code services.AddDbContext<ProductContext>(o => o.UseSqlServer(Configuration.GetConnectionString("ProductDB"))); under ConfigureServices method. Note that in the GetConnectionString method the name of the key of the connection string is passed that was added in appsettings file.
Microservice Using ASP.NET Core 

Adding Repository

Repository works as a micro component of microservice that encapsulates the data access layer and helps in data persistence and testability as well.
  1. Add a new folder named Repository in the project and add an Interface name IProductRepository in that folder. Add the methods in the interface that performs CRUD operations for Product microservice.

    Microservice Using ASP.NET Core
  1. Add a new concrete class named ProductRepository in the same Repository folder that implements IProductRepository. All these methods need:

    Microservice Using ASP.NET Core
  1. Add the implementation for the methods via accessing context methods.

    ProductRepository.cs
    1. using Microsoft.EntityFrameworkCore;  
    2. using ProductMicroservice.DBContexts;  
    3. using ProductMicroservice.Models;  
    4. using System;  
    5. using System.Collections.Generic;  
    6. using System.Linq;  
    7.   
    8. namespace ProductMicroservice.Repository  
    9. {  
    10.   public class ProductRepository: IProductRepository  
    11.   {  
    12.     private readonly ProductContext _dbContext;  
    13.   
    14.     public ProductRepository(ProductContext dbContext)  
    15.     {  
    16.       _dbContext = dbContext;  
    17.     }  
    18.     public void DeleteProduct(int productId)  
    19.     {  
    20.       var product = _dbContext.Products.Find(productId);  
    21.       _dbContext.Products.Remove(product);  
    22.       Save();  
    23.     }  
    24.   
    25.     public Product GetProductByID(int productId)  
    26.     {  
    27.       return _dbContext.Products.Find(productId);  
    28.     }  
    29.   
    30.     public IEnumerable<Product> GetProducts()  
    31.     {  
    32.       return _dbContext.Products.ToList();  
    33.     }  
    34.   
    35.     public void InsertProduct(Product product)  
    36.     {  
    37.       _dbContext.Add(product);  
    38.       Save();    }  
    39.   
    40.     public void Save()  
    41.     {  
    42.       _dbContext.SaveChanges();  
    43.     }  
    44.   
    45.     public void UpdateProduct(Product product)  
    46.     {  
    47.       _dbContext.Entry(product).State = EntityState.Modified;  
    48.       Save();  
    49.     }  
    50.   }  
    51. }  
  1. Open the Startup class in the project and add the code as services.AddTransient<IProductRepository, ProductRepository>(); inside ConfigureServices method so that the repository’s dependency is resolved at a run time when needed.

    Microservice Using ASP.NET Core

Adding Controller

The microservice should have an endpoint for which a controller is needed which exposes the HTTP methods to the client as endpoints of the service methods.
  1. Right click on the Controllers folder and add a new Controller as shown below.

    Microservice Using ASP.NET Core
  1. Select the option “API Controller with read/write actions” to add the controller.

    Microservice Using ASP.NET Core
  1. Give the name of the controller as ProductController.

    Microservice Using ASP.NET Core
  1. A ProductController class will be added in the Controllers folder with default read/write actions that will be replaced later with product read/write actions and HTTP methods are created acting as an endpoint of the service.

    Microservice Using ASP.NET Core
  1. ValuesController can be deleted as it is not needed.

    Microservice Using ASP.NET Core
  1. Add implementation to the methods by calling the repository methods as shown below. The basic implementation is shown here for the sake of understanding the concept. The methods could be attribute routed and could be decorated with more annotations as per need.

    ProductController.cs
    1. using Microsoft.AspNetCore.Mvc;  
    2. using ProductMicroservice.Models;  
    3. using ProductMicroservice.Repository;  
    4. using System;  
    5. using System.Collections.Generic;  
    6. using System.Transactions;  
    7.   
    8. namespace ProductMicroservice.Controllers  
    9. {  
    10.   [Route("api/[controller]")]  
    11.   [ApiController]  
    12.   public class ProductController : ControllerBase  
    13.   {  
    14.   
    15.     private readonly IProductRepository _productRepository;  
    16.   
    17.     public ProductController(IProductRepository productRepository)  
    18.     {  
    19.       _productRepository = productRepository;  
    20.     }  
    21.       
    22.     [HttpGet]  
    23.     public IActionResult Get()  
    24.     {  
    25.       var products = _productRepository.GetProducts();  
    26.       return new OkObjectResult(products);  
    27.     }  
    28.   
    29.     [HttpGet("{id}", Name = "Get")]  
    30.     public IActionResult Get(int id)  
    31.     {  
    32.       var product = _productRepository.GetProductByID(id);  
    33.       return new OkObjectResult(product);  
    34.     }  
    35.   
    36.     [HttpPost]  
    37.     public IActionResult Post([FromBody] Product product)  
    38.     {  
    39.       using (var scope = new TransactionScope())  
    40.       {  
    41.         _productRepository.InsertProduct(product);  
    42.         scope.Complete();  
    43.         return CreatedAtAction(nameof(Get), new { id = product.Id }, product);  
    44.       }  
    45.     }  
    46.   
    47.     [HttpPut]  
    48.     public IActionResult Put([FromBody] Product product)  
    49.     {  
    50.       if (product != null)  
    51.       {  
    52.         using (var scope = new TransactionScope())  
    53.         {  
    54.           _productRepository.UpdateProduct(product);  
    55.           scope.Complete();  
    56.           return new OkResult();  
    57.         }  
    58.       }  
    59.       return new NoContentResult();  
    60.     }  
    61.   
    62.     [HttpDelete("{id}")]  
    63.     public IActionResult Delete(int id)  
    64.     {  
    65.       _productRepository.DeleteProduct(id);  
    66.       return new OkResult();  
    67.     }  
    68.   }  
    69. }  

Entity Framework Core Migrations

Migrations allow us to provide code to change the database from one version to another.
  1. Open Package Manager Console.

    Microservice Using ASP.NET Core
  1. To enable the migration, type the command, Add-Migration and give that a meaningful name for e.g. InitialCreate and press enter.

    Microservice Using ASP.NET Core
  1. Once the command is executed, if we look at our solution now, we see there's a new Migrations folder. And it contains two files. One, a snapshot of our current context model. Feel free to check the files. The files are very much self-explanatory.

    Microservice Using ASP.NET Core
  1. To ensure that migrations are applied to the database there's another command for that. It's called the update-database If executed, the migrations will be applied to the current database.

    Microservice Using ASP.NET Core
  1. Check the SQL Server Management Studio to verify if the database got created.

    Microservice Using ASP.NET Core
  1. When data of the Categories table is viewed the default master data of three categories is shown.

    Microservice Using ASP.NET Core

Run the Product Microservice

The service could be run via IIS Express i.e. Visual Studio default or via Docker container as well.

Via IIS Express

Choose IIS Express in the Visual Studio as shown below and press F5 or click that IIS Express button itself.
Microservice Using ASP.NET Core 
The application will be up once the browser page is launched. Since it has nothing to show, it will be blank, but the service could be tested via any API testing client. Here Postman is used to testing the service endpoints. Keep it opened and application running.
Microservice Using ASP.NET Core 
Install Postman if it is not on the machine and launch it.
Microservice Using ASP.NET Core 
POST
To test the POST method; i.e. create a new resource, select the method as POST in postman and provide the endpoint, i.e. https://localhost:44312/api/product and in the Body section, add a JSON similar to having properties of Product model as shown below and click on Send.
Microservice Using ASP.NET Core 
The response is returned with the Id of the product as well.
Microservice Using ASP.NET Core 
The “Post” method of the controller is responsible to create a resource in the database and send the response.
The line return CreatedAtAction(nameof(Get), new { id=product.Id }, product); returns the location of the created resource that could be checked in Location attribute in the response under Headers tab.
Microservice Using ASP.NET Core 
Perform a select query on the product table and an added row is shown for the newly created product.
Microservice Using ASP.NET Core 
Create one more product in a similar way.
Microservice Using ASP.NET Core 
GET
Perform a GET request now with the same address and two records are shown as a JSON result response.
Microservice Using ASP.NET Core 
DELETE
Perform the delete request by selecting DELETE as the verb and appending id as 1 (if the product with id 1 needs to be deleted) and press Send.
Microservice Using ASP.NET Core 
In the database, one record with Id 1 gets deleted.
Microservice Using ASP.NET Core 
PUT
PUT verb is responsible for updating the resource. Select PUT verb, provide the API address and in the Body section, provide details of which product needs to be updated in JSON format. For example, update the product with Id 2 and update its name, description, and price from Samsung to iPhone specific. Press Send.
Microservice Using ASP.NET Core 
Check the database to see the updated product.
Microservice Using ASP.NET Core 

Via Docker Containers

Running the service could be done via docker commands to be run in docker command prompt and using visual studio as well. Since we added the docker support, it is easy to run the service in docker container using visual studio.
  1. Add container orchestrator support in the solution as shown below.

    Microservice Using ASP.NET Core
  1. This will ask for the orchestrator. Select Docker Compose and press OK.

    Microservice Using ASP.NET Core

    Once added to the solution, the solution will look like shown below having docker-compose with dockerignore and docker-compose.yml and its override file.

    Microservice Using ASP.NET Core

    As soon as the solution is saved, it builds the project under the container and creates a docker image. All the commands execution can be seen in the output window when the solution is saved.
  1. Open the command prompt in admin mode and navigate to the same folder where the project files are.

    Microservice Using ASP.NET Core
  1. Run the command docker images to see all the created images. We see the ProductMicroserviceimage the latest one.

    Microservice Using ASP.NET Core
  1. Now run the application with Docker as an option as shown below.

    Microservice Using ASP.NET Core
  1. Now, run the command docker ps to see the running containers. It shows the container is running on 32773:80 port.

    Microservice Using ASP.NET Core
  1. Since the container is in running state, it is good to test the service now running under the container. To test the service, replace ”values” with “product” in the address as shown below. Ideally, it should get the product details. But it gives exception as shown below.

    Microservice Using ASP.NET Core
  1. Running the same thing under IIS Express works fine i.e. on port 44312. Replace “values” with the product to get the product details,

    Microservice Using ASP.NET Core
  1. Since in IIS Express application runs fine and not in docker container, the error clearly shows that something is wrong with the SQL server that it does not understand our docker container or it is not running under docker container. In this scenario, the docker container is running as a separate machine inside the host computer. So, to connect to the SQL database in the host machine, remote connections to SQL needs to be enabled. We can fix this.
  2. Open the SQL Server Configuration Manager. Now select Protocols for MSSQLSERVER and get the IPAll port number under TCP/IP section.

    Microservice Using ASP.NET Core

    Microservice Using ASP.NET Core

    Microservice Using ASP.NET Core

    Microservice Using ASP.NET Core
  1. The connection string mentioned in the JSON file points to the data source as local which the docker container does not understand. It needs proper IP addresses with port and SQL authentication. So, provide the relevant details i.e. Data Source as Ip address, port number and SQL authentication details as shown below.

    Microservice Using ASP.NET Core
  1. Now again run the application with Docker as an option like done earlier.

    Microservice Using ASP.NET Core

    This time the response is received.
  1. Test the same in the Postman.

    Microservice Using ASP.NET Core
  1. Test again with IIS Express URL.

    Microservice Using ASP.NET Core
This proves that the microservice is running on two endpoints and on two operating systems independently locally deployed.

No comments:

Post a Comment