forked from HariSekhon/DevOps-Perl-tools
-
Notifications
You must be signed in to change notification settings - Fork 0
/
titlecase.pl
executable file
·127 lines (111 loc) · 4.87 KB
/
titlecase.pl
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
127
#!/usr/bin/perl -T
#
# Author: Hari Sekhon
# Date: 2015-05-10 12:07:12 +0100 (Sun, 10 May 2015)
#
# https://github.com/harisekhon/tools
#
# License: see accompanying Hari Sekhon LICENSE file
#
# If you're using my code you're welcome to connect with me on LinkedIn
# and optionally send me feedback to help improve or steer this or other code I publish
#
# https://www.linkedin.com/in/harisekhon
#
$DESCRIPTION="Capitalizes the first letter of each word (eg. to use a sentence as a title).
Works as a standard unix filter program, taking files are arguments or assuming input from standard input and printing to standard output.
Does not uppercase letters immediately after an apostrophe (unless a/i) or a dot except for dotted acronyms (2 or more letters preceeded by dots immediately following each other).
Can optionally also replace specific phrases using \"-r file.txt\" where the format in file.txt is either a correctly cased phrase or a case insensitive regex followed by ' => ' and then a string replacement (preserves whitespace in string):
# comment, camel case myString exactly as written below in -r file.txt
myString
# another comment, make this phrase appear with this exact case, to emphasize the dark side of the force...
Something Something DARK Side
regex1 => MY_String
regex2 => AnotherString
...
Limitations: cannot use capture references in the regex in -f file.txt. It's really meant more for simpler phrase substitution and I couldn't make this work without some kind of ugly or dangerous hack. A better workaround is to simply call a sed or perl inline afterwards such as:
echo \"catch and reprint any number while capitalizing K eg. 100k\" | titlecase.pl | perl -pe 's/(\\d+)k/\$1K/'
";
$VERSION = "0.5";
use strict;
use warnings;
BEGIN {
use File::Basename;
use lib dirname(__FILE__) . "/lib";
}
use HariSekhonUtils;
my $file;
my $lowercase;
my $recase;
%options = (
"f|files=s" => [ \$file, "File(s) to titlecase, non-option arguments are also counted as files. If no files are given uses standard input stream" ],
"l|lowercase" => [ \$lowercase, "Lowercase the rest of the letters (optional)" ],
"r|recase=s" => [ \$recase, "ReCase words/phrases from a text file, see full --help summary for description of format (optional)" ],
);
@usage_order = qw/files lowercase recase/;
get_options();
my @files = parse_file_option($file, "args are files");
my %recase_regexes;
my @recase_phrases;
if($recase){
my $fh = open_file $recase;
my $regex;
while (<$fh>){
chomp;
s/#.*//;
/^\s*$/ and next;
my @parts = split(/\s*=>\s*/, $_, 2);
my $replacement;
if(defined($parts[1])){
# allow first part to be regex
unless(isRegex($parts[0])){
warn "ignoring invalid regex in file '$recase': $parts[0]\n";
next;
}
# don't untaint this, was a risky experiment to eval that proved not worthwhile
#$parts[1] =~ /^(.*)$/;
#$parts[1] = $1;
# usig qr// here wraps the regex in (?^: ) which disables case insenstive matching even when the s/$regex/.../gi below uses the i modifier
$recase_regexes{$parts[0]} = $parts[1];
#vlog3 "regex is $parts[0] ... => ... $parts[1]";
} else {
unless($parts[0] =~ /^[\w\s\.\'\"-]+$/){
warn "skipping invalid phrase string in file '$recase': '$parts[0]' (must be an alphanumeric string - may also contain dots, quotes and dashes only, must not be a regex unless followed by '=> someString'\n";
next;
}
push(@recase_phrases, $parts[0]);
}
}
}
sub titlecase ($) {
my $string = shift;
$string = lc $string if $lowercase;
# exclude letters immediately preceeded by a dot such as file extensions
$string =~ s/\b(?<![\.'])([A-Za-z])/\U$1/g;
# catch letters starting words inside single quoted strings, unfortunately matches I'Ll, set {3,} as a workaround
$string =~ s/'([A-Za-z])([A-Za-z]{3,})/'\u$1$2/g;
# an exception for 'A Nice Title' since 'A isn't an apostrophe abbrieviation
$string =~ s/'([ai])/'$1/g;
# uppercase acronyms to correct for the above exception for file extensions
$string =~ s/(\s[A-Za-z](?:\.[A-Za-z])+)/\U$1/g;
if(@recase_phrases){
foreach my $phrase (@recase_phrases){
#vlog3 "trying phrase $phrase\n";
$string =~ s/\b($phrase)\b/$phrase/gi and vlog3 "replaced phrase $phrase";
}
}
if(%recase_regexes){
foreach my $regex (sort keys %recase_regexes){
$string =~ s/$regex/$recase_regexes{$regex}/gi and vlog3 "replaced regex '$regex' with '$recase_regexes{$regex}'";
}
}
print $string;
}
if(@files){
foreach my $file (@files){
my $fh = open_file $file;
while(<$fh>){ titlecase($_) }
}
} else {
while(<STDIN>){ titlecase($_) }
}