views:

1839

answers:

5

I am trying to set up a simple transaction for my Linq-to-Sql actions against my Sql 2000 database. Using TransactionScope it looks like this:

using (TransactionScope transaction = new TransactionScope())
{
    try
        {
     Store.DBDataContext dc = new Store.DBDataContext();
     Store.Product product = GetProduct("foo");
     dc.InsertOnSubmit(product);
     dc.SubmitChanges();
     transaction.Complete();
    }
    catch (Exception ex)
    {     
     throw ex;
    }
}

However, i keep getting the following error:

The partner transaction manager has disabled its support for remote/network transactions. (Exception from HRESULT: 0x8004D025)

But, if I set up the transaction using a traditional transaction, it works fine. So this works fine:

Store.DBDataContext dc = new Store.DBDataContext();
try
{
    dc.Connection.Open();
    dc.Transaction = dc.Connection.BeginTransaction();
    Store.Product product = GetProduct("foo");
    dc.InsertOnSubmit(product);
    dc.SubmitChanges(); 
    dc.Transaction.Commit();
}
catch (Exception ex)
{
    dc.Transaction.Rollback();
    throw ex;
}
finally
{
    dc.Connection.Close();   
    dc.Transaction = null;
}

I'm wondering if the TransactionScope is doing something different under the covers than my second implementation. If not, what am I losing by not using TransactionScope? Also, any guidance on what is causing the error would be good too. I've confirmed that MSDTC is running in both sql server and on my client machine.

A: 

Have you tried this advice from the MS forums:

First verify the "Distribute Transaction Coordinator" Service is running on both database server computer and client computers 1. Go to "Administrative Tools > Services" 2. Turn on the "Distribute Transaction Coordinator" Service if it is not running

If it is running and client application is not on the same computer as the database server, on the computer running database server 1. Go to "Administrative Tools > Component Services"

  1. On the left navigation tree, go to "Component Services > Computers My Computer" (you may need to double click and wait as some nodes need time to expand)

  2. Right click on "My Computer", select "Properties"

  3. Select "MSDTC" tab

  4. Click "Security Configuration"

  5. Make sure you check "Network DTC Access", "Allow Remote Client", "Allow Inbound/Outbound", "Enable TIP" (Some option may not be necessary)

  6. The service will restart

  7. BUT YOU MAY NEED TO REBOOT YOUR SERVER IF IT STILL DOESN'T WORK

Mitch Wheat
Yeah, I've verified everything on my workstation. However, getting the access to check the actual server is non-trivial. I've been assured that it is set up correctly, however I cannot confirm with my own eyes.
Bramha Ghosh
+1  A: 

have you checked your firewall settings?

John Boker
Windows Firewall is disabled
Bramha Ghosh
do you have some sort of other company firewall that you dont know about installed?
John Boker
You're asking me if I know about something that I don't know about?
Bramha Ghosh
yes, i am, and i need 10 chars
John Boker
As Rumsfeld would say - "As we know, There are known knowns. There are things we know we know. We also know there are known unknowns. That is to say we know there are some things we do not know. But there are also unknown unknowns, the ones we don't know we don't know."
Bramha Ghosh
"You're asking me if I know about something that I don't know about?" LOL
Eduardo Molteni
+3  A: 

Take a look here:

Fast transactions with System.Transactions and Microsoft SQL Server 2000 http://blogs.msdn.com/florinlazar/archive/2005/09/29/475546.aspx

And here:
http://forums.microsoft.com/MSDN/ShowPost.aspx?PostID=230390&SiteID=1

First verify the "Distribute Transaction Coordinator" Service is running on both database server computer and client computers
1. Go to "Administrative Tools > Services"
2. Turn on the "Distribute Transaction Coordinator" Service if it is not running

If it is running and client application is not on the same computer as the database server, on the computer running database server
1. Go to "Administrative Tools > Component Services"
2. On the left navigation tree, go to "Component Services > Computers > My Computer" (you may need to double click and wait as some nodes need time to expand)
3. Right click on "My Computer", select "Properties"
4. Select "MSDTC" tab
5. Click "Security Configuration"
6. Make sure you check "Network DTC Access", "Allow Remote Client", "Allow Inbound/Outbound", "Enable TIP" (Some option may not be necessary, have a try to get your configuration)
7. The service will restart
8. BUT YOU MAY NEED TO REBOOT YOUR SERVER IF IT STILL DOESN'T WORK (This is the thing drove me crazy before)

On your client computer use the same above procedure to open the "Security Configuration" setting, make sure you check "Network DTC Access", "Allow Inbound/Outbound" option, restart service and computer if necessary.

On you SQL server service manager, click "Service" dropdown, select "Distribute Transaction Coordinator", it should be also running on your server computer.

Keith Sirmons
The DatabaseTransactionAdapter referenced in the Florin Lazar post did the trick. I'll post my implementation code as a response.
Bramha Ghosh
The instructions for configuring the MSDTC don't seem to apply to windows 7 I only have one option under the MSDTC tab it says: use local coordinator or specify remote host to use.
Myster
+1  A: 

The DatabaseTransactionAdapter implementation in the Florin Lazar post that Keith Sirmons pointed me to seems to do the trick. Here's my code that calls it:

Store.DBDataContext dc = new Store.DBDataContext();
using (TransactionScope transaction = new TransactionScope())
{
    try
    {
     var dbAdapter = new DatabaseTransactionAdapter(dc.Connection);
     dc.Connection.Open();
     dbAdapter.Begin();
     dc.Transaction = (SqlTransaction)dbAdapter.Transaction;
        Store.Product product = GetProduct("foo");
        dc.InsertOnSubmit(product);
        dc.SubmitChanges();
        transaction.Complete();
    }
    catch (Exception ex)
    {                
        throw ex;
    }
}

The only thing that makes me uneasy is that I'm not explicitly closing the connection even though it's not declared within a 'using' statement.

But according to Florin Lazar, that's on purpose.

And you also must not close the connection, because the connection should stay open until the transaction is completed, which happens after the “using” statement ends. The adapter will take ownership of the connection lifetime and close it when it is done with it.

Bramha Ghosh
A: 

more notice that: - Server Configuration Guide Enable network COM+ access (Windows Server 2003) Start ==> Control Panel ==> Add or Remove Programs ==>Add/Remove Windows Components, Select Application Server, and then click Details. Click Enable network COM+ access, and then click OK. Click Next, and then click Finish.
- If between 2 server have firewall, open firewall for both in/out on this range port: Click Start ==> Control Panels ==> Administrative Tools ==> Component Services. Expand Component service, expand Computers, Right click on My Computer and choose Properties. In the My Computer Properties window, click on tab Default Protocol, click on connection-oriented TCP/IP and choose properties. in the new window, click add and type new DTC ports range. Click on OK to apply these change. Server must be restart for new change take effect