Wednesday, August 16, 2023

A Deep Dive into Computed Columns in Entity Framework Core

Entity Framework Core (EF Core) is a popular Object-Relational Mapping (ORM) framework that simplifies database access for .NET applications. It provides a wide range of features to map your C# classes to database tables and manage the interaction between your application and the underlying database. One powerful feature of EF Core is its support for computed columns, which allow you to define columns in your database that are computed based on other columns' values. In this article, we'll explore the concept of computed columns in EF Core, their benefits, and how to work with them effectively.



Understanding Computed Columns

A computed column is a column in a database table that is calculated based on an expression involving other columns in the same table. Unlike regular columns that store data explicitly provided by users, computed columns derive their values from existing data within the table. This can be incredibly useful for performing calculations or transformations on data without having to retrieve it first and then calculate in your application code.

  1. Computed columns offer several advantages:

  2. Data Consistency: Since computed column values are calculated based on other columns' values, you can ensure consistency in derived data without relying on developers to perform calculations correctly.

  3. Performance: Calculations performed at the database level can be optimized for better performance, especially when dealing with large datasets.

  4. Readability: Computed columns can simplify complex calculations, making your queries and code more readable.

  5. Maintenance: When business logic changes, you only need to update the computed column expression in the database, minimizing code changes.

Using Computed Columns in EF Core

In EF Core, you can define computed columns using the .HasComputedColumnSql() method in your entity configuration. Let's walk through an example of creating a TotalPrice computed column for an Order entity.

Assuming you have an Order entity with a Quantity column and a UnitPrice column, you can calculate the total price using the following code:

public class Order { public int OrderId { get; set; } public int Quantity { get; set; } public decimal UnitPrice { get; set; } public decimal TotalPrice { get; private set; } // Computed column // Other properties and methods } public class MyDbContext : DbContext { public DbSet<Order> Orders { get; set; } protected override void OnModelCreating(ModelBuilder modelBuilder) { modelBuilder.Entity<Order>() .Property(o => o.TotalPrice) .HasComputedColumnSql("[Quantity] * [UnitPrice]"); } }
In this example, the TotalPrice property's value is calculated by multiplying Quantity and UnitPrice. The .HasComputedColumnSql() method takes an SQL expression as a parameter.


Note: Computed columns are supported in various database systems, but the syntax might differ. Be aware of any compatibility issues when working with different databases.

Saturday, May 6, 2023

How to explore/test API endpoints by Visual Studio

By integrating new tools, Visual Studio is changing in a remarkable way every day. One such tool that can make our life easier is the Visual Studio API Endpoint Explorer. This powerful tool allows you to explore and test API endpoints, making it an essential part of any developer's toolkit. What is the Visual Studio API Endpoint Explorer? The Visual Studio API Endpoint Explorer is a tool that allows developers to explore and test API endpoints. It provides a user-friendly interface that makes it easy to navigate through different endpoints and test them with different parameters. The tool supports a wide range of HTTP methods, including GET, POST, PUT, DELETE, and more. NOTE: This capability is available starting with the below version and later of Visual Studio. Microsoft Visual Studio Community 2022 (64-bit) - Preview Version 17.6.0 Preview 6.0

How to Use the Visual Studio API Endpoint Explorer?

To let you explore and interact with the API endpoints specified in the solution, we are working on a new preview tool called the Endpoints Explorer. As this is a preview feature, it must be turned on in order for us to see it. Go to Tools > Options > Environment > Preview Features and choose the Web API Endpoints Explorer to enable the new Tool Window. To narrow down the options, utilize the search text box in that dialog. View the picture below.


After enabling the feature, you have to open the C# web application where the API endpoint exists. You may access the Endpoints Explorer now that you've enabled it by selecting View > Other Windows > Endpoints Explorer. You should see the window similar to what is seen in the following image once you have opened that window. 
If you click Generate Request in this window, an HTTP file will also be generated along with the request. Below is the outcome for the /weatherforecast request after calling that context menu.




Monday, February 13, 2023

Building Umbraco 11 with dotnet 7 and running the web app in docker container

