Postgresql的FSharp数据类型提供程序

时间:2013-07-25 09:26:57

标签: postgresql f# npgsql type-providers f#-3.0

我正在尝试FSharp数据提供程序,但是使用npgsql对Postgresql进行了攻击。而且我在第一线被击败了。

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

  

类型提供者   'Microsoft.FSharp.Data.TypeProviders.DesignTime.DataProviders'   报告错误:不支持关键字:'port:5432; database'。

现在,我使用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

在上面的代码中,第一个注释行无效,而下面的代码设置相同。这意味着只有类型提供者没有连接IMHO的问题。

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

如果需要任何其他详细信息,请与我们联系。

更新

在kvb的评论之后,我试过了两个。这是使用web config更新的代码。

//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

这是web config

  <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服务器。

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

2 个答案:

答案 0 :(得分: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;

要做到这一点,你需要做四件事:

  • GAC 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(与NuGet下载Npgsql.dll的目录相同
  • 将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>
  • 重新启动visual studio。

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

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

  

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

内部异常状态

  

不支持'server'关键字。

with stack-trace

  

at System.Data.EntityClient.EntityConnectionStringBuilder.set_Item(String keyword,Object value)      at System.Data.Common.DbConnectionStringBuilder.set_ConnectionString(String value)      在System.Data.EntityClient.EntityConnectionStringBuilder..ctor(String connectionString)      在SqlEntityConnection1.dbSchema.GetDataContext()

我尝试过连接字符串以使其工作,但我认为这必须是提供程序在运行时和设计时创建连接字符串的错误。由于此代码被发送到动态程序集中,因此您不必明白如何查看代码以查看正在发生的事情。

答案 1 :(得分: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>