Search code examples
asp.netwcfsystem.reflection

Issue with CompileAssemblyFromSource when called from a WCF app


I have a WCF service that exposes a method. When a client calls this method, the following occurs:

  1. the method does some processing
  2. it tries to load an assembly if its already there
  3. if the above dll isn't there, it generates C# code, compiles it using CSharpCodeProvider's CompileAssemblyFromSource api
  4. It then loads the just generated assembly

Now the problem. The first time the method it called, it generates the assembly at point 3), and when it tries to return the assembly reference via CompilerResults.CompiledAssembly it throws a file not found exception. However, I can clearly see that the assembly has been generated at the specified location and I can open it with other applications.

If I call the method again via a client, it is able to load the assembly (it was successfully generated as a result of the previous call) and goes ahead to do the remaining set of tasks. Only when the assembly isnt there and it generates it and goes ahead to load it immediately, I get that exception. Any ideas? I have tried playing around with the web.config, changing impersonation to true/false. I have a seperate app pool to run this web application and I tried changing the identity of the app pool from local service to local system, even gave my windows logon credentials which has admin rights but no luck.

Any help would be much appreciated.


Solution

  • Thanks user1796307 for your input. I had resolved this issue, but forgot to update it. Sharing it for everyone's benefit. It was the .NET Fusion at play. It caches assembly load path and won't even try to load an assembly if a previous attempt to load from same location had failed. In other words:

    if (Assembly.Load("C:\xyz.dll") == null)
     {
        Compile("C:\xyz.dll"); // Now the dll exists
        Assembly.Load("C:\xyz.dll"); // this will still fail
     }
    
    
    The solution is to change it as:
    
        if (!File.Exists("C:\xyz.dll")
    {
      Compile("C:\xyz.dll"); // Now the dll exists
      Assembly.Load("C:\xyz.dll"); // this will now work
    
    }