Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to write a function that returns Vec<Path>?

Tags:

rust

I'm reading the documentation and trying to write some basic file I/O code as a vehicle to help me learn Rust.

The following doesn't compile:

use std::fs;
use std::io;
use std::path::Path;

pub fn read_filenames_from_dir<P>(path: P) -> Result<Vec<Path>, io::Error>
where
    P: AsRef<Path>,
{
    let paths = try!(fs::read_dir(path));
    Ok(paths.unwrap())
}

With the compilation error:

error[E0277]: the trait bound `[u8]: std::marker::Sized` is not satisfied in `std::path::Path`
  --> src/main.rs:5:1
   |
5  | / pub fn read_filenames_from_dir<P>(path: P) -> Result<Vec<Path>, io::Error>
6  | | where
7  | |     P: AsRef<Path>,
8  | | {
9  | |     let paths = try!(fs::read_dir(path));
10 | |     Ok(paths.unwrap())
11 | | }
   | |_^ `[u8]` does not have a constant size known at compile-time
   |
   = help: within `std::path::Path`, the trait `std::marker::Sized` is not implemented for `[u8]`
   = note: required because it appears within the type `std::path::Path`
   = note: required by `std::vec::Vec`

How should I write this function to return the collection of Paths inside the Path that's passed in?

like image 359
Gavin Hope Avatar asked May 25 '16 13:05

Gavin Hope


1 Answers

You don't. Path is a type that has no size, and is only usable through indirection (such as &Path or Box<Path>). In this sense, it is like the type str or [u8] — neither can be directly used, only indirectly.

What you probably want is a PathBuf, which represents an owned path. It is the equivalent of String for &str and Vec<u8> for &[u8].

After changing the return type, you have to properly map the results of the iterator to create your desired type:

use std::{
    fs, io,
    path::{Path, PathBuf},
};

pub fn read_filenames_from_dir<P>(path: P) -> Result<Vec<PathBuf>, io::Error>
where
    P: AsRef<Path>,
{
    fs::read_dir(path)?
        .into_iter()
        .map(|x| x.map(|entry| entry.path()))
        .collect()
}

fn main() {
    println!("{:?}", read_filenames_from_dir("/etc"));
}

See also:

  • What does “`str` does not have a constant size known at compile-time” mean, and what's the simplest way to fix it?
like image 171
Shepmaster Avatar answered Oct 09 '22 23:10

Shepmaster