Есть ли в C # парсер строки подключения? - PullRequest
171 голосов
/ 26 января 2011

У меня есть строка подключения, и я хочу иметь возможность посмотреть, например, «Источник данных».Есть ли парсер или мне нужно искать строку?

Ответы [ 9 ]

285 голосов
/ 26 января 2011

Да, есть класс System.Data.Common.DbConnectionStringBuilder.

Класс DbConnectionStringBuilder предоставляет базовый класс, из которого создаются строго типизированные строители строки соединения (SqlConnectionStringBuilder), OleDbConnectionStringBuilder и т. Д.) Производные.Создатели строк соединений позволяют разработчикам программно создавать синтаксически правильные строки соединений, а также анализировать и перестраивать существующие строки соединений.

Интересующие подклассы:

System.Data.EntityClient.EntityConnectionStringBuilder
System.Data.Odbc.OdbcConnectionStringBuilder
System.Data.OleDb.OleDbConnectionStringBuilder
System.Data.OracleClient.OracleConnectionStringBuilder
System.Data.SqlClient.SqlConnectionStringBuilder

Например,Чтобы «вывести источник данных» из строки подключения SQL-сервера, вы можете сделать:

var builder = new SqlConnectionStringBuilder(connectionString);
var dataSource = builder.DataSource;
33 голосов
/ 20 марта 2013

Существуют специальные компоновщики строк подключения от разных поставщиков, такие как SqlConnectionStringBuilder, MySqlConnectionStringBuilder, SQLiteConnectionStringBuilder и т. Д. (К сожалению, на этот раз нет открытого интерфейса от MS).В противном случае у вас есть DbProviderFactory.CreateConnectionStringBuilder , который даст вам альтернативный способ написания, независимый от поставщика.Вам нужно будет указать провайдера в конфигурационном файле и иметь правильную версию dll.Например,

var c = "server=localhost;User Id=root;database=ppp";
var f = DbProviderFactories.GetFactory("MySql.Data.MySqlClient"); //your provider
var b = f.CreateConnectionStringBuilder();
b.ConnectionString = c;
var s = b["data source"];
var d = b["database"];

Однажды я написал для себя ручной анализ, который не доставил мне никаких проблем.Было бы тривиально расширить это, чтобы предоставить информацию о других параметрах (сейчас это только для простых вещей, таких как имя БД, источник данных, имя пользователя и пароль).Примерно так:

static readonly string[] serverAliases = { "server", "host", "data source", "datasource", "address", 
                                           "addr", "network address" };
static readonly string[] databaseAliases = { "database", "initial catalog" };
static readonly string[] usernameAliases = { "user id", "uid", "username", "user name", "user" };
static readonly string[] passwordAliases = { "password", "pwd" };

public static string GetPassword(string connectionString)
{
    return GetValue(connectionString, passwordAliases);
}

public static string GetUsername(string connectionString)
{
    return GetValue(connectionString, usernameAliases);
}

public static string GetDatabaseName(string connectionString)
{
    return GetValue(connectionString, databaseAliases);
}

public static string GetServerName(string connectionString)
{
    return GetValue(connectionString, serverAliases);
}

static string GetValue(string connectionString, params string[] keyAliases)
{
    var keyValuePairs = connectionString.Split(';')
                                        .Where(kvp => kvp.Contains('='))
                                        .Select(kvp => kvp.Split(new char[] { '=' }, 2))
                                        .ToDictionary(kvp => kvp[0].Trim(),
                                                      kvp => kvp[1].Trim(),
                                                      StringComparer.InvariantCultureIgnoreCase);
    foreach (var alias in keyAliases)
    {
        string value;
        if (keyValuePairs.TryGetValue(alias, out value))
            return value;
    }
    return string.Empty;
}

Для этого вам не нужно ничего особенного в конфигурационном файле или вообще никакой dll.Предложение Contains в Where важно, только если вам нужно обойти плохо отформатированные строки подключения, такие как server = localhost;pp;, где pp ничего не добавляет.Чтобы вести себя как обычные строители (которые в этих случаях взорвутся), измените Where на

.Where(kvp => !string.IsNullOrWhitespace(kvp))
11 голосов
/ 01 июля 2013

Вот пара строк кода, которые анализируют любую строку подключения в словаре:

