Skip to main content

Posts

Showing posts from March, 2021

Kubernetes–Working with multiple namespaces

Kubernetes has the concept of namespaces to logically seperate different parts of your cluster. To simplify working with different namespaces, it is recommend to create a separate context: kubectl config set-context <contexname> --namespace=<namespace> --cluster=<clustername> --user=<clusteruser> After a new context is created, you can switch to this new context using: kubectl config use-context <contextname> All commands you execute afterwards will apply to this context. You can verify your current context by doing the following: kubectl config current-context To view all the contexts: kubectl config view

Kubernetes - Reload your ASP.NET Core configuration when a configmap changes

Yesterday I blogged about configmaps and how changing the configmap is not picked up by ASP.NET Core. In this post I want to dig deeper in why this is the case and suggest a possible solution. The CreateDefaultBuilder() method in ASP.NET Core will load your appsettings.json file and monitor it for changes (see the reloadOnChange: true): Behind the scenes the reloadOnChange setting will monitor the config file for changes based on the last modified date. Seems logical… Let’s continue to find out why this doesn’t work with Kubernetes. In Kubernetes a config map is mounted as a symlink : kubectl exec -it <pod-name> -- bash root@example-deployment-839f6c6546-c783b:/appg # ls -la rwxrwxrwx 3 root root 4096 Mar 25  10:01 . drwxr-xr-x 1 root root 4096 Mar 25 10:47 .. drwxr-xr-x 2 root root 4096 Mar 25 10:01 ..2021_03_25_10_01_16.386067924 lrwxrwxrwx 1 root root 31 Mar 25 10:01 ..data -> ..2021_03_25_10_01_16.386067924 lrw...

Kubernetes–Trigger a new deployment after a configmap changes

We are using configmaps as a way to store configuration data for our pods. At startup these configmaps are mounted as configuration files inside the container. Unfortunately an update of a configmap will not be picked up by ASP.NET Core. A way to solve this is by explicitly triggering a new deployment afterwards: kubectl rollout restart deploy/example-api-deployment I created another(better?) approach but that is for another blog post…

ElasticSearch on AKS–Debug a failed rollout

Installing ElasticSearch in a Kubernetes cluster is easy thanks to the available operator . Unfortunately that doesn’t guarantee a successfull rollout. Let’s see how we can find out why a deployment failed. An ElasticSearch cluster is rolled out as a statefulset. So let’s first have a look a the available statefulsets in our AKS cluster: PS /home/bart> kubectl get statefulset No resources found in default namespace. NAME                READY   AGE search-es-default   0/1     13h OK, we see that the statefulset is not ready. Let’s continue our investigation and check the related pods: PS /home/bart> kubectl get pods -l elasticsearch.k8s.elastic.co/statefulset-name=search-es-default NAME                  READY   STATUS    RESTART...

Await using in C#8

For a long time, the ‘using’ statement in C# introduced some convenience when Disposing classes implementing the IDisposable interface. The following statement: is syntactic sugar for: In C# 8.0 a new System.IAsyncDisposable interface was introduced. This interface enables asynchronous cleanup operations. To use it you need to use the ‘await using’ statement: that is syntactic sugar for:

NPM–401 error when restoring packages from an Azure DevOps Artifacts NPM registry

I had to make some changes to an Angular application, so I cloned the app and ran ‘npm install’. This failed with the following error message: npm WARN tar ENOENT: no such file or directory, open 'D:\Projects\CMS\Bridges.CMS.SPA\node_modules\.staging\dayjs-f8637d82\plugin\weekOfYear.d.ts' npm WARN tar ENOENT: no such file or directory, open 'D:\Projects\CMS\Bridges.CMS.SPA\node_modules\.staging\dayjs-f8637d82\plugin\weekYear.d.ts' npm ERR! code E401 npm ERR! Unable to authenticate, need: Bearer authorization_uri=https://login.windows.net/<guid>, Basic realm=" https://pkgsprodsu3weu.app.pkgs.visualstudio.com/ ", TFS-Federated     Some packages should be downloaded from our own NPM registry hosted in Azure DevOps Artifacts. As I couldn’t authenticate succesfully this resulted in a 401 error. To solve this I first had to install the vsts-npm-auth package: PS D:\Projects\cms\ui\CMS\Bridges.CMS.SPA> npm install -g vsts-npm-auth ...

