views:

868

answers:

3

I'm writing a test winforms / C# / .NET 3.5 application for the system we're developing and we fell in the need to switch between .config files at runtime, but this is turning out to be a nightmare.

Here's the scene: the Winforms application is aimed at testing a WebApp, divided into 5 subsystems. The test proccess works with messages being sent between the subsystems, and for this proccess to be sucessful each subsystem got to have its own .config file.

For my Test Application I wrote 5 separate configuration files. I wish I was able to switch between these 5 files during runtime, but the problem is: I can programatically edit the application .config file inumerous times, but these changes will only take effect once. I've been searching a long time for a form to address this problem but I still wasn't sucessful.

I know the problem definition may be a bit confusing but I would really appreciate it if someone helped me.

Thanks in advance!

--- UPDATE 01-06-10 ---

There's something I didn't mention before. Originally, our system is a Web Application with WCF calls between each subsystem. For performance testing reasons (we're using ANTS 4), we had to create a local copy of the assemblies and reference them from the test project. It may sound a bit wrong, but we couldn't find a satisfying way to measure performance of a remote application.

--- End Update ---

Here's what I'm doing:

public void UpdateAppSettings(string key, string value)
    {

        XmlDocument xmlDoc = new XmlDocument();
        xmlDoc.Load(AppDomain.CurrentDomain.SetupInformation.ConfigurationFile);
        foreach (XmlElement item in xmlDoc.DocumentElement)
        {
            foreach (XmlNode node in item.ChildNodes)
            {

                if (node.Name == key)
                {
                    node.Attributes[0].Value = value;
                    break;
                }
            }
        }
        xmlDoc.Save(AppDomain.CurrentDomain.SetupInformation.ConfigurationFile);

        System.Configuration.ConfigurationManager.RefreshSection("section/subSection");


    }
A: 

My guess is you are not really closing the file handle the first time so windows does not "see" you making the later changes.

My suggestions is to use an API call to IIS and turn off the web app (and pool), make the change, turn on the web app. This way you are sure it will re-read the file and have a "clean" environment for each test.

Hogan
I updated the question with some details. Basically, in this context (performance testing) we're not using IIS, the assemblies are being referenced locally. I'm using Xml libraries to write to the .config file. During execution I can see that the file is being written and changed.
born to hula
A: 

Assuming that the file handle to the configuration file is closed after the configuration file has been read and processed, I would send a message to the application to tell it to re-read the configuration file after you have updated the file. If this approach is not working, then I suspect (as Hogan suggested) that the file handle is not closed. What error codes are you getting from the file opening, reading and closing system calls? (use perror to report the error message)

David Harris
This approach is interesting, but can you give me some steps on how to send this message to the application? I tried calling ConfigurationManager.RefreshSection() but it didn't work out.
born to hula
@born to hula: How do you change the configuration @ runtime? I'd use the same approach to have the application reread the configuration file. What language and OS are you using? What are the messages sent to the application during testing? Does the input to the application come from a web page? Answering these questions would give some clues about the best approach for your situation.
David Harris
I'm using XmlDocument to change the .config file (refer to the updated question). But, regardless of how many times I change it, the changes will only be reflected once on the execution context. I'm using C# and Windows XP. The input from the application comes straight from the form. Basically, the main input is a message, from which a variety of objects is created.
born to hula
Oh, I'm using .NET 3.5.
born to hula
@born to hula: My experience is on Unix; not on Windows. Sorry.
David Harris
A: 

UPDATE

The solution below did not work because XmlDocument does not dispose and it seems some versions of .net do not close correctly when given a file path. The solution (example code in the link) is to open a stream which will do a dispose and pass that stream to the save function.

A solution is shown here. http://www.devnewsgroups.net/group/microsoft.public.dotnet.xml/topic40736.aspx

Old stuff below

Try this:

Note, I changed to xpath, but it has been a while so I might have gotten the xpath wrong, but in any case you should use xpath and not walk the tree. As you can see it is much clearer.

The important point is the using statement which will dispose(), which I think was your problem.

Let me know, good luck.

  public void UpdateAppSettings(string key, string value)
  {
    using (XmlDocument xmlDoc = new XmlDocument())
    {
      xmlDoc.Load(AppDomain.CurrentDomain.SetupInformation.ConfigurationFile);
      xmlDoc.DocumentElement.FirstChild.SelectSingleNode("descendant::"+key).Attributes[0].Value = value;
      xmlDoc.Save(AppDomain.CurrentDomain.SetupInformation.ConfigurationFile);
    }
    System.Configuration.ConfigurationManager.RefreshSection("section/subSection");
  }
Hogan
That kinda makes sense, the problem is XmlDocument doesn't implement IDisposable, so it is not possible to use the "using" scope.
born to hula
Yes you are right... see my comments above.
Hogan
Well, I have a slight guess that this isn't the case. I'm using framework .NET 3.5. Besides, I think if the file stream was still open, by the second time I'd try to write to the xml file, I would get an exception... but thanks anyway.
born to hula
ummm... there is a documented problem with the .net framework with example code that looks exactly the same as yours, I provide you with a link to code that solves that problem and you are not going to try it? Don't you want to fix the problem?
Hogan
I'm gonna give it a try and will update you with the results. Thanks for your help.
born to hula
I've tried it out and apparently the result was the same. Thanks anyway man.
born to hula
To bad -- sorry I could not help.
Hogan