2006-01-27 06:02:33 +01:00
|
|
|
%%%-------------------------------------------------------------------
|
|
|
|
%%% File : ejabberd_update.erl
|
2007-12-24 12:41:41 +01:00
|
|
|
%%% Author : Alexey Shchepin <alexey@process-one.net>
|
|
|
|
%%% Purpose : ejabberd code updater
|
|
|
|
%%% Created : 27 Jan 2006 by Alexey Shchepin <alexey@process-one.net>
|
|
|
|
%%%
|
|
|
|
%%%
|
2008-09-16 16:39:57 +02:00
|
|
|
%%% ejabberd, Copyright (C) 2002-2008 ProcessOne
|
2007-12-24 12:41:41 +01:00
|
|
|
%%%
|
|
|
|
%%% This program is free software; you can redistribute it and/or
|
|
|
|
%%% modify it under the terms of the GNU General Public License as
|
|
|
|
%%% published by the Free Software Foundation; either version 2 of the
|
|
|
|
%%% License, or (at your option) any later version.
|
|
|
|
%%%
|
|
|
|
%%% This program is distributed in the hope that it will be useful,
|
|
|
|
%%% but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
%%% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
%%% General Public License for more details.
|
|
|
|
%%%
|
|
|
|
%%% You should have received a copy of the GNU General Public License
|
|
|
|
%%% along with this program; if not, write to the Free Software
|
|
|
|
%%% Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
|
|
|
|
%%% 02111-1307 USA
|
|
|
|
%%%
|
2006-01-27 06:02:33 +01:00
|
|
|
%%%-------------------------------------------------------------------
|
|
|
|
|
|
|
|
-module(ejabberd_update).
|
2007-12-24 12:41:41 +01:00
|
|
|
-author('alexey@process-one.net').
|
2006-01-27 06:02:33 +01:00
|
|
|
|
|
|
|
%% API
|
2006-02-27 05:43:16 +01:00
|
|
|
-export([update/0, update_info/0]).
|
2006-01-27 06:02:33 +01:00
|
|
|
|
2008-03-21 15:44:16 +01:00
|
|
|
-include("ejabberd.hrl").
|
|
|
|
|
2006-01-27 06:02:33 +01:00
|
|
|
%%====================================================================
|
|
|
|
%% API
|
|
|
|
%%====================================================================
|
|
|
|
update() ->
|
2006-02-27 05:43:16 +01:00
|
|
|
case update_info() of
|
2006-02-27 05:47:30 +01:00
|
|
|
{ok, Dir, _UpdatedBeams, _Script, LowLevelScript, _Check} ->
|
2006-02-27 05:43:16 +01:00
|
|
|
Eval =
|
|
|
|
release_handler_1:eval_script(
|
|
|
|
LowLevelScript, [],
|
|
|
|
[{ejabberd, "", filename:join(Dir, "..")}]),
|
2008-03-21 15:44:16 +01:00
|
|
|
?INFO_MSG("eval: ~p~n", [Eval]),
|
2006-02-27 05:43:16 +01:00
|
|
|
Eval;
|
|
|
|
{error, Reason} ->
|
|
|
|
{error, Reason}
|
|
|
|
end.
|
|
|
|
|
|
|
|
update_info() ->
|
2006-01-27 06:02:33 +01:00
|
|
|
Dir = filename:dirname(code:which(ejabberd)),
|
|
|
|
case file:list_dir(Dir) of
|
|
|
|
{ok, Files} ->
|
|
|
|
Beams = [list_to_atom(filename:rootname(FN)) ||
|
|
|
|
FN <- Files, lists:suffix(".beam", FN)],
|
|
|
|
UpdatedBeams =
|
|
|
|
lists:filter(
|
|
|
|
fun(Module) ->
|
2008-03-10 12:59:53 +01:00
|
|
|
{ok, {Module, NewVsn}} =
|
2006-01-27 06:02:33 +01:00
|
|
|
beam_lib:version(code:which(Module)),
|
|
|
|
case code:is_loaded(Module) of
|
|
|
|
{file, _} ->
|
|
|
|
Attrs = Module:module_info(attributes),
|
2008-03-10 12:59:53 +01:00
|
|
|
{value, {vsn, CurVsn}} =
|
2006-01-27 06:02:33 +01:00
|
|
|
lists:keysearch(vsn, 1, Attrs),
|
|
|
|
NewVsn /= CurVsn;
|
|
|
|
false ->
|
|
|
|
false
|
|
|
|
end
|
|
|
|
end, Beams),
|
2008-03-21 15:44:16 +01:00
|
|
|
?INFO_MSG("beam files: ~p~n", [UpdatedBeams]),
|
2006-01-27 06:02:33 +01:00
|
|
|
Script = make_script(UpdatedBeams),
|
2008-03-21 15:44:16 +01:00
|
|
|
?INFO_MSG("script: ~p~n", [Script]),
|
2006-01-27 06:02:33 +01:00
|
|
|
LowLevelScript = make_low_level_script(UpdatedBeams, Script),
|
2008-03-21 15:44:16 +01:00
|
|
|
?INFO_MSG("low level script: ~p~n", [LowLevelScript]),
|
2006-01-27 06:02:33 +01:00
|
|
|
Check =
|
|
|
|
release_handler_1:check_script(
|
|
|
|
LowLevelScript,
|
|
|
|
[{ejabberd, "", filename:join(Dir, "..")}]),
|
2008-03-21 15:44:16 +01:00
|
|
|
?INFO_MSG("check: ~p~n", [Check]),
|
2006-02-27 05:43:16 +01:00
|
|
|
{ok, Dir, UpdatedBeams, Script, LowLevelScript, Check};
|
2006-01-27 06:02:33 +01:00
|
|
|
{error, Reason} ->
|
|
|
|
{error, Reason}
|
|
|
|
end.
|
|
|
|
|
|
|
|
|
|
|
|
%%--------------------------------------------------------------------
|
|
|
|
%%% Internal functions
|
|
|
|
%%--------------------------------------------------------------------
|
|
|
|
|
|
|
|
%% From systools.hrl
|
|
|
|
-record(application,
|
|
|
|
{name, %% Name of the application, atom().
|
|
|
|
type = permanent, %% Application start type, atom().
|
|
|
|
vsn = "", %% Version of the application, string().
|
|
|
|
id = "", %% Id of the application, string().
|
|
|
|
description = "", %% Description of application, string().
|
|
|
|
modules = [], %% [Module | {Module,Vsn}] of modules
|
|
|
|
%% incorporated in the application,
|
|
|
|
%% Module = atom(), Vsn = string().
|
|
|
|
uses = [], %% [Application] list of applications required
|
|
|
|
%% by the application, Application = atom().
|
|
|
|
includes = [], %% [Application] list of applications included
|
|
|
|
%% by the application, Application = atom().
|
|
|
|
regs = [], %% [RegNames] a list of registered process
|
|
|
|
%% names used by the application, RegNames =
|
|
|
|
%% atom().
|
|
|
|
env = [], %% [{Key,Value}] environment variable of
|
|
|
|
%% application, Key = Value = term().
|
|
|
|
maxT = infinity, %% Max time an application may exist,
|
|
|
|
%% integer() | infinity.
|
|
|
|
maxP = infinity, %% Max number of processes in an application,
|
|
|
|
%% integer() | infinity.
|
|
|
|
mod = [], %% [] | {Mod, StartArgs}, Mod= atom(),
|
|
|
|
%% StartArgs = list().
|
|
|
|
start_phases = [], %% [] | {Phase, PhaseArgs}, Phase = atom(),
|
|
|
|
%% PhaseArgs = list().
|
|
|
|
dir = "" %% The directory where the .app file was
|
|
|
|
%% found (internal use).
|
|
|
|
}).
|
|
|
|
|
|
|
|
|
|
|
|
make_script(UpdatedBeams) ->
|
|
|
|
lists:map(
|
|
|
|
fun(Module) ->
|
2006-01-29 05:38:31 +01:00
|
|
|
{ok, {Module, [{attributes, NewAttrs}]}} =
|
2006-01-27 06:02:33 +01:00
|
|
|
beam_lib:chunks(code:which(Module), [attributes]),
|
2006-01-29 05:38:31 +01:00
|
|
|
CurAttrs = Module:module_info(attributes),
|
|
|
|
case lists:keysearch(update_info, 1, NewAttrs) of
|
|
|
|
{value, {_, [{update, _}]}} ->
|
|
|
|
case lists:keysearch(update_info, 1, CurAttrs) of
|
|
|
|
{value, {_, [{update, Extra}]}} ->
|
|
|
|
{update, Module, {advanced, Extra}};
|
|
|
|
false ->
|
|
|
|
{update, Module, {advanced, 0}}
|
|
|
|
end;
|
2006-01-27 06:02:33 +01:00
|
|
|
false ->
|
|
|
|
{load_module, Module}
|
|
|
|
end
|
|
|
|
end, UpdatedBeams).
|
|
|
|
|
|
|
|
make_low_level_script(UpdatedBeams, Script) ->
|
|
|
|
EJDApp = #application{name = ejabberd,
|
|
|
|
modules = UpdatedBeams},
|
|
|
|
{ok, LowLevelScript} =
|
|
|
|
systools_rc:translate_scripts([Script], [EJDApp], [EJDApp]),
|
|
|
|
LowLevelScript.
|