ASP.NET Core - ASPNETCORE_FORWARDEDHEADERS_ENABLED environment variable

Although you can directly use Kestrel to act as an “Edge” server, a typical approach is to host it behind a reverse proxy(NGINX, Azure Application Gateway, …). If you are running your ASP.NET Core application inside Kubernetes, this is typically handled through an ingress . This ingress defines the hostname and paths your application should be exposed at. An ingress controller will map this configuration to an implementation. This can be a direct mapping to some infrastructure configuration such as a load balancer(e.g. Azure Application Gateway) or a reverse proxy running inside the cluster. When your app is running behind a reverse proxy, it is important to configure your application to use the “forwarded headers” added by the reverse proxy. This can be done through the Forwarded Headers middleware . Starting from .NET Core 3.0 you can take a different approach through the usage of the ASPNETCORE_FORWARDEDHEADERS_ENABLED environment variable. . Turning on the Forwarded Headers ...

Azure DevOps Pipelines–Failing Powershell tasks

Recently some of our Azure DevOps pipelines started to fail. We noticed that this only happened on pipelines where we had tasks that used Powershell behind the scenes. The error message we got looked like this: Cannot process an element with node type "Text" A search on the Internet brought us to a solution . Disabling the “Run PowerShell in Parallel” setting solved the problem. Anyone with a better solution?

Kubernetes–Override an appsettings.json file in an ASP.NET Core app through ConfigMaps

Kubernetes has the concept of a ConfigMap. A ConfigMap gives you an easy way to override files in a Pod what makes it an easy way to replace config files in a container. Let’s first create a ConfigMap that uses an AppSettings.json: Next we need to create a deployment where we load this ConfigMap and mount it on the correct path: Let’s walk to the important parts of the Deployment.yaml step by step: First we need to load the ConfigMap as a volume. We used the name of our configmap ‘example-configmap’ and specified a name to point to this volume ‘appsettings-volume’: volumes: - name: appsettings-volume configMap: name: example-configmap Now we need to mount this volume on the correct path. Notice that we mount the appsettings.json on the ‘app’ path. This is because the root of our .NET Core app is inside the app folder. We  also specified the subPath to ensure that only the specified file is overridden, and any other existing content in the mount ...

Kubernetes–Open a bash prompt on a Pod

When using Kubernetes, sooner or later you’ll have a Pod that doesn’t work as expected. Last friday I blogged about what to do when your Pod is in a crash loop. Today I’ll share a trick to open a shell on a Pod in trouble. To open a Bash shell on a failing pod you can execute the following command: kubectl exec –it <pod name> – bash This will open a Bash shell on the specified pod: PS /home/bart> kubectl exec -it example-pod-86f495c7f7-7s87g -- bash root@example-pod-86f495c7f7-7s87g:/app #

Debug a Pod in a CrashLoopBackoff

When using Kubernetes, sooner or later you’ll encounter a failed deployment where your pods are in CrashLoopBackoff . CrashLoopBackOff tells that a pod crashes right after the start. Kubernetes tries to start pod again, but the pod crashes again and this goes in a loop. It’s hard to know what is going on as the pod is restarting before you even have a chance to take a look at its logs. So how to figure out why your pod is failing? The trick is to call the logs command but include the ‘-p’ parameter to get log messages from previous instantiations as well: kubectl logs [podname] -p

AKS with Azure Application Gateway–Reroute to root path