Dictionary<string, string> connStringParts = connString.Split(';')
    .Select(t => t.Split(new char[] { '=' }, 2))
    .ToDictionary(t => t[0].Trim(), t => t[1].Trim(), StringComparer.InvariantCultureIgnoreCase);

И тогда вы можете получить доступ к любой части:

string dataSource = connStringParts["Data Source"];
6 голосов
/ 23 марта 2011

Вы хотите использовать DbProviderFactory.CreateConnectionStringBuilder () , который предоставляет вам компоновщик / синтаксический анализатор строки подключения, специфичный для вашего коннектора, но не требует использования каких-либо классов, специфичных для коннектора.

6 голосов
/ 26 января 2011

Используйте SqlConnectionStringBuilder К сожалению, вам придется использовать специфичный для БД ConnectionStringBuilder, поскольку строки подключения различаются.

4 голосов
/ 21 июня 2012

Да, вы можете сделать это с помощью классов ConnectionStringBuilder. Вот список доступных реализаций DbConnectionStringBuilder для стандартных поставщиков данных:

System.Data.Odbc.OdbcConnectionStringBuilder
System.Data.OleDb.OleDbConnectionStringBuilder
System.Data.OracleClient.OracleConnectionStringBuilder
System.Data.SqlClient.SqlConnectionStringBuilder

здесь приведен пример примера разбора строки подключения и отображения ее элементов.

 string conString = @"Data Source=.\sqlexpress;" +
                        "Database=Northwind;Integrated Security=SSPI;" +
                        "Min Pool Size=5;Max Pool Size=15;Connection Reset=True;" +
                        "Connection Lifetime=600;";
    // Parse the SQL Server connection string and display it's properties

    SqlConnectionStringBuilder objSB1 = new SqlConnectionStringBuilder(conString);
    Response.Write("<b>Parsed SQL Connection String Parameters:</b>");
    Response.Write(" <br/>  Database Source = " + objSB1.DataSource);
    Response.Write(" <br/>  Database = " + objSB1.InitialCatalog);
    Response.Write(" <br/>  Use Integrated Security = " + objSB1.IntegratedSecurity);
    Response.Write(" <br/>  Min Pool Size = " + objSB1.MinPoolSize);
    Response.Write(" <br/>  Max Pool Size = " + objSB1.MaxPoolSize);
    Response.Write(" <br/>  Lifetime = " + objSB1.LoadBalanceTimeout);
3 голосов
/ 09 февраля 2015

Вы можете использовать DbConnectionStringBuilder, и вам не нужен какой-либо конкретный поставщик:

Следующий код:

var cnstr = "Data Source=data source value;Server=ServerValue";
var builder = new DbConnectionStringBuilder();
builder.ConnectionString = cnstr;
Console.WriteLine("Data Source: {0}", builder["Data Source"]);
Console.WriteLine("Server: {0}", builder["Server"]);

Выходы на консоль:

Data Source: data source value
Server: ServerValue

РЕДАКТИРОВАТЬ:

Поскольку DbConnectionStringBuilder реализует IDictionary, вы можете перечислить параметры строки подключения:

foreach (KeyValuePair<string, object> kv in builder)
{
    Console.WriteLine("{0}: {1}", kv.Key, kv.Value);
}
1 голос
/ 03 декабря 2018

Мне не очень понравились все ответы здесь.Вот что я нашел.

С .NET Core

Вы можете использовать DbConnectionStringBuilder напрямую:

var builder = new System.Data.Common.DbConnectionStringBuilder();
builder.ConnectionString = settings.ConnectionString;
var server = builder["server"];
0 голосов
/ 18 декабря 2017

Итак, я обнаружил, что все существующие ответы были более или менее неправильными.В итоге я получил следующее тривиальное решение:

class ConnectionStringParser: DbConnectionStringBuilder {
    ConnectionStringParser(string c) { Connection = c; }
    public override bool ShouldSerialize(string keyword) => true;
}

Анализатор находится в DbConnectionStringBuilder и довольно прост в обращении.Единственная глупость, которую мы должны сделать, - это установить ShouldSerialize так, чтобы он всегда возвращал true, чтобы предотвратить потерю компонентов при попытке обхода произвольных строк соединения.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...