Merge pull request #261 from ouch-org/refac/optimize-current-dir-call
[ouch.git] / src / commands / list.rs
bloba5832d534e8a1d18d79aca5ed69d1a9d96751d72
1 use std::{
2     io::{self, BufReader, Read},
3     path::Path,
4 };
6 use fs_err as fs;
8 use crate::{
9     commands::warn_user_about_loading_zip_in_memory,
10     extension::CompressionFormat::{self, *},
11     list::{self, FileInArchive, ListOptions},
12     utils::user_wants_to_continue,
13     QuestionAction, QuestionPolicy, BUFFER_CAPACITY,
16 // File at input_file_path is opened for reading, example: "archive.tar.gz"
17 // formats contains each format necessary for decompression, example: [Gz, Tar] (in decompression order)
18 pub fn list_archive_contents(
19     archive_path: &Path,
20     formats: Vec<CompressionFormat>,
21     list_options: ListOptions,
22     question_policy: QuestionPolicy,
23 ) -> crate::Result<()> {
24     let reader = fs::File::open(&archive_path)?;
26     // Zip archives are special, because they require io::Seek, so it requires it's logic separated
27     // from decoder chaining.
28     //
29     // This is the only case where we can read and unpack it directly, without having to do
30     // in-memory decompression/copying first.
31     //
32     // Any other Zip decompression done can take up the whole RAM and freeze ouch.
33     if let &[Zip] = formats.as_slice() {
34         let zip_archive = zip::ZipArchive::new(reader)?;
35         let files = crate::archive::zip::list_archive(zip_archive);
36         list::list_files(archive_path, files, list_options)?;
38         return Ok(());
39     }
41     // Will be used in decoder chaining
42     let reader = BufReader::with_capacity(BUFFER_CAPACITY, reader);
43     let mut reader: Box<dyn Read + Send> = Box::new(reader);
45     // Grab previous decoder and wrap it inside of a new one
46     let chain_reader_decoder =
47         |format: &CompressionFormat, decoder: Box<dyn Read + Send>| -> crate::Result<Box<dyn Read + Send>> {
48             let decoder: Box<dyn Read + Send> = match format {
49                 Gzip => Box::new(flate2::read::GzDecoder::new(decoder)),
50                 Bzip => Box::new(bzip2::read::BzDecoder::new(decoder)),
51                 Lz4 => Box::new(lzzzz::lz4f::ReadDecompressor::new(decoder)?),
52                 Lzma => Box::new(xz2::read::XzDecoder::new(decoder)),
53                 Snappy => Box::new(snap::read::FrameDecoder::new(decoder)),
54                 Zstd => Box::new(zstd::stream::Decoder::new(decoder)?),
55                 Tar | Zip => unreachable!(),
56             };
57             Ok(decoder)
58         };
60     for format in formats.iter().skip(1).rev() {
61         reader = chain_reader_decoder(format, reader)?;
62     }
64     let files: Box<dyn Iterator<Item = crate::Result<FileInArchive>>> = match formats[0] {
65         Tar => Box::new(crate::archive::tar::list_archive(tar::Archive::new(reader))),
66         Zip => {
67             if formats.len() > 1 {
68                 warn_user_about_loading_zip_in_memory();
70                 // give user the option to continue decompressing after warning is shown
71                 if !user_wants_to_continue(archive_path, question_policy, QuestionAction::Decompression)? {
72                     return Ok(());
73                 }
74             }
76             let mut vec = vec![];
77             io::copy(&mut reader, &mut vec)?;
78             let zip_archive = zip::ZipArchive::new(io::Cursor::new(vec))?;
80             Box::new(crate::archive::zip::list_archive(zip_archive))
81         }
82         Gzip | Bzip | Lz4 | Lzma | Snappy | Zstd => {
83             panic!("Not an archive! This should never happen, if it does, something is wrong with `CompressionFormat::is_archive()`. Please report this error!");
84         }
85     };
86     list::list_files(archive_path, files, list_options)?;
87     Ok(())