I’m currently working on a setup where we combine AKS(Azure Kubernetes Service) with Azure Application Gateway for ingress . As always it’s quite an adventure especially in a fast moving ecosystem like Kubernetes. In our setup we have one public IP address and domain. Our different applications are hosted as subsites. An example: Root domain: www.ourdomainname.be App  1: www.ourdomainname.be/app1 App 2: www.ourdomainname.be/app2 To do this we configure our ingress object like this: 2 things important to notice here: We added an ‘kubernetes.io/ingress.class’ annotation to specify that we want to use ‘azure/application-gateway’ . We use the path to redirect the correct prefix to the correct service. e.g. traffic coming from app1 will be directed to service app1-api-service . The problem with the setup above is that although our requests arrive perfectly to the correct service and pods, they arrive at the following relative URI /app1. Our pods e...

.NET Core–Scan your dependencies for vulnerabilities

Did you know that similar to ‘NPM audit’ you can scan your .NET Core dependencies for vulnerabilities? Starting from the .NET 5.0.2 SDK, you can scan and list any known vulnerabilities in your dependencies with the dotnet list package --vulnerable command: By default you only see vulnerabilities within your top-level packages. If you are interested in seeing vulnerabilities within your transitive packages, you can use the --include-transitive parameter: Safe coding!

ASP.NET Core–Prevent redirection on API requests

In one of my ASP.NET Core applications I have Cookie authentication enabled. When a user tries to access a controller and is not authorized, they are redirect to Account/Login?ReturnUrl=[...]. This is great for MVC endpoints but doesn’t make much sense for API requests. There are multiple ways to avoid this from happening. The way I solved it is by checking the incoming request in the CookieAuthenticationEvents. When the request starts with ‘/API’, we return a 401 status code instead of redirecting the user:

EF Core–AsSplitQuery()

Entity Framework makes it really easy to fetch related data. For example let’s get all the Customers with their ordered products. This involves 3 entities: Customer –> Order –> Product. To fetch this in one go in EF Core, you can use the following LINQ query: On the database this will end up in a query like this: The problem is that this results in a cartesian product as the customer data is duplicated for every ordered product in the result set. A solution exists in EF Core through the AsSplitQuery method that allows Entity Framework to split each related entity into a query on each table: More information: https://docs.microsoft.com/en-us/ef/core/querying/single-split-queries

Microsoft Ignite 2021 – Book of news

I’m currently bingwatching all sessions from Microsoft Ignite 2021 . In case you still want to have some free time the upcoming weeks, you can get a nice summary of the most important announcements through the Microsoft Ignite Book of News :  

Editing files in Azure Cloud Shell

Small trick I wasn’t aware of that it was possible in Azure Cloud Shell . You can directly edit files inside the Cloud shell. Therefore enter ‘code <filename>’:

.NET Core HttpClient–Testing redirections

When testing a specific API I had to check if the user was redirected to the correct location. However although I thought that I had written my api correctly, the response code didn’t match. Here is the related test code: Do you spot my mistake? Let’s have a look at the documentation: The default HttpClient will automatically handle redirects what makes it impossible to check for the 302 status code To change this behavior, you need to create your own HttpHandler and use it when building up your HttpClient instance:

NHibernate –Dependency Injection enabled IInterceptor

Yesterday I showed the AuditInterceptor we are using in one of my applications. Maybe you noticed that we were using dependency injection: Important to notice is that we are injecting a scoped dependency that contains the current user id(through the IUserFactory). The trick to get this working is to set the interceptor when the session is created. Therefore we register our IInterceptor implementation in the IoC container(Autofac in our situation): Then when we construct a new session instance we’ll get the IInterceptor from the container and link it to the session:

NHibernate–Interceptor doesn’t work for inherited objects

I got some trouble with NHibernate. I created the following interceptor to update audit fields before saving changes to the database: This interceptor does it’s job for simple objects but seems not to work for inherited objects: To get it working I had to change the interceptor to always return true in the OnSave and OnFlushDirty methods: