Can't use System.Configuration.Configuration manager in a .NET Standard2.0 library on .NET FX4.6

C#.Net.Net Standard.Net Standard-2.0

C# Problem Overview


I have an assembly created in NetStandard2.0. It reads AppSettings using System.Configuration.ConfigurationManager. I have installed nuget package of System.Configuration.ConfigurationManager with version 4.4.X which is suitable for NetStandard2.0.

When I refer this assembly in console app (.Net Core) it is reading AppSettings properly, but when I refer this assembly in old .NetFramework(4.6.X) console app it is not working and throwing an exception.

Please see the code below.

Assembly 1: NetStandard 2.0

Nuget: System.Configuration.ConfigurationManager 4.4.0

using System.Configuration;

namespace Bootstrapper.Lib
{
   public class Bootstrapper
   {
     public Bootstrapper()
     {
     
     }
   
     public void LoadAppSettings()
     {
         string serachPattern= 
         ConfigurationManager.AppSettings["AssemblySearchPattern"];
     }
  }

}

Console App: NetFx 4.6.X

using System;
using Bootstrapper.Lib;
namespace Bootstrapper.Console
{
  class Program
  {
    static void Main(string[] args)
    {
        new Bootstrapper().LoadAppSettings();
    }
  }
}

Exception After Run:

'Could not load file or assembly 'System.Configuration.ConfigurationManager, 
 Version=4.0.0.0, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51' or one 
 of its dependencies. The system cannot find the file specified.'

It will work with Console App developed using .NetCore.

Please help!!!

C# Solutions


Solution 1 - C#

As @kiran mentioned in a comment you can solve this by running:

Install-Package System.Configuration.ConfigurationManager

in NuGet Package Manager

Solution 2 - C#

It is not possible to create .NET Standard library which references System.Configuration.ConfigurationManager package and uses ConfigurationManager class. Once library adds reference to .NET Core specific package it ceases to be portable .NET Standard library since it is bound to framework specific package.

.NET Standard 2.0 does not contain System.Configuration.ConfigurationManager API. Therefore, the only way to use this API is to build one version of the library against .NET Core System.Configuration.ConfigurationManager package which can be used on .NET Core and have a second version of the library which is build against .NET FX System.Configuration assembly and can be used on .NET FX.

Solution 3 - C#

Had the same issue and after installing the same System.Configuration.ConfigurationManager package in the FX4.6 project resolved this issue.

Solution 4 - C#

Alternative solutions are the following:

1. Copy .Net Standard dependencies

Add below line to your .Net Standard csproj file (Assembly 1) as described here.

<CopyLocalLockFileAssemblies>true</CopyLocalLockFileAssemblies>

Then rebuild Console App and System.Configuration.ConfigurationManager.dll with other dependencies of Assembly 1 will be copied to bin directory of Console App.

In my case it copies the below list of dlls Dependencies of .Net Standard

Interesting thing is that when I use another solution and install System.Configuration.ConfigurationManager NuGet package to FX4.6 Console App directly, I have another set of dependencies in result: Dependencies of .Net Framework

Notice that System.Configuration.ConfigurationManager.dll as well as other dlls have different size. As I understand, in first case it was copied from %userprofile%\.nuget\packages\system.configuration.configurationmanager\5.0.0\lib\netstandard2.0, but in second case from %userprofile%\.nuget\packages\system.configuration.configurationmanager\5.0.0\lib\net461

Anyway, both solutions work, so I'm not sure which is more correct.

or

2. Use multi-targeting

In your .Net Standard csproj file (Assembly 1) set second target to net461 as described here with line

<TargetFrameworks>netstandard2.0;net461</TargetFrameworks>

Then on build two versions of library will be generated - in \bin\Release\net461 and in \bin\Release\netstandard2.0. If your Console App targets .Net Framework 4.6.1 or higher, it will automatically take correct version of Assembly 1 as dependency (the one from \net461).

The benefit of that approach is that net461 version of library doesn't require dependency from System.Configuration.ConfigurationManager.dll and can be distributed without it.

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
QuestionkiranView Question on Stackoverflow
Solution 1 - C#Rilcon42View Answer on Stackoverflow
Solution 2 - C#Jacek BlaszczynskiView Answer on Stackoverflow
Solution 3 - C#prcontraView Answer on Stackoverflow
Solution 4 - C#nt86View Answer on Stackoverflow