Как получить последний выполненный SQL-запрос по SqlConnection?

На самом деле, мой сценарий немного отличается от упомянутого здесь. Я задал другой вопрос. Но так как я не могу найти там решения, я решил изменить подход.

у меня есть SqlConnection объект доступен для моего кода. Все остальные объекты ADO.NET, такие как SqlCommand, SqlParameter и т.д. не доступны для меня. Эти другие объекты потребляются Dapper Extensions ORM.

Мое приложение выполняет запросы SQL, используя SqlConnection объект и метод Dapper Extensions. SQL-запрос автоматически генерируется Dapper Extensions; сгенерированный запрос не доступен для меня. Я хочу войти в этот запрос SQL.

У меня уже есть свой модуль логов, и единственное, что мне нужно, это последний SQL-запрос, выполненный объектом соединения.

Как получить последний выполненный SQL-запрос SqlConnection?

Следующее не работает, потому что SqlCommand не доступен

Если я пойму SqlCommandЯ могу построить запрос из него с помощью кода ниже; к сожалению, это не доступно для меня.

public string GetCommandLogString(IDbCommand command)
{
    string outputText;
    if(command.Parameters.Count == 0)
    {
        outputText = command.CommandText;
    }
    else
    {
        StringBuilder output = new StringBuilder();
        output.Append(command.CommandText);
        output.Append("; ");

        IDataParameter objIDataParameter;
        int parameterCount = command.Parameters.Count;
        for(int i = 0; i < parameterCount; i++)
        {
            objIDataParameter = (IDataParameter)command.Parameters[i];
            output.Append(string.Format("{0} = '{1}'", objIDataParameter.ParameterName, objIDataParameter.Value));

            if(i + 1 < parameterCount)
            {
                output.Append(", ");
            }
        }
        outputText = output.ToString();
    }
    return outputText;
}

1 ответ

Решение

Подход, который я использовал в прошлом, когда я не хотел полагаться на какие-либо внешние инструменты (или когда инструментов не хватало, как, например, при работе с Access), - это использовать соединение с базой данных и командовать классы-оболочки, чтобы Я могу добавить протоколирование к любому из их методов или свойств.

Чтобы использовать его, вы передаете любое соединение, которое хотите использовать, в конструктор WrappedDbConnection -

using (var conn = new WrappedDbConnection(GetMyConnection()))
{
    // Do work using Dapper here against "conn"
}

(Примечание. Когда вызывается метод Dispose экземпляра WrappedDbConnection, он передается в базовое соединение, поэтому вам не нужно "использование" для WrappedDbConnection и отдельное "использование" для вашего соединения - вам нужен только один "использование" ", как показано выше).

Два класса, которые вам нужны, определены ниже.

Обратите внимание, что есть вызовы Console.WriteLine в методах "ExecuteNonQuery", "ExecuteReader", "ExecuteReader" и "ExecuteScalar", которые выписывают, какой запрос будет выполняться. Вы можете изменить это в соответствии с вашими требованиями, чтобы выписать запрос после его завершения, или вы можете использовать другой вывод, отличный от Console.Writeline, но эти изменения должны быть достаточно простыми для внесения.

public class WrappedDbConnection : IDbConnection
{
    private readonly IDbConnection _conn;
    public WrappedDbConnection(IDbConnection connection)
    {
        if (connection == null)
            throw new ArgumentNullException(nameof(connection));

        _conn = connection;
    }

    public string ConnectionString
    {
        get { return _conn.ConnectionString; }
        set { _conn.ConnectionString = value; }
    }

    public int ConnectionTimeout
    {
        get { return _conn.ConnectionTimeout; }
    }

    public string Database
    {
        get { return _conn.Database; }
    }

    public ConnectionState State
    {
        get { return _conn.State; }
    }

    public IDbTransaction BeginTransaction()
    {
        return _conn.BeginTransaction();
    }

    public IDbTransaction BeginTransaction(IsolationLevel il)
    {
        return _conn.BeginTransaction(il);
    }

    public void ChangeDatabase(string databaseName)
    {
        _conn.ChangeDatabase(databaseName);
    }

    public void Close()
    {
        _conn.Close();
    }

    public IDbCommand CreateCommand()
    {
        return new WrappedDbCommand(_conn.CreateCommand());
    }

    public void Dispose()
    {
        _conn.Dispose();
    }

    public void Open()
    {
        _conn.Open();
    }
}

public class WrappedDbCommand : IDbCommand
{
    private readonly IDbCommand _cmd;
    public WrappedDbCommand(IDbCommand command)
    {
        if (command == null)
            throw new ArgumentNullException(nameof(command));

        _cmd = command;
    }

    public string CommandText
    {
        get { return _cmd.CommandText; }
        set { _cmd.CommandText = value; }
    }

    public int CommandTimeout
    {
        get { return _cmd.CommandTimeout; }
        set { _cmd.CommandTimeout = value; }
    }

    public CommandType CommandType
    {
        get { return _cmd.CommandType; }
        set { _cmd.CommandType = value; }
    }

    public IDbConnection Connection
    {
        get { return _cmd.Connection; }
        set { _cmd.Connection = value; }
    }

    public IDataParameterCollection Parameters
    {
        get { return _cmd.Parameters; }
    }

    public IDbTransaction Transaction
    {
        get { return _cmd.Transaction; }
        set { _cmd.Transaction = value; }
    }

    public UpdateRowSource UpdatedRowSource
    {
        get { return _cmd.UpdatedRowSource; }
        set { _cmd.UpdatedRowSource = value; }
    }

    public void Cancel()
    {
        _cmd.Cancel();
    }

    public IDbDataParameter CreateParameter()
    {
        return _cmd.CreateParameter();
    }

    public void Dispose()
    {
        _cmd.Dispose();
    }

    public int ExecuteNonQuery()
    {
        Console.WriteLine($"[ExecuteNonQuery] {_cmd.CommandText}");
        return _cmd.ExecuteNonQuery();
    }

    public IDataReader ExecuteReader()
    {
        Console.WriteLine($"[ExecuteReader] {_cmd.CommandText}");
        return _cmd.ExecuteReader();
    }

    public IDataReader ExecuteReader(CommandBehavior behavior)
    {
        Console.WriteLine($"[ExecuteReader({behavior})] {_cmd.CommandText}");
        return _cmd.ExecuteReader();
    }

    public object ExecuteScalar()
    {
        Console.WriteLine($"[ExecuteScalar] {_cmd.CommandText}");
        return _cmd.ExecuteScalar();
    }

    public void Prepare()
    {
        _cmd.Prepare();
    }
}
Другие вопросы по тегам