170 lines
6.0 KiB
C#
170 lines
6.0 KiB
C#
|
using System;
|
||
|
using System.Collections;
|
||
|
using System.Collections.Generic;
|
||
|
using System.Data;
|
||
|
using System.Data.Common;
|
||
|
using System.Data.SqlClient;
|
||
|
using System.Linq.Expressions;
|
||
|
using System.IO;
|
||
|
using System.Reflection;
|
||
|
using System.Text;
|
||
|
using System.Transactions;
|
||
|
|
||
|
namespace System.Data.Linq.SqlClient {
|
||
|
using System.Data.Linq;
|
||
|
using System.Data.Linq.Provider;
|
||
|
|
||
|
internal class SqlConnectionManager : IConnectionManager {
|
||
|
private IProvider provider;
|
||
|
private DbConnection connection;
|
||
|
private bool autoClose;
|
||
|
private bool disposeConnection; // should we dispose this connection when the context is disposed?
|
||
|
private DbTransaction transaction;
|
||
|
private Transaction systemTransaction;
|
||
|
private SqlInfoMessageEventHandler infoMessagehandler;
|
||
|
private List<IConnectionUser> users;
|
||
|
private int maxUsers;
|
||
|
|
||
|
internal SqlConnectionManager(IProvider provider, DbConnection con, int maxUsers, bool disposeConnection) {
|
||
|
this.provider = provider;
|
||
|
this.connection = con;
|
||
|
this.maxUsers = maxUsers;
|
||
|
this.infoMessagehandler = new SqlInfoMessageEventHandler(this.OnInfoMessage);
|
||
|
this.users = new List<IConnectionUser>(maxUsers);
|
||
|
this.disposeConnection = disposeConnection;
|
||
|
}
|
||
|
|
||
|
|
||
|
public DbConnection UseConnection(IConnectionUser user) {
|
||
|
if (user == null) {
|
||
|
throw Error.ArgumentNull("user");
|
||
|
}
|
||
|
if (this.connection.State == ConnectionState.Closed) {
|
||
|
this.connection.Open();
|
||
|
this.autoClose = true;
|
||
|
this.AddInfoMessageHandler();
|
||
|
if (System.Transactions.Transaction.Current != null) {
|
||
|
System.Transactions.Transaction.Current.TransactionCompleted += this.OnTransactionCompleted;
|
||
|
}
|
||
|
}
|
||
|
if (this.transaction == null && System.Transactions.Transaction.Current != null &&
|
||
|
System.Transactions.Transaction.Current != systemTransaction) {
|
||
|
this.ClearConnection();
|
||
|
systemTransaction = System.Transactions.Transaction.Current;
|
||
|
this.connection.EnlistTransaction(System.Transactions.Transaction.Current);
|
||
|
}
|
||
|
|
||
|
if (this.users.Count == this.maxUsers) {
|
||
|
this.BootUser(this.users[0]);
|
||
|
}
|
||
|
this.users.Add(user);
|
||
|
return this.connection;
|
||
|
}
|
||
|
|
||
|
private void BootUser(IConnectionUser user) {
|
||
|
bool saveAutoClose = this.autoClose;
|
||
|
this.autoClose = false;
|
||
|
int index = this.users.IndexOf(user);
|
||
|
if (index >= 0) {
|
||
|
this.users.RemoveAt(index);
|
||
|
}
|
||
|
user.CompleteUse();
|
||
|
this.autoClose = saveAutoClose;
|
||
|
}
|
||
|
|
||
|
internal DbConnection Connection {
|
||
|
get { return this.connection; }
|
||
|
}
|
||
|
|
||
|
internal int MaxUsers {
|
||
|
get { return this.maxUsers; }
|
||
|
}
|
||
|
|
||
|
internal void DisposeConnection() {
|
||
|
// only close this guy if we opened it in the first place
|
||
|
if (this.autoClose) {
|
||
|
this.CloseConnection();
|
||
|
}
|
||
|
|
||
|
// If we created the connection, we need to dispose it even if the user explicitly
|
||
|
// opened it using the Connection property on the context.
|
||
|
if (this.connection != null && this.disposeConnection) {
|
||
|
this.connection.Dispose();
|
||
|
this.connection = null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal void ClearConnection() {
|
||
|
while (this.users.Count > 0) {
|
||
|
this.BootUser(this.users[0]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal bool AutoClose {
|
||
|
get { return this.autoClose; }
|
||
|
set { this.autoClose = value; }
|
||
|
}
|
||
|
|
||
|
internal DbTransaction Transaction {
|
||
|
get { return this.transaction; }
|
||
|
set {
|
||
|
if (value != this.transaction) {
|
||
|
if (value != null) {
|
||
|
if (this.connection != value.Connection) {
|
||
|
throw Error.TransactionDoesNotMatchConnection();
|
||
|
}
|
||
|
}
|
||
|
this.transaction = value;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public void ReleaseConnection(IConnectionUser user) {
|
||
|
if (user == null) {
|
||
|
throw Error.ArgumentNull("user");
|
||
|
}
|
||
|
int index = this.users.IndexOf(user);
|
||
|
if (index >= 0) {
|
||
|
this.users.RemoveAt(index);
|
||
|
}
|
||
|
if (this.users.Count == 0 && this.autoClose && this.transaction == null && System.Transactions.Transaction.Current == null) {
|
||
|
this.CloseConnection();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private void CloseConnection() {
|
||
|
if (this.connection != null && this.connection.State != ConnectionState.Closed) {
|
||
|
this.connection.Close();
|
||
|
}
|
||
|
this.RemoveInfoMessageHandler();
|
||
|
this.autoClose = false;
|
||
|
}
|
||
|
|
||
|
private void OnInfoMessage(object sender, SqlInfoMessageEventArgs args) {
|
||
|
if (this.provider.Log != null) {
|
||
|
this.provider.Log.WriteLine(Strings.LogGeneralInfoMessage(args.Source, args.Message));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private void OnTransactionCompleted(object sender, System.Transactions.TransactionEventArgs args) {
|
||
|
if (this.users.Count == 0 && this.autoClose) {
|
||
|
this.CloseConnection();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private void AddInfoMessageHandler() {
|
||
|
SqlConnection scon = this.connection as SqlConnection;
|
||
|
if (scon != null) {
|
||
|
scon.InfoMessage += this.infoMessagehandler;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private void RemoveInfoMessageHandler() {
|
||
|
SqlConnection scon = this.connection as SqlConnection;
|
||
|
if (scon != null) {
|
||
|
scon.InfoMessage -= this.infoMessagehandler;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|