#![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 * * 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) -> 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 (reader: &mut BufferedReader, 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::(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); }