2014-06-08 21:51:22 +00:00
|
|
|
/*
|
|
|
|
* This file is part of the uutils coreutils package.
|
|
|
|
*
|
|
|
|
* (c) Rolf Morel <rolfmorel@gmail.com>
|
|
|
|
*
|
|
|
|
* For the full copyright and license information, please view the LICENSE
|
|
|
|
* file that was distributed with this source code.
|
|
|
|
*/
|
|
|
|
|
|
|
|
use std;
|
|
|
|
|
|
|
|
#[deriving(PartialEq,Eq,PartialOrd,Ord,Show)]
|
|
|
|
pub struct Range {
|
|
|
|
pub low: uint,
|
|
|
|
pub high: uint,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl std::from_str::FromStr for Range {
|
|
|
|
fn from_str(s: &str) -> Option<Range> {
|
|
|
|
use std::uint::MAX;
|
|
|
|
|
|
|
|
let mut parts = s.splitn('-', 1);
|
|
|
|
|
|
|
|
match (parts.next(), parts.next()) {
|
|
|
|
(Some(nm), None) => {
|
|
|
|
from_str::<uint>(nm).filtered(|nm| *nm > 0)
|
2014-06-25 17:17:38 +00:00
|
|
|
.map(|nm| Range { low: nm, high: nm })
|
2014-06-08 21:51:22 +00:00
|
|
|
}
|
|
|
|
(Some(n), Some(m)) if m.len() == 0 => {
|
|
|
|
from_str::<uint>(n).filtered(|low| *low > 0)
|
2014-06-25 17:17:38 +00:00
|
|
|
.map(|low| Range { low: low, high: MAX })
|
2014-06-08 21:51:22 +00:00
|
|
|
}
|
|
|
|
(Some(n), Some(m)) if n.len() == 0 => {
|
|
|
|
from_str::<uint>(m).filtered(|high| *high >= 1)
|
2014-06-25 17:17:38 +00:00
|
|
|
.map(|high| Range { low: 1, high: high })
|
2014-06-08 21:51:22 +00:00
|
|
|
}
|
|
|
|
(Some(n), Some(m)) => {
|
|
|
|
match (from_str::<uint>(n), from_str::<uint>(m)) {
|
|
|
|
(Some(low), Some(high)) if low > 0 && low <= high => {
|
2014-06-25 17:17:38 +00:00
|
|
|
Some(Range { low: low, high: high })
|
2014-06-08 21:51:22 +00:00
|
|
|
}
|
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => unreachable!()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Range {
|
|
|
|
pub fn from_list(list: &str) -> Result<Vec<Range>, String> {
|
|
|
|
use std::cmp::max;
|
|
|
|
|
|
|
|
let mut ranges = vec!();
|
|
|
|
|
|
|
|
for item in list.split(',') {
|
|
|
|
match from_str::<Range>(item) {
|
|
|
|
Some(range_item) => ranges.push(range_item),
|
|
|
|
None => return Err(format!("range '{}' was invalid", item))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ranges.sort();
|
|
|
|
|
|
|
|
// merge overlapping ranges
|
|
|
|
for i in range(0, ranges.len()) {
|
|
|
|
let j = i + 1;
|
|
|
|
|
|
|
|
while j < ranges.len() && ranges.get(j).low <= ranges.get(i).high {
|
|
|
|
let j_high = ranges.remove(j).unwrap().high;
|
|
|
|
ranges.get_mut(i).high = max(ranges.get(i).high, j_high);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(ranges)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn complement(ranges: &Vec<Range>) -> Vec<Range> {
|
|
|
|
use std::uint;
|
|
|
|
|
|
|
|
let mut complements = Vec::with_capacity(ranges.len() + 1);
|
|
|
|
|
|
|
|
if ranges.len() > 0 && ranges.get(0).low > 1 {
|
2014-06-25 17:17:38 +00:00
|
|
|
complements.push(Range { low: 1, high: ranges.get(0).low - 1 });
|
2014-06-08 21:51:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let mut ranges_iter = ranges.iter().peekable();
|
|
|
|
loop {
|
|
|
|
match (ranges_iter.next(), ranges_iter.peek()) {
|
|
|
|
(Some(left), Some(right)) => {
|
|
|
|
if left.high + 1 != right.low {
|
2014-06-25 17:17:38 +00:00
|
|
|
complements.push(Range {
|
|
|
|
low: left.high + 1,
|
|
|
|
high: right.low - 1
|
|
|
|
});
|
2014-06-08 21:51:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
(Some(last), None) => {
|
|
|
|
if last.high < uint::MAX {
|
2014-06-25 17:17:38 +00:00
|
|
|
complements.push(Range {
|
|
|
|
low: last.high + 1,
|
|
|
|
high: uint::MAX
|
|
|
|
});
|
2014-06-08 21:51:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
complements
|
|
|
|
}
|