8

我正在尝试 FSharp 数据提供程序,但针对使用 npgsql 的 Postgresql。我在第一线就失败了。

当我尝试创建 SqlDataConnection 时,它会抛出错误消息,连接字符串不正确。

类型提供程序“Microsoft.FSharp.Data.TypeProviders.DesignTime.DataProviders”报告错误:不支持关键字:“端口:5432;数据库”。

现在,我使用 Servicestack.Ormlite 测试连接字符串和数据。这基本上使用 IdbConnection。所以,连接都是正确的。但我不知道为什么 Type Provider 不起作用。

这是代码。

    //type dbSchema = SqlDataConnection<ConnectionString = "Server=localhost;Port=5432; Database=TestDB;User Id=postgres;Password=g00gle*92;" >
[<CLIMutable>]
type Person = 
    { ID : int;
      FirstName : string;
      LastName : string }

[<EntryPoint>]
let main args = 
    let dbFactory = 
        OrmLiteConnectionFactory
            (
             "Server=localhost;Port=5432; Database=TestDB;User Id=postgres;Password=*****;", 
             PostgreSqlDialect.Provider)
    use dbConnection = dbFactory.OpenDbConnection()
    Console.WriteLine dbConnection.State
    let persons = dbConnection.Select<Person>()
    persons.ForEach(fun p -> Console.WriteLine p.FirstName)
    Console.Read() |> ignore
    0

在上面的代码中,第一个注释行不起作用,而下面的代码使用相同的设置。这意味着问题仅与类型提供程序有关,而不是与连接恕我直言。

我是否需要做任何其他设置。

如果需要任何其他详细信息,请告诉我。

更新

在kvb的评论之后,我尝试了两者。这是带有网络配置的更新代码。

//type dbSchema = SqlEntityConnection<ConnectionStringName = "TestDB", Provider="Npgsql">
    type dbSchema = SqlEntityConnection< ConnectionStringName="TestDB" >

    [<CLIMutable>]
    type Person = 
        { ID : int;
          FirstName : string;
          LastName : string }

    [<EntryPoint>]
    let main args = 
        let dbFactory = 
            OrmLiteConnectionFactory
                (
                 "Server=localhost;Port=5432; Database=TestDB;User Id=postgres;Password=*******;", 
                 PostgreSqlDialect.Provider)
        use dbConnection = dbFactory.OpenDbConnection()
        Console.WriteLine dbConnection.State
        let persons = dbConnection.Select<Person>()
        persons.ForEach(fun p -> Console.WriteLine p.FirstName)
        Console.Read() |> ignore
        0

这是网络配置

  <system.data>
    <DbProviderFactories>
      <add name="Npgsql Data Provider"
            invariant="Npgsql"
            description="Data Provider for PostgreSQL"
            type="Npgsql.NpgsqlFactory, Npgsql" />
    </DbProviderFactories>
  </system.data>
  <connectionStrings>
    <add name="TestDB"
          connectionString="Server=localhost:5432; Database=TestDB;User Id=postgres;Password=******;"
          providerName="Npgsql" />

  </connectionStrings>

这是appconfig中的程序集。我不认为它会在 GAC 中,因为我通过 nuget 添加

 <dependentAssembly>
    <assemblyIdentity name="Npgsql" publicKeyToken="5d8b90d52f46fda7" culture="neutral" />
    <bindingRedirect oldVersion="0.0.0.0-2.0.12.0" newVersion="2.0.12.0" />
 </dependentAssembly>

上面两个都被评论,另一个没有评论,两者都失败并出现不同的错误。第一个因错误而失败

类型提供程序“Microsoft.FSharp.Data.TypeProviders.DesignTime.DataProviders”报告错误:读取架构时出错。错误 7001:在配置中找不到指定的存储提供程序“Npgsql”,或者“Npgsql”无效。找不到请求的 .Net Framework 数据提供程序。它可能没有安装。

第二个是这个错误

类型提供程序“Microsoft.FSharp.Data.TypeProviders.DesignTime.DataProviders”报告错误:读取架构时出错。错误 7001:提供程序未返回 ProviderManifestToken 字符串。建立与 SQL Server 的连接时发生与网络相关或特定于实例的错误。服务器未找到或无法访问。验证实例名称是否正确以及 SQL Server 是否配置为允许远程连接。(提供者:命名管道提供者,错误:40 - 无法打开与 SQL Server 的连接)找不到网络路径

