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
use std::cmp::Ordering;
use std::fmt;
#[derive(Eq, PartialEq, Clone, Default, Hash)]
pub struct StringPair {
buffer: String,
split: usize,
}
impl<S1: Into<String>, S2: AsRef<str>> From<(S1, S2)> for StringPair {
#[inline]
fn from((s1, s2): (S1, S2)) -> StringPair {
StringPair::new(s1, s2)
}
}
impl StringPair {
pub fn new<S1: Into<String>, S2: AsRef<str>>(s1: S1, s2: S2) -> StringPair {
let s1 = s1.into();
let s2 = s2.as_ref();
let split = s1.len();
let mut buffer = s1;
buffer.push_str(s2.as_ref());
StringPair { buffer, split }
}
#[inline]
pub fn from_raw<S: Into<String>>(string: S, split: usize) -> StringPair {
let buffer = string.into();
assert!(split <= buffer.len(), "StrPair: split index was out of bounds");
StringPair { buffer, split }
}
#[inline]
pub unsafe fn from_raw_unchecked<S: Into<String>>(string: S, split: usize) -> StringPair {
let buffer = string.into();
StringPair { buffer, split }
}
#[inline]
pub fn left(&self) -> &str {
unsafe {
self.buffer.slice_unchecked(0, self.split)
}
}
#[inline]
pub fn right(&self) -> &str {
unsafe {
self.buffer.slice_unchecked(self.split, self.buffer.len())
}
}
}
impl PartialOrd for StringPair {
fn partial_cmp(&self, rhs: &StringPair) -> Option<Ordering> {
Some(self.left().cmp(rhs.left()).then_with(|| self.right().cmp(rhs.right())))
}
}
impl Ord for StringPair {
fn cmp(&self, rhs: &StringPair) -> Ordering {
self.left().cmp(rhs.left()).then_with(|| self.right().cmp(rhs.right()))
}
}
impl fmt::Debug for StringPair {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.debug_tuple("StringPair::new")
.field(&self.left().to_owned())
.field(&self.right().to_owned())
.finish()
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn raw_left() {
assert_eq!(StringPair::from_raw("helloworld", 5).left(), "hello");
}
#[test]
fn raw_right() {
assert_eq!(StringPair::from_raw("helloworld", 5).right(), "world");
}
quickcheck! {
fn partial_ord(s1: String, s2: String, s3: String, s4: String) -> bool {
let lhs = StringPair::new(&*s1, &s2);
let rhs = StringPair::new(&*s3, &s4);
lhs.partial_cmp(&rhs) == (&s1, &s2).partial_cmp(&(&s3, &s4))
}
fn ord(s1: String, s2: String, s3: String, s4: String) -> bool {
let lhs = StringPair::new(&*s1, &s2);
let rhs = StringPair::new(&*s3, &s4);
lhs.cmp(&rhs) == (&s1, &s2).cmp(&(&s3, &s4))
}
fn left(s1: String, s2: String) -> bool {
let pair = StringPair::new(&*s1, &s2);
pair.left() == s1
}
fn right(s1: String, s2: String) -> bool {
let pair = StringPair::new(&*s1, &s2);
pair.right() == s2
}
}
}