2018-02-22 09:09:53 +03:00
|
|
|
// errors2.rs
|
2023-05-29 20:39:08 +03:00
|
|
|
//
|
2016-06-21 17:07:21 +03:00
|
|
|
// Say we're writing a game where you can buy items with tokens. All items cost
|
|
|
|
// 5 tokens, and whenever you purchase items there is a processing fee of 1
|
2023-05-29 20:39:08 +03:00
|
|
|
// token. A player of the game will type in how many items they want to buy, and
|
2023-10-13 22:47:38 +03:00
|
|
|
// the `total_cost` function will calculate the total cost of the items. Since
|
2023-05-29 20:39:08 +03:00
|
|
|
// the player typed in the quantity, though, we get it as a string-- and they
|
|
|
|
// might have typed anything, not just numbers!
|
|
|
|
//
|
2016-06-21 17:07:21 +03:00
|
|
|
// Right now, this function isn't handling the error case at all (and isn't
|
2023-05-29 20:39:08 +03:00
|
|
|
// handling the success case properly either). What we want to do is: if we call
|
2023-08-14 22:49:28 +03:00
|
|
|
// the `total_cost` function on a string that is not a number, that function
|
|
|
|
// will return a `ParseIntError`, and in that case, we want to immediately
|
|
|
|
// return that error from our function and not try to multiply and add.
|
2023-05-29 20:39:08 +03:00
|
|
|
//
|
|
|
|
// There are at least two ways to implement this that are both correct-- but one
|
|
|
|
// is a lot shorter!
|
|
|
|
//
|
|
|
|
// Execute `rustlings hint errors2` or use the `hint` watch subcommand for a
|
|
|
|
// hint.
|
2016-06-21 17:07:21 +03:00
|
|
|
|
2019-11-11 15:38:24 +03:00
|
|
|
|
2016-06-21 17:07:21 +03:00
|
|
|
use std::num::ParseIntError;
|
|
|
|
|
|
|
|
pub fn total_cost(item_quantity: &str) -> Result<i32, ParseIntError> {
|
|
|
|
let processing_fee = 1;
|
|
|
|
let cost_per_item = 5;
|
2024-02-02 05:58:03 +03:00
|
|
|
// Same as the match expression below
|
|
|
|
let qty = item_quantity.parse::<i32>()?;
|
|
|
|
|
|
|
|
// match item_quantity.parse::<i32>() {
|
|
|
|
// Ok(quantity) => Ok(quantity * cost_per_item + processing_fee),
|
|
|
|
// Err(e) => Err(e),
|
|
|
|
// }
|
2016-06-21 17:07:21 +03:00
|
|
|
|
2024-02-02 05:58:03 +03:00
|
|
|
// I tried to use the if let statement but that's for Pattern matching
|
|
|
|
// This is not patter matching because it's looking for conditional checks.
|
2016-06-21 17:07:21 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn item_quantity_is_a_valid_number() {
|
2019-05-22 14:48:32 +03:00
|
|
|
assert_eq!(total_cost("34"), Ok(171));
|
2016-06-21 17:07:21 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn item_quantity_is_an_invalid_number() {
|
|
|
|
assert_eq!(
|
|
|
|
total_cost("beep boop").unwrap_err().to_string(),
|
|
|
|
"invalid digit found in string"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|