 Building Umbraco 11 with dotnet 7 and running the web app in the docker container

Since its original release in March 2013, Docker has been around and has truly impacted the structure of web apps and services and how applications are deployed and scaled.

In our world of content management websites, it hasn't nearly had the same effect, and I believe we're missing out on some excellent chances to make the sites we develop faster, more effective, more scalable, and more durable. In this post, I'll walk you through how to achieve this.

Prerequisites

In order to run this application you will need the following installed on your machine.

  • Visual Studio Code
  • WSL2
    • Windows Subsystem for Linux
  • .NET SDK version 7
    • This will also work with the earlier version .NET version core, but you need a compatible SDK
  • Docker Desktop
    • The best solution if you're using Windows or a Mac is Docker desktop, a free (for individuals) program offered by Docker that simply configures everything for you. It takes some effort to set up Docker on Linux. I won't go through particular instructions for each platform here, but you can find some pretty decent information on how to do it here: https://docs.docker.com/desktop/windows/install/ and https://docs.docker.com/get-docker/
  • Few key concepts of Docker
    • What is Docker, Docker Image, Docker Container, Container Registry, Dockerfile, and Docker Networking are a few fundamental concepts.

Anyone can find the code-base from the Github repo here.

Steps to follow:

  1. Start with a simple Umbraco website.
  2. Make an MSSQL database instance running on Docker container.
  3. Make the application container and have them communicate with one another.

Building an Umbraco container application

Let's start from scratch and build a brand-new standalone Umbraco application on our local machine. This Umbraco 11 site will be backed by .NET 7.

Install the Umbraco templates first and then create a new solution named “MyUmbracoAppOnDocker” and then add the project to the solution, and install a starter kit

Open the command window in elevated mode and run below commands

dotnet new -i Umbraco.Templates::11
dotnet new sln --name MyUmbracoAppOnDocker
dotnet new umbraco -n MyUmbracoAppOnDocker --friendly-name "Admin User" --email "admin@umbraco.com" --password "1234567890" --connection-string "Data Source=(localdb)\MSSQLLocalDB;AttachDbFilename=|DataDirectory|\MyUmbracoAppOnDockerData.mdf;Integrated Security=True"
dotnet sln MyUmbracoAppOnDocker.sln add MyUmbracoAppOnDocker
dotnet add MyUmbracoAppOnDocker package Clean

Now run the site by following command and it will create new localDB

dotnet run --project MyUmbracoAppOnDocker

 

We can see what port the site is using in the output, and we can launch the website in any browser of choice. This step must be finished to build the Database.

The project is now running locally on the development workstation as an ordinary Umbraco site.



Now it’s time to Run the database from a container.


Stop the running application before running the below command.

mkdir UmbracoData;
$currDir = Get-Location; Copy-Item -Path $currDir/MyUmbracoAppOnDocker/umbraco/Data/MyUmbracoAppOnDockerData.mdf -Destination $currDir/UmbracoData -PassThru;
$currDir = Get-Location; Copy-Item -Path $currDir/MyUmbracoAppOnDocker/umbraco/Data/MyUmbracoAppOnDockerData_log.ldf -Destination $currDir/UmbracoData -PassThru;

 

Now we will create three files in newly created folder “UmbracoData”

1.         startup.sh

2.         setup.sql

3.         Dockerfile

 

To start the MSSQL database, we will now create a bash program called Startup.sh, which we will call from a Docker file. The main job of startup.sh is to start setup.sql as the server admin (or sa) account and sleep for 15 seconds while the sql server is starting up. We have to define the password here.

#!/bin/bash
set -e
 
if [ "$1" = '/opt/mssql/bin/sqlservr' ]; then
# If this is the container's first run, initialize the application database
if [ ! -f /tmp/app-initialized ]; then
    # Initialize the application database asynchronously in a background process. This allows a) the SQL Server process to be the main process in the container, which allows graceful shutdown and other goodies, and b) us to only start the SQL Server process once, as opposed to starting, stopping, then starting it again.
    function initialize_app_database() {
    # Wait a bit for SQL Server to start. SQL Server's process doesn't provide a clever way to check if it's up or not, and it needs to be up before we can import the application database
    sleep 15s
 
    #run the setup script to create the DB and the schema in the DB
    /opt/mssql-tools/bin/sqlcmd -S localhost -U sa -P Abcd!234 -d master -i setup.sql
 
    # Note that the container has been initialized so future starts won't wipe changes to the data
    touch /tmp/app-initialized
    }
    initialize_app_database &
fi
fi
exec "$@"

 

The setup.sql file looks like:

 

USE [master]
GO
 
IF NOT EXISTS (SELECT * FROM sys.databases WHERE name = 'MyUmbracoAppOnDockerDB')
BEGIN
 
