-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathwebb_action.lua
378 lines (326 loc) · 10.1 KB
/
webb_action.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
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
--[==[
webb | action-based routing with multi-language support
Written by Cosmin Apreutesei. Public Domain.
ACTION ALIASES
lang([s]) -> s get/set current language
default_lang() -> s get default language
alias(name_en, name, lang) set an action alias for a language
href(s|t[, target_lang]) -> s translate page URL based on alias
setlinks(s) -> s html filter to translate URLs
ACTIONS
action(name, args...) -> t|f execute action as http response
fileaction(path) -> t|f serve a plain file
execaction(name, args...) -> ret...|true execute action internally
function action.NAME(args) end set an inline action handler
CONFIG
config('lang', 'en') default language
config('root_action', 'en') name of the '/' (root) action
config('404_html_action', '404.html') 404 action for text/html
config('404_png_action' , '404.png' ) 404 action for image/png
config('404_jpeg_action', '404.jpg' ) 404 action for image/jpeg
TODO
* cascaded actions: html.m.lua, html.m.lp, etc.
* .markdown filter.
]==]
require'webb'
--multi-language actions & links ---------------------------------------------
function default_lang()
return config('lang', 'en')
end
function lang(s)
if s then cx().lang = s end
return cx().lang or args'lang' or default_lang()
end
function S(id, en_s, ...)
local t = S_texts(lang(), 'lua')
local s = t[id] or en_s or ''
if select('#', ...) > 0 then
return glue.subst(s, ...)
else
return s
end
end
--[[
It is assumed that action names are always in english even if they actually
request a page in the default language which can configured to be different
than english. Action name translation is done automatically provided that
1) all links are passed through href(), 2) routing is done by calling
action(req, unpack(args())) which calls find_action() and 3) action names are
translated in different languages with alias(). Using action aliases is
the key to avoiding the appending of ?lang=xx to links. Aliases for the
root action ('en') are also allowed in order to avoid the ?lang param.
]]
local aliases = {} --{alias={lang=, action=}}
local aliases_json = {to_en = {}, to_lang = {}}
config('aliases', aliases_json) --we pass those to the client
local function action_name(action)
return action:gsub('-', '_')
end
function alias(en_action, alias_action, alias_lang)
local default_lang = config('lang', 'en')
alias_lang = alias_lang or default_lang
alias_action = action_name(alias_action)
en_action = action_name(en_action)
aliases[alias_action] = {lang = alias_lang, action = en_action}
--if the default language is not english and we're making
--an alias for the default language, then we can safely assign
--the english action name for the english language, whereas before
--we would use the english action name for the default language.
if default_lang ~= 'en' and alias_lang == default_lang then
if not aliases[en_action] then --user can override this
aliases[en_action] = {lang = 'en', action = en_action}
glue.attr(aliases_json.to_lang, en_action).en = en_action
end
end
aliases_json.to_en[alias_action] = en_action
glue.attr(aliases_json.to_lang, en_action)[alias_lang] = alias_action
end
--given a url (in encoded or decoded form), if it's an action url,
--replace its action name with a language-specific alias for a given
--(or current) language if any, or add ?lang= if the given language
--is not the default language.
function href(s, target_lang)
local t = url_arg(s)
local segs = t.segments
local action = not t.host and (segs[1] == '' and segs[2] and action_name(segs[2])) or nil
if not action then
return url(s)
end
local default_lang = config('lang', 'en')
local target_lang = target_lang or (t.args and t.args.lang) or lang()
local is_root = segs[2] == ''
if is_root then
action = action_name(config('root_action', 'en'))
end
local at = aliases_json.to_lang[action]
local lang_action = at and at[target_lang]
if lang_action then
if not (is_root and target_lang == default_lang) then
t[2] = lang_action
end
elseif target_lang ~= default_lang then
attr(t, 'args').lang = target_lang
end
return url(t)
end
--given a list of path elements, find the action they point to
--and change the current language if necessary.
local function find_action(action, ...)
if action == '' then --root action in current language
action = config('root_action', 'en')
else
local alias = aliases[action_name(action)] --look for a regional alias
if alias then
if not args'lang' then --?lang= has priority
lang(alias.lang)
end
action = alias.action
end
end
return action, ...
end
--html output filter for rewriting links based on current language aliases
function setlinks(s)
local function repl(prefix, s)
return prefix..href(s)
end
s = s:gsub('(%shref=")([^"]+)', repl)
s = s:gsub('(%shref=)([^ >]+)', repl)
return s
end
--override redirect to automatically translate URLs.
local webb_redirect = redirect
function redirect(url, ...)
return webb_redirect(href(url), ...)
end
--serving plain files --------------------------------------------------------
local ffi = require'ffi'
local fs = require'fs'
local function plain_file_handler(path)
local f = fs.open(path, 'r')
if not f then
return
end
local mtime, err = f:attr'mtime'
if not mtime then
f:close()
error(err)
end
check_etag(tostring(mtime))
local file_size, err = f:attr'size'
if not file_size then
f:close()
error(err)
end
setheader('content-length', file_size)
return function()
local filebuf_size = math.min(file_size, 65536)
local filebuf = ffi.new('char[?]', filebuf_size)
while true do
local len, err = f:read(filebuf, filebuf_size)
if not len then
f:close()
error(err)
elseif len == 0 then
f:close()
break
else
out(filebuf, len)
end
end
end
end
--output filters -------------------------------------------------------------
local function html_filter(handler, ...)
local s = record(handler, ...)
local s = setlinks(filter_lang(filter_comments(s), lang()))
check_etag(s)
outall(s)
end
local function json_filter(handler, ...)
local s = handler(...)
if type(s) == 'table' then
s = json(s)
end
if s then
check_etag(s)
outall(s)
end
end
local function js_filter(handler, ...)
if not config'minify_js' then
handler(...)
return
end
local minify = require'jsmin'.minify
local s = record(handler, ...)
s = minify(s)
check_etag(s)
outall(s)
end
local mime_type_filters = {
['text/html'] = html_filter,
['application/json'] = json_filter,
['text/javascript'] = js_filter,
}
--routing logic --------------------------------------------------------------
local file_handlers = {
cat = function(file, ...)
outcatlist(file, ...)
end,
lua = function(file, ...)
return run(file, nil, ...)
end,
lp = function(file, ...)
include(file)
end,
}
local actions_ext = glue.keys(file_handlers, true)
local actions = {} --{action -> handler | s}
local function action_handler(action, ...)
local action_ext = fileext(action)
local action_no_ext
local action_with_ext = action
if not action_ext then --add the default .html extension to the action
action_no_ext = action
action_ext = 'html'
action_with_ext = action .. '.' .. action_ext
elseif action_ext == 'html' then
action_no_ext = action:gsub('%.html$', '')
end
local ext = action_ext
local handler =
(action_no_ext and actions[action_name(action_no_ext)]) --look in the default action table
or actions[action_name(action_with_ext)] --look again with .html extension
if not handler then --look in the filesystem
local file = table.concat({action, ...}, '/')
file_ext = fileext(file)
local plain_file_allowed = not (file_ext and file_handlers[file_ext])
if plain_file_allowed then
handler = wwwfile[file] and wwwfile(file)
if not handler then
local path = wwwpath(file)
handler = path and assert(plain_file_handler(path))
end
end
if handler then
ext = file_ext
end
end
if not handler then
local action_file, file_handler
for i,ext in ipairs(actions_ext) do
local action_file1 = action_with_ext..'.'..ext
if wwwfile[action_file1] or wwwpath(action_file1) then
glue.assert(not action_file,
'multiple action files for %s (%s, was %s)',
action_with_ext, action_file1, action_file)
file_handler = file_handlers[ext]
action_file = action_file1
end
end
handler = action_file and function(...)
return file_handler(action_file, ...)
end
end
if handler and type(handler) ~= 'function' then
local s = handler
handler = function()
outall(s)
end
end
return handler, ext
end
--exec an action without setting content type, looking for a 404 handler
--or filtering the output based on mime type, instead returns whatever
--the action returns (good for exec'ing json actions which return a table).
local function pass(arg1, ...)
if not arg1 then
return true, ...
else
return arg1, ...
end
end
function execaction(action, ...)
local handler = action_handler(action, ...)
if not handler then return false end
return pass(handler(...))
end
local function run_action(fallback, action, handler, ext, ...)
local mime = mime_types[ext]
if not handler then
if not fallback then
return false
end
local not_found_actions = {
['text/html' ] = config('404_html_action', '404.html'),
['image/png' ] = config('404_png_action' , '404.png'),
['image/jpeg'] = config('404_jpeg_action', '404.jpg'),
}
local nf_action = not_found_actions[mime]
if not nf_action then
webb.note('webb', '404', '%s', table.concat({action, ...}, '/'))
return false
end
local handler, ext = action_handler(nf_action, ...)
return run_action(false, nf_action, handler, ext, action, ...)
end
setmime(ext)
local filter = mime_type_filters[mime]
if filter then
filter(handler, ...)
else
handler(...)
end
return true
end
setmetatable(actions, {__call = function(self, action, ...)
local handler, ext = action_handler(find_action(action, ...))
return run_action(true, action, handler, ext, ...)
end})
action = actions
function fileaction(path)
local ext = fileext(path)
local handler = plain_file_handler(path)
return run_action(true, path, handler, ext)
end