【发布时间】:2018-11-17 11:41:21
【问题描述】:
我正在尝试创建一个简单的函数,该函数使用 hyper 将远程文件下载到本地文件路径。我也需要文件写入是异步的(在我的情况下,我为此使用tokio_fs)。代码如下:
// Parts of the code were omitted, see the playground for full source code
pub fn download_file(
uri: Uri,
file_location: &Path,
) -> Box<Future<Item = (), Error = DownloadFileError>> {
let temp_dir_path = tempfile::tempdir().unwrap().into_path();
let file_name = match file_location.file_name() {
Some(file_name) => file_name,
None => return Box::new(futures::failed(DownloadFileError::IncorrectFilePath)),
};
let temp_filepath = temp_dir_path.join(&file_name);
let connector = HttpsConnector::new(2).unwrap();
let client: Client<_, Body> = Client::builder().build(connector);
let response_future = client
.get(uri)
.map_err(|err| DownloadFileError::GetRequest(err));
let create_file_future =
File::create(temp_filepath).map_err(|err| DownloadFileError::CreateFile(err));
Box::new(
response_future
.join(create_file_future)
.and_then(move |(res, file)| {
res.into_body()
.map_err(|e| DownloadFileError::GetRequest(e))
.for_each(move |chunk| {
io::write_all(file, chunk)
.map(|_| ())
.map_err(|_| DownloadFileError::FileWrite)
})
}),
)
}
但是,我收到以下错误:
error[E0507]: cannot move out of captured outer variable in an `FnMut` closure
--> src/lib.rs:79:39
|
75 | .and_then(move |(res, file)| {
| ---- captured outer variable
...
79 | io::write_all(file, chunk)
| ^^^^ cannot move out of captured outer variable in an `FnMut` closure
从概念上讲,我理解错误的含义:由于 FnMut 通过可变引用捕获变量,因此我无法移动捕获的变量。但是,我不明白如何在给出的示例中解决此问题,因为我需要将流写入Join 未来返回的文件。
Write trait 中的 write_all 方法在这里可以工作,因为它将文件作为可变引用,但问题是它在同一个线程上进行写入。
【问题讨论】:
-
我认为使用
for_each()、into_body()返回一个选项没有意义,所以只需使用map()。但也许我完全错了,我对超级/期货一无所知 -
据我了解,他们两个都使用
write_all同步写入文件。就我而言,io::write_all也是异步完成的。
标签: rust future borrow-checker rust-tokio hyper