    CREATE DATABASE [MyUmbracoAppOnDockerDB] ON
    ( FILENAME = N'/var/opt/sqlserver/MyUmbracoAppOnDockerData.mdf' ),
    ( FILENAME = N'/var/opt/sqlserver/MyUmbracoAppOnDockerData_log.ldf' )
    FOR ATTACH
 
END;
GO
 
USE MyUmbracoAppOnDockerDB;

 

Now we will create a Dockerfile that describes the Database server that will run on Docker. The first line in Dockerfile describes the image that will be used: SQL Server 2019 that running on Ubuntu. After that we will define the password for SA account.

FROM mcr.microsoft.com/mssql/server:2019-GDR1-ubuntu-16.04
ENV ACCEPT_EULA=Y
ENV SA_PASSWORD=Abcd!234
ENV MSSQL_PID=Express
 
USER root
 
RUN mkdir /var/opt/sqlserver
 
RUN chown mssql /var/opt/sqlserver
 
ENV MSSQL_BACKUP_DIR="/var/opt/sqlserver"
ENV MSSQL_DATA_DIR="/var/opt/sqlserver"
ENV MSSQL_LOG_DIR="/var/opt/sqlserver"
 
COPY setup.sql /
COPY startup.sh /
 
COPY MyUmbracoAppOnDockerData.mdf /var/opt/sqlserver
COPY MyUmbracoAppOnDockerData_log.ldf /var/opt/sqlserver
 
ENTRYPOINT [ "/bin/bash", "startup.sh" ]
CMD [ "/opt/mssql/bin/sqlservr" ]

 

Go back to our Umbraco project now and change the connection string to link to this SQL server that are running on Docker. Modify the app settings. According to appsettings.Development.json

"ConnectionStrings": {
    "umbracoDbDSN": "Server=localhost,1401;Database= MyUmbracoAppOnDockerDB;User Id=sa;Password=Abcd!234;"
  }

 

Note: Because it's likely you already have MSSQL Server installed locally and using port 1433 would cause a problem, we're utilizing port 1400 instead of the standard 1433.

 

Now we will create the MSSQL Docker image by using the command.

docker build --tag=umbracodata .\UmbracoData

It will create a database image in local docker host.





After creating this docker image, it’s time to run the docker SQL server by executing below command. Again, take notice of the port that is being used, while the Docker image still uses 1433 internally, 1401 is used externally to avoid conflict with any other local SQL servers.

docker run --name umbracodata -p 1401:1433 --volume sqlserver:/var/opt/sqlserver -d umbracodata