我仍然不明白它为什么要搜索 SQL Server。

如果需要任何进一步的信息,请告诉我。

4

2 回答 2

4

我将发布部分答案,希望有人可以解决下一个问题。

以下代码将编译:

open Microsoft.FSharp.Data.TypeProviders
open System.Data.Entity // this is important -- you cannot see any tables without it

type internal dbSchema = 
    SqlEntityConnection<
        ConnectionString="Server=localhost;Database=testdb;User Id=postgres;Password=password;", 
        Provider="Npgsql">

[<EntryPoint>]
let main argv = 
    let context = dbSchema.GetDataContext()
    query { for item in context.test_table do
            select item }
    |> Seq.iter (fun item -> printfn "%A" item)
    0

对于 通过创建test_table的数据库中的表testdb

CREATE TABLE test_table
(
  id integer NOT NULL,
  value text,
  CONSTRAINT "PK_test_x_Id" PRIMARY KEY (id)
)
WITH (
  OIDS=FALSE
);
ALTER TABLE test_table
  OWNER TO postgres;

为此,您需要做四件事:

  • 广汽 Npgsql.dll ( "C:\Program Files (x86)\Microsoft SDKs\Windows\v8.0A\bin\NETFX 4.0 Tools\x64\gacutil.exe" /i [filename])
  • GAC Mono.Security.dll(在 Npgsql.dll 被 NuGet 下载到的同一目录中
  • 将 DbProviderFactory 添加到您的 .NET 4 64 位 machine.config ( "C:\Windows\Microsoft.NET\Framework64\v4.0.30319\Config\machine.config"):这与您在 app.config 中的内容相同,但添加到 machine.config 中的相应部分,我目前有一个条目用于Microsoft SQL Server Compact Data Provider. 请记住包含正确的公钥令牌。
<system.data>
  <DbProviderFactories>
    <add name="Npgsql Data Provider"
      invariant="Npgsql"
      description="Data Provider for PostgreSQL"
      type="Npgsql.NpgsqlFactory, Npgsql, Version=2.0.12.0, Culture=neutral, PublicKeyToken=5d8b90d52f46fda7" />
</DbProviderFactories>
</system.data>
  • 重新启动视觉工作室。

现在SqlEntityConnection在设计时进行编译,您将能够看到所有可用的表。这也将愉快地编译成可执行文件。

这回答了你的问题;但现在对于奇怪的一点,这意味着你仍然不开心。当您运行此代码时,它会在调用时抛出ArgumentException一个dbSchema.GetDataContext()

提供的连接字符串应该是有效的特定于提供程序的连接字符串或 EntityClient 接受的有效连接字符串。

内部异常状态

不支持“服务器”关键字。

带有堆栈跟踪

在 System.Data.EntityClient.EntityConnectionStringBuilder.set_Item(String 关键字,对象值) 在 System.Data.Common.DbConnectionStringBuilder.set_ConnectionString(String value) 在 System.Data.EntityClient.EntityConnectionStringBuilder..ctor(String connectionString) 在 SqlEntityConnection1.dbSchema .GetDataContext()

我已经尝试过操纵连接字符串以使其工作,但我认为这一定是提供程序在运行时与设计时创建连接字符串的方式的一个错误。由于此代码被发送到动态程序集中,因此您如何查看代码以了解发生了什么并不明显。

于 2013-08-10T00:38:18.163 回答
1

我遇到了同样的困难。这可以通过在 App.config 文件中添加来解决:

<entityFramework>
<defaultConnectionFactory type="System.Data.Entity.Infrastructure.LocalDbConnectionFactory, EntityFramework">
  <parameters>
    <parameter value="v11.0" />
  </parameters>
</defaultConnectionFactory>
<providers>
  <provider invariantName="System.Data.SqlClient" type="System.Data.Entity.SqlServer.SqlProviderServices, EntityFramework.SqlServer" />
  <provider invariantName="Npgsql" type="Npgsql.NpgsqlServices, Npgsql.EntityFramework" />
</providers>

于 2014-10-08T05:47:03.770 回答