FSharp.Data 程序集之 Http

(**

# F# Data: HTTP Utilities

.NET 库提供了强大的 API,产生和发送 HTTP WEB 请求,有两个类型,一个简单,`WebClient`(参见 [MSDN][1]) ,另一个稍微复杂,`HttpWebRequest`,(参见 [MSDN][2]) 。然而,这两个类型的使用相当困难,如果只打算运行简单的 HTTP 请求,指定参数,比如,方法、HTTP POST 数据或者额外的头。

F# Data 库提供简单的 Http 类型,有两个重载的方法:`Request` and `AsyncRequest`,用于创建同步、异步执行的请求。

[1]: http://msdn.microsoft.com/en-us/library/system.net.webclient.aspx

 [2]: http://msdn.microsoft.com/en-us/library/system.net.httpwebrequest.aspx

要使用这个类,首先要引用库,F# interactive 中使用 `#r`,在项目中添加引用,然后加载 `FSharp.Net` namespace:

*)

Credentials <- NetworkCredential ("hadstj@hotmail.com","kokyWfYXaG5H63g3CvS3UW5mgnydQFvwXyc1h+p7U+I=")

#r "../../bin/FSharp.Data.dll"

open FSharp.Net

(**

##   发送简单的请求

在发送简单的 HTTP (GET) 请求,下载指定的页面,可以使用 `Http.Request` and `Http.AsyncRequest`,只需要一个参数:

*)

// 下载页面

Http.Request("http://tomasp.net")

// 异步下载页面

async { let! html = Http.AsyncRequest("http://tomasp.net")

        printfn "%d" html.Length }

|> Async.Start

(**

在下面的文档中,我们只讨论 `Request` 方法,因为 `AsyncRequest` 的用法完全相同:

## 查询参数与头

指定查询参数的方法,可以是把参数包含在 URL 中 (例如 `http://...?test=foo&more=bar`),也可以使用可靠的参数 `query` 进行传递。下面的例子,还显式指定了 GET 方法,如果不指定,会自动设置:

*)

Http.Request("http://httpbin.org/get", query=["test", "foo"], meth="GET")

