mirror of
https://github.com/uutils/coreutils
synced 2025-01-05 17:59:00 +00:00
335 lines
12 KiB
Rust
335 lines
12 KiB
Rust
#![crate_id(name="nl", vers="1.0.0", author="Tobias Schottdorf")]
|
|
#![feature(macro_rules)]
|
|
/*
|
|
* This file is part of the uutils coreutils package.
|
|
*
|
|
* (c) Tobias Bohumir Schottdorf <tobias.schottdorf@gmail.com>
|
|
*
|
|
* For the full copyright and license information, please view the LICENSE
|
|
* file that was distributed with this source code.
|
|
*
|
|
*/
|
|
|
|
#![feature(phase)]
|
|
#[phase(plugin)]
|
|
extern crate regex_macros;
|
|
extern crate regex;
|
|
extern crate getopts;
|
|
|
|
use std::os;
|
|
use std::io::{stdin};
|
|
use std::io::BufferedReader;
|
|
use std::io::fs::File;
|
|
use std::path::Path;
|
|
use getopts::{optopt, optflag, getopts, usage, OptGroup};
|
|
|
|
#[path="../common/util.rs"]
|
|
mod util;
|
|
mod helper;
|
|
|
|
static NAME: &'static str = "nl";
|
|
static USAGE: &'static str = "nl [OPTION]... [FILE]...";
|
|
// A regular expression matching everything.
|
|
static REGEX_DUMMY: &'static regex::Regex = ®ex!(r".?");
|
|
|
|
// Settings store options used by nl to produce its output.
|
|
struct Settings {
|
|
// The variables corresponding to the options -h, -b, and -f.
|
|
header_numbering: NumberingStyle,
|
|
body_numbering: NumberingStyle,
|
|
footer_numbering: NumberingStyle,
|
|
// The variable corresponding to -d
|
|
section_delimiter: [char, ..2],
|
|
// The variables corresponding to the options -v, -i, -l, -w.
|
|
starting_line_number: u64,
|
|
line_increment: u64,
|
|
join_blank_lines: u64,
|
|
number_width: uint, // Used with String::from_char, hence uint.
|
|
// The format of the number and the (default value for)
|
|
// renumbering each page.
|
|
number_format: NumberFormat,
|
|
renumber: bool,
|
|
// The string appended to each line number output.
|
|
number_separator: String
|
|
}
|
|
|
|
// NumberingStyle stores which lines are to be numberd.
|
|
// The possible options are:
|
|
// 1. Number all lines
|
|
// 2. Number only nonempty lines
|
|
// 3. Don't number any lines at all
|
|
// 4. Number all lines that match a basic regular expression.
|
|
enum NumberingStyle {
|
|
NumberForAll,
|
|
NumberForNonEmpty,
|
|
NumberForNone,
|
|
NumberForRegularExpression(regex::Regex)
|
|
}
|
|
|
|
// NumberFormat specifies how line numbers are output within their allocated
|
|
// space. They are justified to the left or right, in the latter case with
|
|
// the option of having all unused space to its left turned into leading zeroes.
|
|
enum NumberFormat {
|
|
Left,
|
|
Right,
|
|
RightZero,
|
|
}
|
|
|
|
#[allow(dead_code)]
|
|
fn main () {
|
|
os::set_exit_status(uumain(os::args()));
|
|
}
|
|
|
|
pub fn uumain(args: Vec<String>) -> int {
|
|
let possible_options = [
|
|
optopt("b", "body-numbering", "use STYLE for numbering body lines", "STYLE"),
|
|
optopt("d", "section-delimiter", "use CC for separating logical pages", "CC"),
|
|
optopt("f", "footer-numbering", "use STYLE for numbering footer lines", "STYLE"),
|
|
optopt("h", "header-numbering", "use STYLE for numbering header lines", "STYLE"),
|
|
optopt("i", "line-increment", "line number increment at each line", ""),
|
|
optopt("l", "join-blank-lines", "group of NUMBER empty lines counted as one", "NUMBER"),
|
|
optopt("n", "number-format", "insert line numbers according to FORMAT", "FORMAT"),
|
|
optflag("p", "no-renumber", "do not reset line numbers at logical pages"),
|
|
optopt("s", "number-separator", "add STRING after (possible) line number", "STRING"),
|
|
optopt("v", "starting-line-number", "first line number on each logical page", "NUMBER"),
|
|
optopt("w", "number-width", "use NUMBER columns for line numbers", "NUMBER"),
|
|
optflag("", "help", "display this help and exit"),
|
|
optflag("V", "version", "version"),
|
|
];
|
|
|
|
// A mutable settings object, initialized with the defaults.
|
|
let mut settings = Settings {
|
|
header_numbering: NumberForNone,
|
|
body_numbering: NumberForAll,
|
|
footer_numbering: NumberForNone,
|
|
section_delimiter: ['\\', ':'],
|
|
starting_line_number: 1,
|
|
line_increment: 1,
|
|
join_blank_lines: 1,
|
|
number_width: 6,
|
|
number_format: Right,
|
|
renumber: true,
|
|
number_separator: String::from_str("\t"),
|
|
};
|
|
|
|
let given_options = match getopts(args.tail(), possible_options) {
|
|
Ok (m) => { m }
|
|
Err(f) => {
|
|
show_error!("{}", f);
|
|
print_usage(possible_options);
|
|
return 1
|
|
}
|
|
};
|
|
|
|
if given_options.opt_present("help") {
|
|
print_usage(possible_options);
|
|
return 0;
|
|
}
|
|
if given_options.opt_present("version") { version(); return 0; }
|
|
|
|
// Update the settings from the command line options, and terminate the
|
|
// program if some options could not successfully be parsed.
|
|
let parse_errors = helper::parse_options(&mut settings, &given_options);
|
|
if parse_errors.len() > 0 {
|
|
show_error!("Invalid arguments supplied.");
|
|
for message in parse_errors.iter() {
|
|
println!("{}", message.as_slice());
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
let files = given_options.free;
|
|
let mut read_stdin = files.is_empty();
|
|
|
|
for file in files.iter() {
|
|
if file.as_slice() == "-" {
|
|
// If both file names and '-' are specified, we choose to treat first all
|
|
// regular files, and then read from stdin last.
|
|
read_stdin = true;
|
|
continue
|
|
}
|
|
let path = Path::new(file.as_slice());
|
|
let reader = File::open(&path).unwrap();
|
|
let mut buffer = BufferedReader::new(reader);
|
|
nl(&mut buffer, &settings);
|
|
}
|
|
|
|
if read_stdin {
|
|
let mut buffer = BufferedReader::new(stdin());
|
|
nl(&mut buffer, &settings);
|
|
}
|
|
0
|
|
}
|
|
|
|
// nl implements the main functionality for an individual buffer.
|
|
fn nl<T: Reader> (reader: &mut BufferedReader<T>, settings: &Settings) {
|
|
let mut line_no = settings.starting_line_number;
|
|
// The current line number's width as a string. Using to_str is inefficient
|
|
// but since we only do it once, it should not hurt.
|
|
let mut line_no_width = line_no.to_str().len();
|
|
let line_no_width_initial = line_no_width;
|
|
// Stores the smallest integer with one more digit than line_no, so that
|
|
// when line_no >= line_no_threshold, we need to use one more digit.
|
|
let mut line_no_threshold = std::num::pow(10, line_no_width) as u64;
|
|
let mut empty_line_count: u64 = 0;
|
|
let fill_char = match settings.number_format {
|
|
RightZero => '0',
|
|
_ => ' '
|
|
};
|
|
// Initially, we use the body's line counting settings
|
|
let mut regex_filter = match settings.body_numbering {
|
|
NumberForRegularExpression(ref re) => re,
|
|
_ => REGEX_DUMMY,
|
|
};
|
|
let mut line_filter = pass_regex;
|
|
for mut l in reader.lines().map(|r| r.unwrap()) {
|
|
// Sanitize the string. We want to print the newline ourselves.
|
|
if l.as_slice().chars().rev().next().unwrap() == '\n' {
|
|
l.pop_char();
|
|
}
|
|
// Next we iterate through the individual chars to see if this
|
|
// is one of the special lines starting a new "section" in the
|
|
// document.
|
|
let line = l.as_slice();
|
|
let mut odd = false;
|
|
// matched_group counts how many copies of section_delimiter
|
|
// this string consists of (0 if there's anything else)
|
|
let mut matched_groups = 0u8;
|
|
for c in line.chars() {
|
|
// If this is a newline character, the loop should end.
|
|
if c == '\n' {
|
|
break;
|
|
}
|
|
// If we have already seen three groups (corresponding to
|
|
// a header) or the current char does not form part of
|
|
// a new group, then this line is not a segment indicator.
|
|
if matched_groups >= 3
|
|
|| settings.section_delimiter[std::bool::to_bit::<uint>(odd)] != c {
|
|
matched_groups = 0;
|
|
break;
|
|
}
|
|
if odd {
|
|
// We have seen a new group and count it.
|
|
matched_groups += 1;
|
|
}
|
|
odd = !odd;
|
|
}
|
|
|
|
// See how many groups we matched. That will tell us if this is
|
|
// a line starting a new segment, and the number of groups
|
|
// indicates what type of segment.
|
|
if matched_groups > 0 {
|
|
// The current line is a section delimiter, so we output
|
|
// a blank line.
|
|
println!("");
|
|
// However the line does not count as a blank line, so we
|
|
// reset the counter used for --join-blank-lines.
|
|
empty_line_count = 0;
|
|
match *match matched_groups {
|
|
3 => {
|
|
// This is a header, so we may need to reset the
|
|
// line number and the line width
|
|
if settings.renumber {
|
|
line_no = settings.starting_line_number;
|
|
line_no_width = line_no_width_initial;
|
|
line_no_threshold = std::num::pow(10, line_no_width) as u64;
|
|
}
|
|
&settings.header_numbering
|
|
},
|
|
1 => {
|
|
&settings.footer_numbering
|
|
},
|
|
// The only option left is 2, but rust wants
|
|
// a catch-all here.
|
|
_ => {
|
|
&settings.body_numbering
|
|
}
|
|
} {
|
|
NumberForAll => {
|
|
line_filter = pass_all;
|
|
},
|
|
NumberForNonEmpty => {
|
|
line_filter = pass_nonempty;
|
|
},
|
|
NumberForNone => {
|
|
line_filter = pass_none;
|
|
}
|
|
NumberForRegularExpression(ref re) => {
|
|
line_filter = pass_regex;
|
|
regex_filter = re;
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
// From this point on we format and print a "regular" line.
|
|
if line == "" {
|
|
// The line is empty, which means that we have to care
|
|
// about the --join-blank-lines parameter.
|
|
empty_line_count += 1;
|
|
} else {
|
|
// This saves us from having to check for an empty string
|
|
// in the next selector.
|
|
empty_line_count = 0;
|
|
}
|
|
if !line_filter(line, regex_filter)
|
|
|| ( empty_line_count > 0 && empty_line_count < settings.join_blank_lines) {
|
|
// No number is printed for this line. Either we did not
|
|
// want to print one in the first place, or it is a blank
|
|
// line but we are still collecting more blank lines via
|
|
// the option --join-blank-lines.
|
|
println!("{}", line);
|
|
continue;
|
|
}
|
|
// If we make it here, then either we are printing a non-empty
|
|
// line or assigning a line number to an empty line. Either
|
|
// way, start counting empties from zero once more.
|
|
empty_line_count = 0;
|
|
// A line number is to be printed.
|
|
let mut w: uint = 0;
|
|
if settings.number_width > line_no_width {
|
|
w = settings.number_width - line_no_width;
|
|
}
|
|
let fill = String::from_char(w, fill_char);
|
|
match settings.number_format {
|
|
Left => {
|
|
println!("{1}{0}{2}{3}", fill, line_no, settings.number_separator, line)
|
|
},
|
|
_ => {
|
|
println!("{0}{1}{2}{3}", fill, line_no, settings.number_separator, line)
|
|
}
|
|
}
|
|
// Now update the variables for the (potential) next
|
|
// line.
|
|
line_no += settings.line_increment;
|
|
while line_no >= line_no_threshold {
|
|
// The line number just got longer.
|
|
line_no_threshold *= 10;
|
|
line_no_width += 1;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
fn pass_regex(line: &str, re: ®ex::Regex) -> bool {
|
|
re.is_match(line)
|
|
}
|
|
|
|
fn pass_nonempty(line: &str, _: ®ex::Regex) -> bool {
|
|
line.len() > 0
|
|
}
|
|
|
|
fn pass_none(_: &str, _: ®ex::Regex) -> bool {
|
|
false
|
|
}
|
|
|
|
fn pass_all(_: &str, _: ®ex::Regex) -> bool {
|
|
true
|
|
}
|
|
|
|
fn print_usage(opts: &[OptGroup]) {
|
|
println!("{:s}", usage(USAGE, opts));
|
|
}
|
|
|
|
fn version () {
|
|
println!("{} version 1.0.0", NAME);
|
|
}
|