我正在使用默认的Yesod脚手架项目
我创建了一个页面,显示了一个上传文件的简单表单
(该表单可能会在客户端上使用Javascript创建。)
为了简洁起见,表单只有一个文件输入:
<form method="post" action=@{UploadR}>
<input type="file" name="myfile">
<button type="submit">
我的目标是处理表单数据,然后将文件上传到web服务
我处理表单没有问题,我关心的是与web服务的交互
例如,给定以下Yesod处理程序:
postUploadR :: Handler Html
postUploadR = do
mgr <- fmap httpManager getYesod
fi <- runInputPost $ ireq fileField "myfile"
let fSource = fileSource fi
fName = fileName fi
req <- parseUrl "http://webservice/upload"
let areq = req { method = methodPost
, requestBody = requestBodySourceChunked fSource
}
res <- httpLbs areq mgr
defaultLayout $ do
setTitle "file uploaded"
[whamlet|
<h3> Success
<p> You uploaded #{fName}.
|]
Web服务返回错误:fail post content-length
,但其他一切都按预期工作。也许服务器不支持分块请求正文?
我认为您对分块请求体的猜测是正确的。你需要做的是:
- 将上传的内容流式传输到临时文件中
- 获取该文件的大小
- 使用
requestBodySource
并提供文件长度及其内容
幸运的是,sinkCacheLength函数可以非常容易地处理步骤(1)和(2)。你最终会得到这样的东西:
(fSize, fSource) <- fileSource fi $$ sinkCacheLength