Managing Technical Debt with NDepend!

Managing Technical Debt with NDepend!

What the hell is Technical Debt?

You are working on a cool new project, and want to implement the amazing new design pattern that you learned about a few days ago. All is well. Until your boss comes to your desk and tells you the release is preponed (is that even a word?) by 10 days. WTH man! Now whatever plans you had to make the code readable, testable, SOLID - goes for a toss, and you stuff the code to make up for the early release.

So you bought some time by writing "not-so-good code" to meet the deadline.

Let this sink in for a moment.

You know that the code that you had written needs to be refactored, but you don't have the bandwidth (not the internet one) to do so, so you live with this "bad code" - aka technical debt.

Well, debts inherently aren't bad, you take debts all the time - Credit Cards, Car loans, House loans. But you need to pay the interest, and as long you re-pay, it's well and good. The same goes for the technical debt. It's a trade-off between Product Delivery and impeccable code.

How do I know that I have a Technical Debt

Some time ago, I did not have any single tool which can give a clear idea of what kind of and how much technical debt I had.

I was looking at different metrics e.g., Code Duplication, Cyclomatic Complexity, Test Coverage, Dependency and Coupling, and more to get an idea.

Then I stumbled upon NDepend.

NDepende can scan your code base and tell you how much technical debt you have in a single actionable dashboard.

What is NDepend

NDepend can do much more than that, e.g.,

  • Code Rule and Code Query over LINQ (CQLinq)

  • Powerful Dependency Graph and Matrix

  • Smart Technical Debt Estimation

  • Quality Gates

  • In-Depth Issues Management

  • Trend Monitoring

  • Harness Test Coverage Data

  • Code Quality and Code Metrics

  • Detect Dependency Cycles

  • Code Diff since Baseline

  • Enforce Immutability and Purity

  • Complexity and Diagrams

  • Continuous Integration Reporting

  • Warnings on Build

  • Process Health

and more.

They consider themselves, The "Swiss Army Knife" for .NET Developers, Architects and Teams

Visit their official site to know more.

The NDepend team was generous to give me a copy of their license to try out the product

If you're new to NDepend or want to know how to install and get started, folks at NDepend have written wonderful step-by-step documentation.

The thing I like about NDepend is, it gives me a visual dashboard that is actionable. And all the data it provides, has LINQ queries to support that and accommodate custom rules as well.

How to find out Technical Debt

For this post, I've taken the Clean Architecture repo from Jason Taylor. If you don't know Jason, he's the Clean Architecture guy, a great developer, trainer, architect, and more.

Once you clone the repo in Visual Studio, here's what the Project structure looks like:

project-structure

An Angular SPA with Application Layer, Domain Layer, and Infrastructure Layer - the Clean Architecture.

Attached is the NDepend to this solution:

attach-to-solution

NDepend will detect the projects and exclude test projects. Click on Analyze X NET Assemblies.

pre-run

Give it some time to finish.

finished

Click on the View NDepend Dashboard.

Dashboard

The dashboard gives you a plethora of information about the Project code quality, lines of code, issues, and comments, but we're interested in technical debt.

dashboard

In this case, you can see there is a 5.56% of technical debt. Well done, Jason!

The Smart Technical Debt feature of NDepend also tells us how many days it would take us to resolve the technical debt.

Amazing, right?

Additionally, it gives us the Rating of the project based on Technical Debt. In our case, B. is Pretty good. And how much effort is required to reach A?

Exploring Technical Debt

If you click on the rating, NDepend will show you the hot spots of the code where most of the debt is.

hotspot

Clicking on the Debt % gives the list of all the issue

rules

Click on a specific rule to find out the details and what all code has contributed to this technical debt.

rule-detail

In the above example, our code base has violated the following rule:

• Rule Name: Avoid namespaces dependency cycles • Rule Id: ND1401 • Rule Explicit Id: AvoidNamespacesDependencyCycles

• Rule Description:

This rule lists all application namespace dependency cycles. Each row shows a different cycle, indexed with one of the namespaces entangled in the cycle.

Read our white books relative to partitioning code, to know more about namespaces dependency cycles, and why avoiding them is a simple yet efficient solution to clean the architecture of a code base. https://www.ndepend.com/docs/white-books

• How to Fix Issues of this Rule:

Removing the first pairs of mutually dependent namespaces will eliminate most namespaces dependency cycles. This is why it is recommended to focus first on matches of the default rule Avoid namespaces mutually dependent before attempting to fix issues of the present rule.

Once all mutually dependent namespace occurrences are solved, remaining cycles matched by the present rule necessarily involve 3 or more namespaces like in: A is using B is using C is using A.

To browse a cycle on the dependency graph or the dependency matrix, right-click a cycle cell in the result of the present rule and export the matched namespaces to the dependency graph or matrix. This is illustrated here: https://www.ndepend.com/docs/visual-studio-dependency-graph#Entangled

With such a cycle graph visualized you can determine which dependencies should be discarded to break the cycle. To do so, you need to identify which namespace should be at low level and which one should be at high-level.

In the A is using B is using C is using A cycle example, if A should be at a low level then C should be at a higher level than A. As a consequence C shouldn't use A and this dependency should be removed. To remove a dependency you can refer to patterns described in the HowToFix section of the rule Avoid namespaces mutually dependent.

Notice that the dependency matrix can also help to visualize and breaking cycles. In the matrix, cycles are represented with red squares and black cells. To easily browse dependency cycles, the dependency matrix comes with an option: Display Direct and Indirect Dependencies. See related documentation here: [ndepend.com/docs/dependency-structure-matri.. ndepend.com/docs/dependency-structure-matri.. ndepend.com/docs/dependency-structure-matri..)

The estimated Debt, which means the effort to fix such issue, doesn't depend on the cycle length. First, because fixing the rule Avoid namespaces mutually dependent will fix most cycles reported here, and second because even a long cycle can be broken by removing a single or a few dependencies.

• How to Suppress an Issue of this Rule:

In the source code, tag the concerned code element with this attribute: [SuppressMessage("NDepend", "ND1401:AvoidNamespacesDependencyCycles", Justification="...")] This attribute requires the compilation symbol CODE_ANALYSIS to be set on each Visual Studio project relying on it. Without the CODE_ANALYSIS symbol defined, the attribute is not compiled and the issues are not suppressed.

Pretty detailed, right?

Next steps

Now you have a basic understanding of what Technical Debt is and how NDepend can take you into your journey of becoming Technical-Debt-free (if that's a thing).

Visit the NDepend site to learn more.

Here's the repo used in this post, give this a star:

https://github.com/iSatishYadav/CleanArchitectureCodeBase

Originally posted at:

https://blog.satishyadav.com/managing-technical-debt-with-ndepend

Happy Coding!

Did you find this article valuable?

Support Satish Yadav by becoming a sponsor. Any amount is appreciated!