 It will create a MSSQL container from the image umbracodata

 

Verify that the website is still functional.

Now we will run the Umbraco application locally by pointing the database serving from Docker. The command window will once again show which port should be used to access the site.

dotnet run --project MyUmbracoAppOnDocker

 

Now it’s time to create another docker file to Run the application in Docker.

We must add a new file called Dockerfile inside the root of our web project in order to start the project in Docker.

# Use the SDK image to build and publish the website
FROM mcr.microsoft.com/dotnet/sdk:7.0 AS build
WORKDIR /src
COPY ["MyUmbracoAppOnDocker.csproj", "."]
RUN dotnet restore "MyUmbracoAppOnDocker.csproj"
COPY . .
RUN dotnet publish "MyUmbracoAppOnDocker.csproj" -c Release -o /app/publish
 
# Copy the published output to the final running image
FROM mcr.microsoft.com/dotnet/aspnet:7.0 AS final
WORKDIR /app
COPY --from=build /app/publish .
ENTRYPOINT ["dotnet", "MyUmbracoAppOnDocker.dll"]
 

This section will describes that most developers struggle with is the Docker Networking.

We must have a basic understanding of Docker networking before we can run the fully functional website in a container. In Docker, containers will communicate via a variety of networking protocols. One of those is Bridge networks, which come in both default and user-defined Bridge network.

Any container may connect with any other container on the default bridge network to which all containers are can communicate if none are specified. However, it can only do so using IP addresses, which are assigned dynamically.

Containers can use the name of the network to interact in a user-defined bridge network, but only containers that have been explicitly added to the network can use the container name to connect with one another.

We will create a user-defined Bridge network first and then attach the network with the previously created database container by using the container name (umbracodata). Stop the running web application if it's still active by pressing Ctrl + C.

docker network create -d bridge umbracoNetwork
docker network connect umbracoNetwork umbracodata

 

Connectionstring for Application Container:

Create an appsettings.Staging.config file by coping the content of current appsettings.Development.json  file and replace the connectionstring as below. This is necessary to use nonstandard port. Important part is Server=umbracodata where umbracodata is the name of the database container.

"umbracoDbDSN": "Server=umbracodata;Database=MyUmbracoAppOnDockerDB;User Id=sa;Password=Abcd!234;"

 

This stage we will Create the application container image.

We can create our application image now by running the below command. Our database container is running and attached to the user-defined network umbracoNetwork.

docker build --tag=umbracoappondocker .\MyUmbracoAppOnDocker

This command will create Docker image of the Umbraco application. Observe that the web image is not being used, but you can see that the database image is being used since it is now operating.

 

 


 

Now we can create a container from umbracoappondocker image for our Umbraco application on Docker container.

docker run --name umbracoappondocker -p 8000:80 -v media:/app/wwwroot/media -v logs:/app/umbraco/Logs -e ASPNETCORE_ENVIRONMENT='Staging' --network=umbracoNetwork -d umbracoappondocker

 

 

Closure

In this tutorial, we've generated two containers and run them locally on our Docker instance, one for our Umbraco 11 application and the other for the MSSQL database server and they can communicate each other by user defined Bridge network. We've discussed networking, Docker images, and some of the prerequisites for getting things up and running.

 

Happy coding!

Saturday, January 28, 2023

Why should you avoid making async void methods? Make the methods instead return Task.

 Why should you avoid making async void methods? Make the methods instead return Task.

  • 1. Processes are always crashed by exceptions thrown by async void methods.
  • 2. Even the option to wait for the result is not available to callers.
  • 3. The caller cannot report exceptions to telemetry.
  • 4. If your VS package was started in this manner, it is unable to be responsibly blocked in Package.Close until your async work is complete.
  • 5. Use caution: when passing an async delegate or async () => become async void methods when passed to a method that accepts Action delegates. If a method accepts Func<Task> or Func<Task<T>> parameters, only pass async delegates to those methods.

A Deep Dive into Computed Columns in Entity Framework Core

Entity Framework Core (EF Core) is a popular Object-Relational Mapping (ORM) framework that simplifies database access for .NET applications...