forked from macournoyer/neuralconvo
-
Notifications
You must be signed in to change notification settings - Fork 128
/
Copy pathmovie_script_parser.lua
116 lines (96 loc) · 2.18 KB
/
movie_script_parser.lua
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
local Parser = torch.class("neuralconvo.MovieScriptParser")
function Parser:parse(file)
local f = assert(io.open(file, 'r'))
self.input = f:read("*all")
f:close()
self.pos = 0
self.match = nil
-- Find start of script
repeat self:acceptLine() until self:accept("<pre>")
local dialogs = {}
-- Apply rules until end of script
while not self:accept("</pre>") and self:acceptLine() do
local dialog = self:parseDialog()
if dialog then
table.insert(dialogs, dialog)
end
end
return dialogs
end
-- Returns true if regexp matches and advance position
function Parser:accept(regexp)
local match = string.match(self.input, "^" .. regexp, self.pos)
if match then
self.pos = self.pos + #match
self.match = match
return true
end
end
-- Accept anything up to the end of line
function Parser:acceptLine()
return self:accept(".-\n")
end
function Parser:acceptSep()
while self:accept("</?b>") or self:accept(" +") do end
return self:accept("\n")
end
function Parser:parseDialog()
local dialogs = {}
repeat
local dialog = self:parseSpeech()
if dialog then
table.insert(dialogs, dialog)
end
until not self:acceptSep()
if #dialogs > 0 then
return dialogs
end
end
-- Matches:
--
-- NAME
-- some nice text
-- more text.
--
-- or
--
-- NAME; text
function Parser:parseSpeech()
local name
self:accept("</b>")
self:accept("<b>")
-- Get the character name (all caps)
-- TODO remove parenthesis from name
if self:accept(" +") and self:accept("[A-Z][A-Z%- %.%(%)]+") then
name = self.match
else
return
end
-- Handle inline dialog: `NAME; text`
if self:accept(";") and self:accept("[^\n]+") then
return {
character = name,
text = self.match
}
end
self:accept("\n")
if not self:accept("</b>") then
return
end
-- Get the dialog lines
-- TODO remove parenthesis from text
local lines = {}
while self:accept(" +") do
-- The actual line of dialog
if self:accept("[^\n]+") then
table.insert(lines, self.match)
end
self:accept("\n")
end
if #lines > 0 then
return {
character = name,
text = table.concat(lines)
}
end
end