1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
#[derive(Debug)]
pub struct Nuban {
bank_code: String,
serial_number: String,
}
impl Nuban {
pub fn new(bankcode: &str, serialnumber: &str) -> Nuban {
Nuban {
bank_code: String::from(bankcode),
serial_number: String::from(serialnumber),
}
}
pub fn full(&mut self) -> String {
let full_nuban =
self.bank_code.clone() + &self.serial_number.clone() + &self.check_digit().to_string();
full_nuban
}
pub fn check_digit(&self) -> u32 {
let mut sum: u32 = 3
* (&self.index(0) + &self.index(2) + &self.index(3) + &self.index(5) + &self.index(6)
+ &self.index(8) + &self.index(9) + &self.index(11)) as u32;
sum += 7 * (&self.index(1) + &self.index(4) + &self.index(7) + &self.index(10)) as u32;
10 - (sum % 10)
}
fn index(&self, index: usize) -> u8 {
let value = self.bank_code.clone() + &self.serial_number.clone();
let value = value.chars().nth(index);
match value {
Some(index_value) => {
if !index_value.is_numeric() {
panic!("Invalid characters detected in NUBAN digits.");
}
index_value.to_digit(10).unwrap_or_default() as u8
}
None => 0,
}
}
}
#[cfg(test)]
mod tests {
use super::Nuban;
#[test]
fn generated_nuban_is_valid() {
let mut nuban_no = Nuban::new("011", "000001457");
assert_eq!(nuban_no.full(), "0110000014579");
}
#[test]
fn nuban_check_digit_is_valid() {
let nuban_no = Nuban::new("011", "000001457");
assert_eq!(nuban_no.check_digit(), 9);
}
}