Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feature/errors #58

Open
wants to merge 2 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ repository = "https://github.com/muja/unrar.rs"
regex = "1"
bitflags = "1"
widestring = "1"
thiserror = "1.0.50"

[dependencies.unrar_sys]
path = "unrar_sys"
Expand Down
5 changes: 2 additions & 3 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -100,15 +100,14 @@ For more sophisticated examples, please look inside the `examples/` folder.
Here's what a function that returns the first content of a file could look like:

```rust
fn first_file_content<P: AsRef<Path>>(path: P) -> UnrarResult<Vec<u8>> {
let archive = Archive::new(&path).open_for_processing()?; // cursor: before header
fn first_file_content<P: AsRef<Path>>(path: P) -> Result<Vec<u8>, unrar::Error> {
let archive = unrar::Archive::new(&path).open_for_processing()?; // cursor: before header
let archive = archive.read_header()?.expect("empty archive"); // cursor: before file
dbg!(&archive.entry().filename);
let (data, _rest) = archive.read()?; // cursor: before header
Ok(data)
}
# use std::path::Path;
# use unrar::{Archive, UnrarResult};
#
# let data = first_file_content("data/version.rar").unwrap();
# assert_eq!(std::str::from_utf8(&data), Ok("unrar-0.4.0"));
Expand Down
7 changes: 3 additions & 4 deletions examples/basic_extract.rs
Original file line number Diff line number Diff line change
@@ -1,10 +1,9 @@
use unrar::Archive;

fn main() -> Result<(), Box<dyn std::error::Error>> {
let mut archive =
Archive::new("../archive.rar")
.open_for_processing()
.unwrap();
let args = std::env::args();
let file = args.skip(1).next().unwrap_or("archive.rar".to_owned());
let mut archive = Archive::new(&file).open_for_processing().unwrap();
while let Some(header) = archive.read_header()? {
println!(
"{} bytes: {}",
Expand Down
2 changes: 1 addition & 1 deletion examples/read_named.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
use unrar::{Archive, UnrarResult};
use unrar::{Archive, Result as UnrarResult};

fn main() -> UnrarResult<()> {
// Basic args parsing
Expand Down
57 changes: 30 additions & 27 deletions src/archive.rs
Original file line number Diff line number Diff line change
Expand Up @@ -323,21 +323,25 @@ impl<'a> Archive<'a> {
///
/// See also: [`Process`]
///
/// # Panics
/// # Errors
///
/// Panics if `self.filename` contains nul values.
pub fn open_for_processing(self) -> UnrarResult<OpenArchive<Process, CursorBeforeHeader>> {
/// - `NulError` if `self.filename` or `self.password` contains nul values.
/// - `RarError` if there was an error opening/reading/decoding the archive.
///
pub fn open_for_processing(self) -> Result<OpenArchive<Process, CursorBeforeHeader>> {
self.open(None)
}

/// Opens the underlying archive for listing its entries, i.e. the payloads are skipped automatically.
///
/// See also: [`List`]
///
/// # Panics
/// # Errors
///
/// - `NulError` if `self.filename` or `self.password` contains nul values.
/// - `RarError` if there was an error opening/reading/decoding the archive.
///
/// Panics if `self.filename` contains nul values.
pub fn open_for_listing(self) -> UnrarResult<OpenArchive<List, CursorBeforeHeader>> {
pub fn open_for_listing(self) -> Result<OpenArchive<List, CursorBeforeHeader>> {
self.open(None)
}

Expand All @@ -348,23 +352,20 @@ impl<'a> Archive<'a> {
///
/// See also: [`ListSplit`]
///
/// # Panics
/// # Errors
///
/// Panics if `self.filename` contains nul values.

pub fn open_for_listing_split(self) -> UnrarResult<OpenArchive<ListSplit, CursorBeforeHeader>> {
/// - `NulError` if `self.filename` or `self.password` contains nul values.
/// - `RarError` if there was an error opening/reading/decoding the archive.
///
pub fn open_for_listing_split(self) -> Result<OpenArchive<ListSplit, CursorBeforeHeader>> {
self.open(None)
}

/// Opens the underlying archive with the provided parameters.
///
/// # Panics
///
/// Panics if `path` contains nul values.
fn open<M: OpenMode>(
self,
recover: Option<&mut Option<OpenArchive<M, CursorBeforeHeader>>>,
) -> UnrarResult<OpenArchive<M, CursorBeforeHeader>> {
) -> Result<OpenArchive<M, CursorBeforeHeader>> {
OpenArchive::new(&self.filename, self.password, recover)
}

Expand All @@ -378,8 +379,8 @@ impl<'a> Archive<'a> {
/// # Example: I don't care if there was a recoverable error
///
/// ```no_run
/// # use unrar::{Archive, List, UnrarResult};
/// # fn x() -> UnrarResult<()> {
/// # use unrar::{Archive, List, Result};
/// # fn x() -> Result<()> {
/// let mut open_archive = Archive::new("file").break_open::<List>(None)?;
/// // use open_archive
/// # Ok(())
Expand All @@ -389,8 +390,8 @@ impl<'a> Archive<'a> {
/// # Example: I want to know if there was a recoverable error
///
/// ```no_run
/// # use unrar::{Archive, List, UnrarResult};
/// # fn x() -> UnrarResult<()> {
/// # use unrar::{Archive, List, Result};
/// # fn x() -> Result<()> {
/// let mut possible_error = None;
/// let mut open_archive = Archive::new("file").break_open::<List>(Some(&mut possible_error))?;
/// // check the error, e.g.:
Expand All @@ -400,21 +401,23 @@ impl<'a> Archive<'a> {
/// # }
/// ```
///
/// # Panics
/// # Errors
///
/// - `NulError` if `self.filename` or `self.password` contains nul values.
/// - `RarError` if there was an error opening/reading/decoding the archive.
///
/// Panics if `path` contains nul values.
pub fn break_open<M: OpenMode>(
self,
error: Option<&mut Option<UnrarError>>,
) -> UnrarResult<OpenArchive<M, CursorBeforeHeader>> {
error: Option<&mut Option<rar::Error>>,
) -> Result<OpenArchive<M, CursorBeforeHeader>> {
let mut recovered = None;
self.open(Some(&mut recovered))
.or_else(|x| match recovered {
Some(archive) => {
error.map(|error| *error = Some(x));
.or_else(|e| match (recovered, e) {
(Some(archive), Error::RarError(e)) => {
error.map(|error| *error = Some(e));
Ok(archive)
}
None => Err(x),
(_, _) => Err(e),
})
}
}
Expand Down
152 changes: 18 additions & 134 deletions src/error.rs
Original file line number Diff line number Diff line change
@@ -1,146 +1,30 @@
#![allow(missing_docs)]

use super::*;
use std::error;
use std::ffi;
use std::fmt;
use std::result::Result;
pub(crate) mod nul;
pub(crate) mod rar;

use thiserror::Error;

#[derive(PartialEq, Eq, Debug, Clone, Copy)]
#[repr(i32)]
pub enum Code {
Success = native::ERAR_SUCCESS,
EndArchive = native::ERAR_END_ARCHIVE,
NoMemory = native::ERAR_NO_MEMORY,
BadData = native::ERAR_BAD_DATA,
BadArchive = native::ERAR_BAD_ARCHIVE,
UnknownFormat = native::ERAR_UNKNOWN_FORMAT,
EOpen = native::ERAR_EOPEN,
ECreate = native::ERAR_ECREATE,
EClose = native::ERAR_ECLOSE,
ERead = native::ERAR_EREAD,
EWrite = native::ERAR_EWRITE,
SmallBuf = native::ERAR_SMALL_BUF,
Unknown = native::ERAR_UNKNOWN,
MissingPassword = native::ERAR_MISSING_PASSWORD,
// From the UnRARDLL docs:
// When attempting to unpack a reference record (see RAR -oi switch),
// source file for this reference was not found.
// Entire archive needs to be unpacked to properly create file references.
// This error is returned when attempting to unpack the reference
// record without its source file.
EReference = native::ERAR_EREFERENCE,
BadPassword = native::ERAR_BAD_PASSWORD,
}

#[derive(Debug, PartialEq, Eq, Clone, Copy)]
pub enum When {
Open,
Read,
Process,
}

impl Code {
pub fn from(code: i32) -> Option<Self> {
use Code::*;
match code {
native::ERAR_SUCCESS => Some(Success),
native::ERAR_END_ARCHIVE => Some(EndArchive),
native::ERAR_NO_MEMORY => Some(NoMemory),
native::ERAR_BAD_DATA => Some(BadData),
native::ERAR_BAD_ARCHIVE => Some(BadArchive),
native::ERAR_UNKNOWN_FORMAT => Some(UnknownFormat),
native::ERAR_EOPEN => Some(EOpen),
native::ERAR_ECREATE => Some(ECreate),
native::ERAR_ECLOSE => Some(EClose),
native::ERAR_EREAD => Some(ERead),
native::ERAR_EWRITE => Some(EWrite),
native::ERAR_SMALL_BUF => Some(SmallBuf),
native::ERAR_UNKNOWN => Some(Unknown),
native::ERAR_MISSING_PASSWORD => Some(MissingPassword),
native::ERAR_EREFERENCE => Some(EReference),
native::ERAR_BAD_PASSWORD => Some(BadPassword),
_ => None,
}
}
}

#[derive(PartialEq)]
pub struct UnrarError {
pub code: Code,
pub when: When,
}

impl std::error::Error for UnrarError {}

impl fmt::Debug for UnrarError {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{:?}@{:?}", self.code, self.when)?;
write!(f, " ({})", self)
}
}
pub type Result<T, E = Error> = std::result::Result<T, E>;
pub use rar::Error as RarError;
pub use nul::Error as NulError;

impl fmt::Display for UnrarError {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
use self::Code::*;
use self::When::*;
match (self.code, self.when) {
(BadData, Open) => write!(f, "Archive header damaged"),
(BadData, Read) => write!(f, "File header damaged"),
(BadData, Process) => write!(f, "File CRC error"),
(UnknownFormat, Open) => write!(f, "Unknown encryption"),
(EOpen, Process) => write!(f, "Could not open next volume"),
(UnknownFormat, _) => write!(f, "Unknown archive format"),
(EOpen, _) => write!(f, "Could not open archive"),
(NoMemory, _) => write!(f, "Not enough memory"),
(BadArchive, _) => write!(f, "Not a RAR archive"),
(ECreate, _) => write!(f, "Could not create file"),
(EClose, _) => write!(f, "Could not close file"),
(ERead, _) => write!(f, "Read error"),
(EWrite, _) => write!(f, "Write error"),
(SmallBuf, _) => write!(f, "Archive comment was truncated to fit to buffer"),
(MissingPassword, _) => write!(f, "Password for encrypted archive not specified"),
(EReference, _) => write!(f, "Cannot open file source for reference record"),
(BadPassword, _) => write!(f, "Wrong password was specified"),
(Unknown, _) => write!(f, "Unknown error"),
(EndArchive, _) => write!(f, "Archive end"),
(Success, _) => write!(f, "Success"),
}
}
}

impl UnrarError {
pub fn from(code: Code, when: When) -> Self {
UnrarError { code, when }
}
}

pub type UnrarResult<T> = Result<T, UnrarError>;

#[derive(Debug)]
pub struct NulError(usize);

impl fmt::Display for NulError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "nul value found at position: {}", self.0)
}
}

impl error::Error for NulError {
fn description(&self) -> &str {
"nul value found"
}
#[derive(Error, Debug, PartialEq, Eq, Clone, Copy)]
pub enum Error {
#[error(transparent)]
RarError(#[from] rar::Error),
#[error(transparent)]
NulError(#[from] nul::Error),
}

impl<C> From<widestring::error::ContainsNul<C>> for NulError {
fn from(e: widestring::error::ContainsNul<C>) -> NulError {
NulError(e.nul_position())
impl<C> From<widestring::error::ContainsNul<C>> for Error {
fn from(e: widestring::error::ContainsNul<C>) -> Error {
nul::Error(e.nul_position()).into()
}
}

impl From<ffi::NulError> for NulError {
fn from(e: ffi::NulError) -> NulError {
NulError(e.nul_position())
impl From<std::ffi::NulError> for Error {
fn from(e: std::ffi::NulError) -> Error {
nul::Error(e.nul_position()).into()
}
}
5 changes: 5 additions & 0 deletions src/error/nul.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
use thiserror::Error;

#[derive(PartialEq, Eq, Error, Debug, Clone, Copy)]
#[error("unexpected NUL at {0}")]
pub struct Error(pub usize);
Loading
Loading