-
Notifications
You must be signed in to change notification settings - Fork 777
/
Copy pathtrain-truecaser.perl
executable file
·127 lines (117 loc) · 4.1 KB
/
train-truecaser.perl
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/env perl
#
# This file is part of moses. Its use is licensed under the GNU Lesser General
# Public License version 2.1 or, at your option, any later version.
# $Id: train-recaser.perl 1326 2007-03-26 05:44:27Z bojar $
#
# Options:
#
# --possiblyUseFirstToken : boolean option; the default behaviour (when this
# option is not provided) is that the first token of a sentence is ignored, on
# the basis that the first word of a sentence is always capitalized; if this
# option is provided then: a) if a sentence-initial token is *not* capitalized,
# then it is counted, and b) if a capitalized sentence-initial token is the
# only token of the segment, then it is counted, but with only 10% of the
# weight of a normal token.
use warnings;
use strict;
use Getopt::Long "GetOptions";
# apply switches
my ($MODEL,$CORPUS);
die("train-truecaser.perl --model truecaser --corpus cased [--possiblyUseFirstToken]")
unless &GetOptions('corpus=s' => \$CORPUS,
'model=s' => \$MODEL,
'possiblyUseFirstToken' => \(my $possiblyUseFirstToken = 0))
&& defined($CORPUS) && defined($MODEL);
my %CASING;
my %SENTENCE_END = ("."=>1,":"=>1,"?"=>1,"!"=>1);
my %DELAYED_SENTENCE_START = ("("=>1,"["=>1,"\""=>1,"'"=>1,"'"=>1,"""=>1,"["=>1,"]"=>1);
open(CORPUS,$CORPUS) || die("ERROR: could not open '$CORPUS'");
binmode(CORPUS, ":utf8");
while(<CORPUS>) {
chop;
my ($WORD,$MARKUP) = split_xml($_);
my $start = 0;
while($start<=$#$WORD && defined($DELAYED_SENTENCE_START{$$WORD[$start]})) { $start++; }
my $firstWordOfSentence = 1;
for(my $i=$start;$i<=$#$WORD;$i++) {
my $currentWord = $$WORD[$i];
if (! $firstWordOfSentence && defined($SENTENCE_END{$$WORD[$i-1]})) {
$firstWordOfSentence = 1;
}
if ($currentWord !~ /[\p{Ll}\p{Lu}\p{Lt}]/) {
# skip words with nothing to case
$firstWordOfSentence = 0;
next;
}
my $currentWordWeight = 0;
if (! $firstWordOfSentence) {
$currentWordWeight = 1;
} elsif ($possiblyUseFirstToken) {
# gated special handling of first word of sentence
my $firstChar = substr($currentWord, 0, 1);
if (lc($firstChar) eq $firstChar) {
# if the first character is not upper case, count the token as full evidence (because if it's not capitalized, then there's no reason to be wary that the given casing is only due to being sentence-initial)
$currentWordWeight = 1;
} elsif (scalar(@$WORD) == 1) {
# if the first character is upper case, but the current token is the only token of the segment, then count the token as partial evidence (because the segment is presumably not a sentence and the token is therefore not the first word of a sentence and is possibly in its natural case)
$currentWordWeight = 0.1;
}
}
if ($currentWordWeight > 0) {
$CASING{ lc($currentWord) }{ $currentWord } += $currentWordWeight;
}
$firstWordOfSentence = 0;
}
}
close(CORPUS);
open(MODEL,">$MODEL") || die("ERROR: could not create '$MODEL'");
binmode(MODEL, ":utf8");
foreach my $type (keys %CASING) {
my ($score,$total,$best) = (-1,0,"");
foreach my $word (keys %{$CASING{$type}}) {
my $count = $CASING{$type}{$word};
$total += $count;
if ($count > $score) {
$best = $word;
$score = $count;
}
}
print MODEL "$best ($score/$total)";
foreach my $word (keys %{$CASING{$type}}) {
print MODEL " $word ($CASING{$type}{$word})" unless $word eq $best;
}
print MODEL "\n";
}
close(MODEL);
# store away xml markup
sub split_xml {
my ($line) = @_;
my (@WORD,@MARKUP);
my $i = 0;
$MARKUP[0] = "";
while($line =~ /\S/) {
# XML tag
if ($line =~ /^\s*(<\S[^>]*>)(.*)$/) {
$MARKUP[$i] .= $1." ";
$line = $2;
}
# non-XML text
elsif ($line =~ /^\s*([^\s<>]+)(.*)$/) {
$WORD[$i++] = $1;
$MARKUP[$i] = "";
$line = $2;
}
# '<' or '>' occurs in word, but it's not an XML tag
elsif ($line =~ /^\s*(\S+)(.*)$/) {
$WORD[$i++] = $1;
$MARKUP[$i] = "";
$line = $2;
}
else {
die("ERROR: huh? $line\n");
}
}
chop($MARKUP[$#MARKUP]);
return (\@WORD,\@MARKUP);
}