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?
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:
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With