22. March 2023 By Vincent Scherb
Troubleshooting open source NuGet package
For a project I decided to use the libgit2 library which, fortunately, has a port to C#. When adding the NuGet package in the latest version 0.26.2
(release December 11, 2019) the program ran into a problem at runtime as the shared library cannot be loaded.
Let’s explore the steps to find more about that error and the resulting actions to resolve that problem and continue on with the project.
First I will create a MVP to replicate the problem and explore further without the other dependencies of the project. Then I will follow the process of resolving the problem and finding a solution.
This happens on the machine with Majaro x64 with .Net 6.
Replicate the error with a MVP
So far, my observation was, that every call to the library caused a problem. Therefore a minimal reproduction of that problem would be achieved by including the NuGet package and calling anything from the library.
// Get the directory where the executable is run.
string currentDirectory = Directory.GetCurrentDirectory();
// Call the library to check whether the provided location is a valid
// git repository.
bool directoryIsARepository = LibGit2Sharp.Repository.IsValid(currentDirectory);
// Set the exit code to non zero to indicate an error or the uncaught
// exception will print the details.
return directoryIsARepository ? 0 : 1;
Fig1: Simplified version of Program.cs
So after recreating the error on my machine, I created a new repository and added the least required code to reproduce the error. The repository is uploaded at sr.ht.
$ git log --oneline -n1
e8f4902 Add simple check for the library
~/Projects/poc_libgit2sharp main $ make run
dotnet build ./poc_libgit2sharp/poc_libgit2sharp.csproj
Microsoft (R) Build Engine version 17.0.0+c9eb9dd64 for .NET
Copyright (C) Microsoft Corporation. All rights reserved.
Determining projects to restore...
All projects are up-to-date for restore.
poc_libgit2sharp -> /home/vince/Projects/poc_libgit2sharp/poc_libgit2sharp/bin/Debug/net6.0/poc_libgit2sharp.dll
Build succeeded.
0 Warning(s)
0 Error(s)
Time Elapsed 00:00:01.33
dotnet run --project ./poc_libgit2sharp/poc_libgit2sharp.csproj
Current directory is '/home/vince/Projects/poc_libgit2sharp'
Checking whether the directory contains a valid git repository.
Should return true as the program is run from a repository.
Unhandled exception. System.TypeInitializationException: The type initializer for 'LibGit2Sharp.Core.NativeMethods' threw an exception.
---> System.DllNotFoundException: Unable to load shared library 'git2-106a5f2' or one of its dependencies. In order to help diagnose loading problems, consider setting the LD_DEBUG environment variable: libgit2-106a5f2: cannot open shared object file: No such file or directory
at LibGit2Sharp.Core.NativeMethods.git_libgit2_init()
at LibGit2Sharp.Core.NativeMethods.InitializeNativeLibrary()
at LibGit2Sharp.Core.NativeMethods..cctor()
--- End of inner exception stack trace ---
at LibGit2Sharp.Core.NativeMethods.git_repository_open_ext(git_repository*& repository, FilePath path, RepositoryOpenFlags flags, FilePath ceilingDirs)
at LibGit2Sharp.Core.Proxy.git_repository_open_ext(String path, RepositoryOpenFlags flags, String ceilingDirs)
at LibGit2Sharp.Repository.IsValid(String path)
at Program.<Main>$(String[] args) in /home/vince/Projects/poc_libgit2sharp/poc_libgit2sharp/Program.cs:line 7
make: *** [Makefile:18: run] Error 134
Fig2: Recreating the error on the MVP
Now let us dissect the thrown error message a bit further to understand the underlying problem before taking further actions or diving deeper into the code of the open source repository libgit2.
Unhandled exception. System.TypeInitializationException:
The type initializer for 'LibGit2Sharp.Core.NativeMethods' threw an exception.
---> System.DllNotFoundException:
Unable to load shared library 'git2-106a5f2' or one of its dependencies.
In order to help diagnose loading problems, consider setting the LD_DEBUG environment variable:
libgit2-106a5f2: cannot open shared object file: No such file or directory
Fig3: Information from the exception
We know, that the library is a wrapper around the C implementation of libgit2 to enjoy it in the managed world of dotnet. Armed with this knowledge, we can suspect that the reference to git2-106a5f2
of the System.DllNotFoundException
is about the native library. The next step is to find out where the library is loaded from and where the error occurs to get a better understanding of the problem.
Exploration with Debug options
The error message gives us all the information we need to proceed further as it indicated the usage of LD_DEBUG
. So lets explore the output with that environment variable set (output is greatly condensed):
~/Projects/poc_libgit2sharp main $ LD_DEBUG=libs make run
dotnet run --project ./poc_libgit2sharp/poc_libgit2sharp.csproj
1022222: initialize program: /home/vince/Projects/poc_libgit2sharp/poc_libgit2sharp/bin/Debug/net6.0/poc_libgit2sharp
Current directory is '/home/vince/Projects/poc_libgit2sharp'
Checking whether the directory contains a valid git repository.
Should return true as the program is run from a repository.
1022222: find library=git2-106a5f2.so [0]; searching
1022222: search cache=/etc/ld.so.cache
1022222: search path=/usr/lib/tls:/usr/lib (system search path)
1022222: trying file=/usr/lib/tls/git2-106a5f2.so
1022222: trying file=/usr/lib/git2-106a5f2.so
1022222:
1022222: [...]
1022222: find library=libgit2-106a5f2.so [0]; searching
1022222: search cache=/etc/ld.so.cache
1022222: search path=/usr/lib/tls:/usr/lib (system search path)
1022222: trying file=/usr/lib/tls/libgit2-106a5f2.so
1022222: trying file=/usr/lib/libgit2-106a5f2.so
1022222:
1022222: [...]
1022222: find library=git2-106a5f2 [0]; searching
1022222: search cache=/etc/ld.so.cache
1022222: search path=/usr/lib/tls:/usr/lib (system search path)
1022222: trying file=/usr/lib/tls/git2-106a5f2
1022222: trying file=/usr/lib/git2-106a5f2
1022222:
1022222: find library=libgit2-106a5f2 [0]; searching
1022222: search cache=/etc/ld.so.cache
1022222: search path=/usr/lib/tls:/usr/lib (system search path)
1022222: trying file=/usr/lib/tls/libgit2-106a5f2
1022222: trying file=/usr/lib/libgit2-106a5f2
1022222:
Unhandled exception.
System.TypeInitializationException: The type initializer for 'LibGit2Sharp.Core.NativeMethods' threw an exception.
---> System.DllNotFoundException: Unable to load shared library 'git2-106a5f2' or one of its dependencies.
[...]
make: *** [Makefile:18: run] Error 134
Fig4: LD_DEBUG
run to list the libraries
When setting the LD_DEBUG parameter to all
or files
we get similar results:
1022654: file=/home/vince/Projects/poc_libgit2sharp/poc_libgit2sharp/bin/Debug/net6.0/runtimes/linux-x64/native/git2-106a5f2.so [0]; dynamically loaded by /usr/share/dotnet/shared/Microsoft.NETCore.App/6.0.2/libcoreclr.so [0]
1022654:
1022654: file=/usr/share/dotnet/shared/Microsoft.NETCore.App/6.0.2/git2-106a5f2.so [0]; dynamically loaded by /usr/share/dotnet/shared/Microsoft.NETCore.App/6.0.2/libcoreclr.so [0]
1022654:
1022654: file=/home/vince/Projects/poc_libgit2sharp/poc_libgit2sharp/bin/Debug/net6.0/git2-106a5f2.so [0]; dynamically loaded by /usr/share/dotnet/shared/Microsoft.NETCore.App/6.0.2/libcoreclr.so [0]
1022654:
1022654: file=git2-106a5f2.so [0]; dynamically loaded by /usr/share/dotnet/shared/Microsoft.NETCore.App/6.0.2/libcoreclr.so [0]
[...]
1022654: file=/home/vince/Projects/poc_libgit2sharp/poc_libgit2sharp/bin/Debug/net6.0/runtimes/linux-x64/native/libgit2-106a5f2.so [0]; dynamically loaded by /usr/share/dotnet/shared/Microsoft.NETCore.App/6.0.2/libcoreclr.so [0]
1022654: file=/home/vince/Projects/poc_libgit2sharp/poc_libgit2sharp/bin/Debug/net6.0/runtimes/linux-x64/native/libgit2-106a5f2.so [0]; generating link map
1022654: dynamic: 0x00007fda6ad24d90 base: 0x00007fda6aa00000 size: 0x000000000032a5d0
1022654: entry: 0x00007fda6aa166f0 phdr: 0x00007fda6aa00040 phnum: 7
Fig5: Output from LD_DEBUG=files
The library files ./bin/Debug/net6.0/runtimes/linux-x64/native/libgit2-106a5f2.so
do exist and they are imported correctly. On the other hand the referenced file /usr/share/dotnet/shared/Microsoft.NETCore.App/6.0.2/git2-106a5f2.so
does not exist at this global location.
If there would be no specific version applied to the shared object file I would say the file, should come from the system. Let’s check the package manager and install the latest version of the underlying library libgit2
on the system.
$ sudo pacman -S libgit2
Fig6: Installing the libgit2 package globally
Even after the installation/update the error persists. So, as expected, there is no correlation with the global installation of the library and it should come all bundled with the NuGet package.
Going further and exploring the issues on GitHub and discovering a potential fix will be the next step.
Exploring GitHub issues for potential fixes
Before exploring the installation scripts or the code of the library let us explore the issues on GitHub.
There are some issues hinting that the correct dependency of the package should be used, which is LibGit2Sharp.NativeBinaries
in version 2.0.306
, which was already the case when inspecting the file ./obj/project.assets.json
.
On the other hand, there are multiple issues indicating that there are issues with .Net 5, which should be fixed with the preview version of 0.27.0
of the LibGit2Sharp
NuGet package.
Installing the preview version of the library
Now lets just update the package reference to check if the preview version will fix it for our .Net 6 application.
<ItemGroup>
<PackageReference Include="LibGit2Sharp" Version="0.27.0-preview-0182" />
</ItemGroup>
Fig7: poc_libgit2sharp.csproj
~/Projects/poc_libgit2sharp main $ make run
[...]
dotnet run --project ./poc_libgit2sharp/poc_libgit2sharp.csproj
Current directory is '/home/vince/Projects/poc_libgit2sharp'
Checking whether the directory contains a valid git repository.
Should return true as the program is run from a repository.
This message indicates that no exception was thrown and the library works as expected.
The directory '/home/vince/Projects/poc_libgit2sharp' is a git repository 'True'
Fig8: Execute with preview version
Perfect, now no exception gets thrown from the library and the expected result holds true. The working directory is indeed a git repository.
Now, having a reference to a preview version is always unpleasant and should be avoided. As we know with git itself, we have the possibility of including the sources directly and building them within our project. To draw a conclusion for the project, lets explore both options in this MVP.
Building from source
Now for the last step, lets try to build the project ourselves and include the result in our project. With this option, we can directly link the master
branch of the repository and update it accordingly to receive the latest updates instead of waiting for an update on the NuGet package.
$ git submodule add https://github.com/libgit2/libgit2sharp.git libgit2sharp
[...]
$ ls -l
-rw-r--r-- 1 vince vince 349 Apr 21 20:23 .gitignore
-rw-r--r-- 1 vince vince 99 Apr 21 22:16 .gitmodules
[...]
drwxr-xr-x 9 vince vince 4096 Apr 21 22:24 libgit2sharp
-rw-r--r-- 1 vince vince 432 Apr 21 20:54 Makefile
drwxr-xr-x 4 vince vince 4096 Apr 21 20:21 poc_libgit2sharp
$ cat .gitmodules
[submodule "libgit2sharp"]
path = libgit2sharp
url = https://github.com/libgit2/libgit2sharp.git
Fig9: Installing the submodule
And finally, we can update the project reference.
$ git diff d85adbb~..d85adbb
diff --git a/poc_libgit2sharp/poc_libgit2sharp.csproj b/poc_libgit2sharp/poc_libgit2sharp.csproj
index 4329b2b..f3f8c62 100644
--- a/poc_libgit2sharp/poc_libgit2sharp.csproj
+++ b/poc_libgit2sharp/poc_libgit2sharp.csproj
@@ -8,7 +8,7 @@
</PropertyGroup>
<ItemGroup>
- <PackageReference Include="LibGit2Sharp" Version="0.27.0-preview-0182" />
+ <ProjectReference Include="..\libgit2sharp\LibGit2Sharp\LibGit2Sharp.csproj" />
</ItemGroup>
</Project
Fig10: Update the reference
The submodule works as expected just as the preview version of the NuGet package. With the submodule, we have similar downsides to using a package manager, as the reference has to be updated manually when a new version is published.
When using the submodule, we can directly check out the master
branch and check the latest development version. It is worth noting that when a stable release of 0.27.0
is released of the package this choice has to be reevaluated.
Conclusion
To see all the changes made since the last published version, we can check out the diff on GitHub directly v0.26..master.
The help on submodules is extensive and can be viewed under man 1 git-submodul
e and man 7 gitsubmodules
.
For my project I decided to go with the submodule workaround since it gives me a few advantages:
- Building regularly lets me observe the submodule for updates
- I can keep up with changes to the library
- Give feedback to the library developers since I work with a version containing new features and fixes
- On different branches I can try different versions of the library
You can find more exciting topics from the adesso world in our blog articles published so far.