(**

指定额外的头也很相似,使用可选的参数 `headers`。这个集合不仅可以包含标准的头,比如 Accept 头(当使用 `HttpWebRequest` 时,必须显式指定),还可以包含自定义的头。

下面的例子使用 [电影数据库] (http://www.themoviedb.org) API,搜索 "batman"。要运行这个例子,需要注册,并提供 API 的 key:

这里提供了一个 key,是 F# 大拿演示用的。请仅用于测试程序。

*)

// API key for http://www.themoviedb.org

let apiKey = "6ce0ef5b176501f8c07c634dfa933cff"

// HTTP 请求

Http.Request

  ( "http://api.themoviedb.org/3/search/movie",

    query   = [ "api_key", apiKey; "query", "batman" ],

    headers = [ "accept", "application/json" ])

(**

## 发送请求数据

如果打算发送有 HTTP POST 数据的 POST 请求,可以使用 `body` 参数,以字符串形式指定额外的数据,也可以在 `bodyValues` 参数中以键-值对的形式指定数据。如果指定 body 数据,不需要指定 `meth` 参数,自动设置成 `GET` 方法。

下面的例子使用 [httpbin.org](http://httpbin.org) 服务,它会返回请求的详细信息:

*)

Http.Request("http://httpbin.org/post", bodyValues=["test", "foo"])

(**

通常,Content-Type 头被设置为 `application/x-www-form-urlencoded`,但是,也可以改变,只要使用可选的 `headers`参数,在头列表中添加 `content-type`:

*)

Http.Request

  ( "http://httpbin.org/post",

    headers = ["content-type", "application/json"],

    body = """ {"test": 42} """)

(**

## 在请求之间保持 cookies

如果想在请求之间保持 cookies,可以指定 `cookieContainer` 参数。

下面的例子是检索 MSDN 文档中有关 `HttpRequest` 类,只包含 C# 的代码片段,而没有 F# 的:

*)

// 为给定类的文档构建 URL

let msdnUrl className =

  let root = "http://msdn.microsoft.com"

  sprintf "%s/en-gb/library/%s.aspx" root className

// 得到页面,查找 F# 代码

let docInCSharp = Http.Request(msdnUrl "system.web.httprequest")

docInCSharp.Contains "<a>F#</a>"

(**

如果我们现在到另外的 MSDN 页面,在 F# 代码示例上单击,然后,返回 `HttpRequest` 类文档,它会保持相同的 `cookieContainer`,会得到 F# 的代码片段:

*)

open System.Net

let cc = CookieContainer()

// 发送请求,并切换语言

Http.Request

  ( msdnUrl "system.datetime",

    query = ["cs-save-lang", "1"; "cs-lang","fsharp"],

    cookieContainer = cc) |> ignore

// 再次请求文档,并查找 F#

let docInFSharp =

  Http.Request

    ( msdnUrl "system.web.httprequest",

      cookieContainer = cc )

docInFSharp.Contains "<a>F#</a>"

(**

如果想看到更多有关这个的信息,包括响应的头,返回的 cookie,以及响应的 URL(如果有重定向的话,它会不同于传递进去的 URL),可以使用 `RequestDetailed` 方法:

*)

let response = Http.RequestDetailed(msdnUrl "system.web.httprequest")

// 检查响应的信息

response.Cookies

response.ResponseUrl

(**

## 请求二进制数据

`Request` 方法总是把响应返回成字符串,但是,如果使用 `RequestDetailed` 方法,可以根据响应的 `content-type` 头属性,返回 `HttpResponseBody.Text` 或 `HttpResponseBody.Binary`:

*)

let logoUrl = "https://raw.github.com/fsharp/FSharp.Data/master/misc/logo.png"

match Http.RequestDetailed(logoUrl).Body with

| HttpResponseBody.Text text ->

    printfn "Got text content: %s" text

| HttpResponseBody.Binary bytes ->

    printfn "Got %d bytes of binary content" bytes.Length

(**

## 发送客户端证书

如果想在请求中加上客户端证书,可以使用可选的参数 `certificate`,然后传递 `X509ClientCertificate` 值。这需要引入 `System.Security.Cryptography` 中的 `X509Certificates` 命名空间。

假设证书保存在  `myCertificate.pfx`,程序可以这样写:

*)

open System.Security.Cryptography.X509Certificates

// 从文件加载证书

let clientCert =

  new X509Certificate2(".\myCertificate.pfx", "password")

// 发送带证书的请求

Http.Request

  ( "http://yourprotectedresouce.com/data",

    certificate = clientCert)

FSharp.Data 程序集之 Http的更多相关文章

  1. 未能加载文件或程序集"System.Data,Version=2.0.0.0,Culture=neutral,PublicKeyToken=b77a5c561934e089"或它的某一个依赖项。系统找不到指定的文件。

    sqlserver 2005打开出现无法正常访问数据,提示信息: 未能加载文件或程序集"System.Data,Version=2.0.0.0,Culture=neutral,PublicK ...

  2. 《powershell 的版本号所引起的载入 FSharp 编译器问题》基本解决

    <powershell 的版本号所引起的载入 FSharp 编译器问题>基本解决 1.FSharp.Core.dll.不光要 Add-Type,还要在编译中引用.可是,在 VS2012 的 ...

  3. 【C#】解析C#程序集的加载和反射

    目录结构: contents structure [+] 程序集 程序集的加载 发现程序集中的类型 反射对类型成员的常规操作 发现类型的成员 创建类型的实例 绑定句柄减少进程的内存消耗 解析自定义特性 ...

  4. 编译 Deedle

    编译 Deedle Deedle 中含有 RProvider. 要编译 Deedle.须要先下载 R.地址: http://cran.cnr.berkeley.edu/bin/windows/base ...

  5. [转]查询表达式 (F#)

    本文转自:http://msdn.microsoft.com/zh-cn/library/hh225374.aspx 查询表达式可以查询数据源并将数据是一种预期形式.             查询表达 ...

  6. 一系列令人敬畏的.NET核心库,工具,框架和软件

    内容 一般 框架,库和工具 API 应用框架 应用模板 身份验证和授权 Blockchain 博特 构建自动化 捆绑和缩小 高速缓存 CMS 代码分析和指标 压缩 编译器,管道工和语言 加密 数据库 ...

  7. Github上优秀的.NET Core项目

    Github上优秀的.NET Core开源项目的集合.内容包括:库.工具.框架.模板引擎.身份认证.数据库.ORM框架.图片处理.文本处理.机器学习.日志.代码分析.教程等. Github地址:htt ...

  8. 【转载】Github上优秀的.NET Core项目

    Github上优秀的.NET Core项目 Github上优秀的.NET Core开源项目的集合.内容包括:库.工具.框架.模板引擎.身份认证.数据库.ORM框架.图片处理.文本处理.机器学习.日志. ...

  9. Github上优秀的.NET Core开源项目的集合

    内容包括:库.工具.框架.模板引擎.身份认证.数据库.ORM框架.图片处理.文本处理.机器学习.日志.代码分析.教程等. Github地址:https://github.com/jasonhua95/ ...

随机推荐

  1. ZOJ3527

    题意:给你一个有向图,一共N个顶点,且每个顶点只有一个前驱或后继,在顶点上建立圣地,那么就可以获得一个信仰值,如果在这个顶点的后继节点上也建立圣地,那么将改变一定的信仰值,求解能获取的最大信仰值. 思 ...

  2. JQuery__Tab实践

    刚开始学做网站的时候,是从DIV+CSS开始的,那时候不明白“幻灯片”“二级导航”等,更不明白“动态网站”.后来,需要用到幻灯片banner的时候,老师没有仔细讲解JS,就说:从网站找来,会用就好!于 ...

  3. dede织梦后台页面及功能修改及精简操作方法

    先让我们来看看都有哪些页面控制着后台的功能和显示.下方为系统默认的后台界面图,为了便于下面的说明我对各个部分进行了一些标示.共A.B.C.D.E五个区域. 常用:A区域[顶部LOGO行]对应文件:/d ...

  4. 关于svn获取获取文件时 Unable to connect to a repository at URL"https://..."执行上下文错误:参数错误

    错误提示: 下面的六种解决方案都未能解决: 1.不提示输入用户名和密码,不管重装多少次都一样. 2.TortoiseSVN的setting->Saved Data->Authenticat ...

  5. [转帖]AVS音视频编解码技术了解

    AVS高清立体视频编码器 电视技术在经历了从黑白到彩色.从模拟到数字的技术变革之后正在酝酿另一场技术革命,从单纯观看二维场景的平面电视跨越到展现三维场景的立体电视3DTV.3DTV系统的核心问题之一是 ...

  6. oracle中创建一个用户,只能查看指定的视图,如何授权,创建别名

    1.create user A identified by Apassword,创建用户,用户名是A,密码是Apassword2.grant connect to A --授予connect权限3.g ...

  7. jrae源码解析(一)

    jare用java实现了论文<Semi-Supervised Recursive Autoencoders for Predicting Sentiment Distributions>中 ...

  8. python列表、字典与csv

    在日常数据分析时最常打交道的是csv文件和list,dict类型.涉及到的主要需求有: 将一个二重列表[[],[]]写入到csv文件中 从文本文件中读取返回为列表 将一字典写入到csv文件中 从csv ...

  9. 【NOI2006】最大获利

    [问题描述] 新的技术正冲击着手机通讯市场,对于各大运营商来说,这既是机遇,更是挑战.THU 集团旗下的CS&T 通讯公司在新一代通讯技术血战的前夜,需要做太多的准备工作,仅就站址选择一项,就 ...

  10. hdu 2460 poj 3694 (双联通+LCA)

    在给出的两个点上加一条边,求剩下桥的数量,,不会LCA在线,就用了最普通的,先Tarjan双联通缩点,然后将缩完的图建成一棵树,树的所有边就是桥了,如果在任意两点间加一条边的话,那么从两点到最近公共祖 ...