mirror of
https://github.com/processone/ejabberd.git
synced 2024-06-02 21:17:12 +02:00
Improve error handling/reporting when loading language translations
Also speed up loading on multi-core machines
This commit is contained in:
parent
04ccba0347
commit
691f9e0bf7
|
@ -39,6 +39,9 @@
|
||||||
|
|
||||||
-define(ZERO_DATETIME, {{0,0,0}, {0,0,0}}).
|
-define(ZERO_DATETIME, {{0,0,0}, {0,0,0}}).
|
||||||
|
|
||||||
|
-type error_reason() :: file:posix() | {integer(), module(), term()} |
|
||||||
|
badarg | terminated | system_limit | bad_file.
|
||||||
|
|
||||||
-record(state, {}).
|
-record(state, {}).
|
||||||
|
|
||||||
start_link() ->
|
start_link() ->
|
||||||
|
@ -46,9 +49,13 @@ start_link() ->
|
||||||
|
|
||||||
init([]) ->
|
init([]) ->
|
||||||
process_flag(trap_exit, true),
|
process_flag(trap_exit, true),
|
||||||
load(),
|
case load() of
|
||||||
|
ok ->
|
||||||
xmpp:set_tr_callback({?MODULE, translate}),
|
xmpp:set_tr_callback({?MODULE, translate}),
|
||||||
{ok, #state{}}.
|
{ok, #state{}};
|
||||||
|
{error, Reason} ->
|
||||||
|
{stop, Reason}
|
||||||
|
end.
|
||||||
|
|
||||||
handle_call(_Request, _From, State) ->
|
handle_call(_Request, _From, State) ->
|
||||||
Reply = ok,
|
Reply = ok,
|
||||||
|
@ -66,23 +73,25 @@ terminate(_Reason, _State) ->
|
||||||
code_change(_OldVsn, State, _Extra) ->
|
code_change(_OldVsn, State, _Extra) ->
|
||||||
{ok, State}.
|
{ok, State}.
|
||||||
|
|
||||||
-spec reload() -> ok.
|
-spec reload() -> ok | {error, error_reason()}.
|
||||||
reload() ->
|
reload() ->
|
||||||
load(true).
|
load(true).
|
||||||
|
|
||||||
-spec load() -> ok.
|
-spec load() -> ok | {error, error_reason()}.
|
||||||
load() ->
|
load() ->
|
||||||
load(false).
|
load(false).
|
||||||
|
|
||||||
-spec load(boolean()) -> ok.
|
-spec load(boolean()) -> ok | {error, error_reason()}.
|
||||||
load(ForceCacheRebuild) ->
|
load(ForceCacheRebuild) ->
|
||||||
{MsgsDirMTime, MsgsDir} = get_msg_dir(),
|
{MsgsDirMTime, MsgsDir} = get_msg_dir(),
|
||||||
{CacheMTime, CacheFile} = get_cache_file(),
|
{CacheMTime, CacheFile} = get_cache_file(),
|
||||||
{FilesMTime, MsgFiles} = get_msg_files(MsgsDir),
|
{FilesMTime, MsgFiles} = get_msg_files(MsgsDir),
|
||||||
LastModified = lists:max([MsgsDirMTime, FilesMTime]),
|
LastModified = lists:max([MsgsDirMTime, FilesMTime]),
|
||||||
if ForceCacheRebuild orelse CacheMTime < LastModified ->
|
if ForceCacheRebuild orelse CacheMTime < LastModified ->
|
||||||
load(MsgFiles, MsgsDir),
|
case load(MsgFiles, MsgsDir) of
|
||||||
dump_to_file(CacheFile);
|
ok -> dump_to_file(CacheFile);
|
||||||
|
Err -> Err
|
||||||
|
end;
|
||||||
true ->
|
true ->
|
||||||
case ets:file2tab(CacheFile) of
|
case ets:file2tab(CacheFile) of
|
||||||
{ok, _} ->
|
{ok, _} ->
|
||||||
|
@ -91,17 +100,18 @@ load(ForceCacheRebuild) ->
|
||||||
load(MsgFiles, MsgsDir);
|
load(MsgFiles, MsgsDir);
|
||||||
{error, {read_error, {file_error, _, Reason}}} ->
|
{error, {read_error, {file_error, _, Reason}}} ->
|
||||||
?WARNING_MSG("Failed to read translation cache from ~s: ~s",
|
?WARNING_MSG("Failed to read translation cache from ~s: ~s",
|
||||||
[CacheFile, file:format_error(Reason)]),
|
[unicode:characters_to_binary(CacheFile),
|
||||||
|
format_error(Reason)]),
|
||||||
load(MsgFiles, MsgsDir);
|
load(MsgFiles, MsgsDir);
|
||||||
{error, Reason} ->
|
{error, Reason} ->
|
||||||
?WARNING_MSG("Failed to read translation cache from ~s: ~p",
|
?WARNING_MSG("Failed to read translation cache from ~s: ~p",
|
||||||
[CacheFile, Reason]),
|
[unicode:characters_to_binary(CacheFile),
|
||||||
|
Reason]),
|
||||||
load(MsgFiles, MsgsDir)
|
load(MsgFiles, MsgsDir)
|
||||||
end
|
end
|
||||||
end.
|
end.
|
||||||
|
|
||||||
-spec load([file:filename()], file:filename()) -> ok.
|
-spec load([file:filename()], file:filename()) -> ok | {error, error_reason()}.
|
||||||
|
|
||||||
load(Files, Dir) ->
|
load(Files, Dir) ->
|
||||||
try ets:new(translations, [named_table, public]) of
|
try ets:new(translations, [named_table, public]) of
|
||||||
_ -> ok
|
_ -> ok
|
||||||
|
@ -110,79 +120,43 @@ load(Files, Dir) ->
|
||||||
case Files of
|
case Files of
|
||||||
[] ->
|
[] ->
|
||||||
?WARNING_MSG("No translation files found in ~s, "
|
?WARNING_MSG("No translation files found in ~s, "
|
||||||
"check directory access", [Dir]);
|
"check directory access",
|
||||||
|
[unicode:characters_to_binary(Dir)]);
|
||||||
_ ->
|
_ ->
|
||||||
|
?INFO_MSG("Building language translation cache", []),
|
||||||
|
Objs = lists:flatten(misc:pmap(fun load_file/1, Files)),
|
||||||
|
case lists:keyfind(error, 1, Objs) of
|
||||||
|
false ->
|
||||||
ets:delete_all_objects(translations),
|
ets:delete_all_objects(translations),
|
||||||
?INFO_MSG("Building translation cache, this may take a while", []),
|
ets:insert(translations, Objs),
|
||||||
lists:foreach(
|
?DEBUG("Language translation cache built successfully", []);
|
||||||
fun(File) ->
|
{error, File, Reason} ->
|
||||||
BaseName = filename:basename(File),
|
?ERROR_MSG("Failed to read translation file ~s: ~s",
|
||||||
Lang = str:to_lower(filename:rootname(BaseName)),
|
[unicode:characters_to_binary(File),
|
||||||
load_file(iolist_to_binary(Lang), File)
|
format_error(Reason)]),
|
||||||
end, Files)
|
{error, Reason}
|
||||||
|
end
|
||||||
end.
|
end.
|
||||||
|
|
||||||
load_file(Lang, File) ->
|
-spec load_file(file:filename()) -> [{{binary(), binary()}, binary()} |
|
||||||
case file:open(File, [read]) of
|
{error, file:filename(), error_reason()}].
|
||||||
{ok, Fd} ->
|
load_file(File) ->
|
||||||
case io:setopts(Fd, [{encoding,latin1}]) of
|
Lang = lang_of_file(File),
|
||||||
ok ->
|
case file:consult(File) of
|
||||||
load_file_loop(Fd, 1, File, Lang),
|
{ok, Lines} ->
|
||||||
file:close(Fd);
|
lists:map(
|
||||||
{error, Error} ->
|
fun({In, Out}) ->
|
||||||
ExitText = iolist_to_binary([File, ": ",
|
InB = iolist_to_binary(In),
|
||||||
file:format_error(Error)]),
|
OutB = iolist_to_binary(Out),
|
||||||
?ERROR_MSG("Problem loading translation file ~n~s",
|
{{Lang, InB}, OutB};
|
||||||
[ExitText]),
|
(_) ->
|
||||||
exit(ExitText)
|
{error, File, bad_file}
|
||||||
end;
|
end, Lines);
|
||||||
{error, Error} ->
|
|
||||||
ExitText = iolist_to_binary([File, ": ",
|
|
||||||
file:format_error(Error)]),
|
|
||||||
?ERROR_MSG("Problem loading translation file ~n~s",
|
|
||||||
[ExitText]),
|
|
||||||
exit(ExitText)
|
|
||||||
end.
|
|
||||||
|
|
||||||
load_file_loop(Fd, Line, File, Lang) ->
|
|
||||||
case io:read(Fd, '', Line) of
|
|
||||||
{ok,{Orig, Trans}, NextLine} ->
|
|
||||||
Trans1 = case Trans of
|
|
||||||
<<"">> -> Orig;
|
|
||||||
_ -> Trans
|
|
||||||
end,
|
|
||||||
ets:insert(translations,
|
|
||||||
{{Lang, iolist_to_binary(Orig)},
|
|
||||||
iolist_to_binary(Trans1)}),
|
|
||||||
|
|
||||||
load_file_loop(Fd, NextLine, File, Lang);
|
|
||||||
{ok,_, _NextLine} ->
|
|
||||||
ExitText = iolist_to_binary([File,
|
|
||||||
" approximately in the line ",
|
|
||||||
Line]),
|
|
||||||
?ERROR_MSG("Problem loading translation file ~n~s",
|
|
||||||
[ExitText]),
|
|
||||||
exit(ExitText);
|
|
||||||
{error,
|
|
||||||
{_LineNumber, erl_parse, _ParseMessage} = Reason} ->
|
|
||||||
ExitText = iolist_to_binary([File,
|
|
||||||
" approximately in the line ",
|
|
||||||
file:format_error(Reason)]),
|
|
||||||
?ERROR_MSG("Problem loading translation file ~n~s",
|
|
||||||
[ExitText]),
|
|
||||||
exit(ExitText);
|
|
||||||
{error, Reason} ->
|
{error, Reason} ->
|
||||||
ExitText = iolist_to_binary([File, ": ",
|
[{error, File, Reason}]
|
||||||
file:format_error(Reason)]),
|
|
||||||
?ERROR_MSG("Problem loading translation file ~n~s",
|
|
||||||
[ExitText]),
|
|
||||||
exit(ExitText);
|
|
||||||
{eof,_Line} ->
|
|
||||||
ok
|
|
||||||
end.
|
end.
|
||||||
|
|
||||||
-spec translate(binary(), binary()) -> binary().
|
-spec translate(binary(), binary()) -> binary().
|
||||||
|
|
||||||
translate(Lang, Msg) ->
|
translate(Lang, Msg) ->
|
||||||
LLang = ascii_tolower(Lang),
|
LLang = ascii_tolower(Lang),
|
||||||
case ets:lookup(translations, {LLang, Msg}) of
|
case ets:lookup(translations, {LLang, Msg}) of
|
||||||
|
@ -203,6 +177,7 @@ translate(Lang, Msg) ->
|
||||||
end
|
end
|
||||||
end.
|
end.
|
||||||
|
|
||||||
|
-spec translate(binary()) -> binary().
|
||||||
translate(Msg) ->
|
translate(Msg) ->
|
||||||
case ejabberd_option:language() of
|
case ejabberd_option:language() of
|
||||||
<<"en">> -> Msg;
|
<<"en">> -> Msg;
|
||||||
|
@ -227,13 +202,15 @@ translate(Msg) ->
|
||||||
end
|
end
|
||||||
end.
|
end.
|
||||||
|
|
||||||
ascii_tolower(B) ->
|
-spec ascii_tolower(list() | binary()) -> binary().
|
||||||
iolist_to_binary(ascii_tolower_s(binary_to_list(B))).
|
ascii_tolower(B) when is_binary(B) ->
|
||||||
|
<< <<(if X >= $A, X =< $Z ->
|
||||||
ascii_tolower_s([C | Cs]) when C >= $A, C =< $Z ->
|
X + 32;
|
||||||
[C + ($a - $A) | ascii_tolower_s(Cs)];
|
true ->
|
||||||
ascii_tolower_s([C | Cs]) -> [C | ascii_tolower_s(Cs)];
|
X
|
||||||
ascii_tolower_s([]) -> [].
|
end)>> || <<X>> <= B >>;
|
||||||
|
ascii_tolower(S) ->
|
||||||
|
ascii_tolower(unicode:characters_to_binary(S)).
|
||||||
|
|
||||||
-spec get_msg_dir() -> {calendar:datetime(), file:filename()}.
|
-spec get_msg_dir() -> {calendar:datetime(), file:filename()}.
|
||||||
get_msg_dir() ->
|
get_msg_dir() ->
|
||||||
|
@ -243,7 +220,8 @@ get_msg_dir() ->
|
||||||
{MTime, Dir};
|
{MTime, Dir};
|
||||||
{error, Reason} ->
|
{error, Reason} ->
|
||||||
?ERROR_MSG("Failed to read directory ~s: ~s",
|
?ERROR_MSG("Failed to read directory ~s: ~s",
|
||||||
[Dir, file:format_error(Reason)]),
|
[unicode:characters_to_binary(Dir),
|
||||||
|
format_error(Reason)]),
|
||||||
{?ZERO_DATETIME, Dir}
|
{?ZERO_DATETIME, Dir}
|
||||||
end.
|
end.
|
||||||
|
|
||||||
|
@ -252,12 +230,21 @@ get_msg_files(MsgsDir) ->
|
||||||
Res = filelib:fold_files(
|
Res = filelib:fold_files(
|
||||||
MsgsDir, ".+\\.msg", false,
|
MsgsDir, ".+\\.msg", false,
|
||||||
fun(File, {MTime, Files} = Acc) ->
|
fun(File, {MTime, Files} = Acc) ->
|
||||||
|
case xmpp_lang:is_valid(lang_of_file(File)) of
|
||||||
|
true ->
|
||||||
case file:read_file_info(File) of
|
case file:read_file_info(File) of
|
||||||
{ok, #file_info{mtime = Time}} ->
|
{ok, #file_info{mtime = Time}} ->
|
||||||
{lists:max([MTime, Time]), [File|Files]};
|
{lists:max([MTime, Time]), [File|Files]};
|
||||||
{error, Reason} ->
|
{error, Reason} ->
|
||||||
?ERROR_MSG("Failed to read translation file ~s: ~s",
|
?ERROR_MSG("Failed to read translation file ~s: ~s",
|
||||||
[File, file:format_error(Reason)]),
|
[unicode:characters_to_binary(File),
|
||||||
|
format_error(Reason)]),
|
||||||
|
Acc
|
||||||
|
end;
|
||||||
|
false ->
|
||||||
|
?WARNING_MSG("Ignoring translation file ~s: file name "
|
||||||
|
"must be a valid language tag",
|
||||||
|
[unicode:characters_to_binary(File)]),
|
||||||
Acc
|
Acc
|
||||||
end
|
end
|
||||||
end, {?ZERO_DATETIME, []}),
|
end, {?ZERO_DATETIME, []}),
|
||||||
|
@ -267,7 +254,8 @@ get_msg_files(MsgsDir) ->
|
||||||
{ok, _} -> ok;
|
{ok, _} -> ok;
|
||||||
{error, Reason} ->
|
{error, Reason} ->
|
||||||
?ERROR_MSG("Failed to read directory ~s: ~s",
|
?ERROR_MSG("Failed to read directory ~s: ~s",
|
||||||
[MsgsDir, file:format_error(Reason)])
|
[unicode:characters_to_binary(MsgsDir),
|
||||||
|
format_error(Reason)])
|
||||||
end;
|
end;
|
||||||
_ ->
|
_ ->
|
||||||
ok
|
ok
|
||||||
|
@ -290,5 +278,18 @@ dump_to_file(CacheFile) ->
|
||||||
ok -> ok;
|
ok -> ok;
|
||||||
{error, Reason} ->
|
{error, Reason} ->
|
||||||
?WARNING_MSG("Failed to create translation cache in ~s: ~p",
|
?WARNING_MSG("Failed to create translation cache in ~s: ~p",
|
||||||
[CacheFile, Reason])
|
[unicode:characters_to_binary(CacheFile), Reason])
|
||||||
end.
|
end.
|
||||||
|
|
||||||
|
-spec lang_of_file(file:filename()) -> binary().
|
||||||
|
lang_of_file(FileName) ->
|
||||||
|
BaseName = filename:basename(FileName),
|
||||||
|
ascii_tolower(filename:rootname(BaseName)).
|
||||||
|
|
||||||
|
-spec format_error(error_reason()) -> string().
|
||||||
|
format_error(bad_file) ->
|
||||||
|
"corrupted or invalid translation file";
|
||||||
|
format_error({_, _, _} = Reason) ->
|
||||||
|
"at line " ++ file:format_error(Reason);
|
||||||
|
format_error(Reason) ->
|
||||||
|
file:format_error(Reason).
|
||||||
|
|
Loading…
Reference in New Issue
Block a user