Start work on making the EGS server OTP-compliant. Add egs_app and egs_sup.
This commit is contained in:
parent
49016cafaa
commit
f46ba0e1f9
@ -17,6 +17,8 @@
|
|||||||
% along with EGS. If not, see <http://www.gnu.org/licenses/>.
|
% along with EGS. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
{'src/egs.erl', [{outdir, "ebin"}]}.
|
{'src/egs.erl', [{outdir, "ebin"}]}.
|
||||||
|
{'src/egs_app.erl', [{outdir, "ebin"}]}.
|
||||||
|
{'src/egs_sup.erl', [{outdir, "ebin"}]}.
|
||||||
{'src/egs_cron.erl', [{outdir, "ebin"}]}.
|
{'src/egs_cron.erl', [{outdir, "ebin"}]}.
|
||||||
{'src/egs_db.erl', [{outdir, "ebin"}]}.
|
{'src/egs_db.erl', [{outdir, "ebin"}]}.
|
||||||
{'src/egs_game.erl', [{outdir, "ebin"}]}.
|
{'src/egs_game.erl', [{outdir, "ebin"}]}.
|
||||||
|
4
Makefile
4
Makefile
@ -31,10 +31,10 @@ clean:
|
|||||||
rm -f erl_crash.dump
|
rm -f erl_crash.dump
|
||||||
|
|
||||||
fclean: clean
|
fclean: clean
|
||||||
rm -rf Mnesia.console*
|
rm -rf Mnesia.egs*
|
||||||
|
|
||||||
run:
|
run:
|
||||||
@echo "EGS is free software available under the GNU GPL version 3"
|
@echo "EGS is free software available under the GNU GPL version 3"
|
||||||
@echo "Copyright (C) 2010 Loic Hoguin"
|
@echo "Copyright (C) 2010 Loic Hoguin"
|
||||||
@echo
|
@echo
|
||||||
erl -ssl protocol_version '{sslv3}' -sname console -pa ebin -eval 'egs:start()'
|
erl -ssl protocol_version '{sslv3}' -sname egs -pa ebin -s egs
|
||||||
|
30
ebin/egs.app
Normal file
30
ebin/egs.app
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
%%-*- mode: erlang -*-
|
||||||
|
{application, egs, [
|
||||||
|
{description, "EGS online action-RPG game server"},
|
||||||
|
{vsn, "0.1"},
|
||||||
|
{modules, [
|
||||||
|
egs,
|
||||||
|
egs_app,
|
||||||
|
egs_sup,
|
||||||
|
egs_cron,
|
||||||
|
egs_db,
|
||||||
|
egs_game,
|
||||||
|
egs_login,
|
||||||
|
egs_patch,
|
||||||
|
egs_proto,
|
||||||
|
psu_appearance,
|
||||||
|
psu_characters,
|
||||||
|
psu_missions,
|
||||||
|
psu_parser
|
||||||
|
]},
|
||||||
|
{registered, []},
|
||||||
|
{applications, [
|
||||||
|
kernel,
|
||||||
|
stdlib,
|
||||||
|
crypto,
|
||||||
|
ssl,
|
||||||
|
mnesia
|
||||||
|
]},
|
||||||
|
{mod, {egs_app, []}},
|
||||||
|
{env, []}
|
||||||
|
]}.
|
76
src/egs.erl
76
src/egs.erl
@ -1,59 +1,73 @@
|
|||||||
% EGS: Erlang Game Server
|
%% @author Loïc Hoguin <essen@dev-extend.eu>
|
||||||
% Copyright (C) 2010 Loic Hoguin
|
%% @copyright 2010 Loïc Hoguin.
|
||||||
%
|
%% @doc EGS startup code.
|
||||||
% This file is part of EGS.
|
%%
|
||||||
%
|
%% This file is part of EGS.
|
||||||
% EGS is free software: you can redistribute it and/or modify
|
%%
|
||||||
% it under the terms of the GNU General Public License as published by
|
%% EGS is free software: you can redistribute it and/or modify
|
||||||
% the Free Software Foundation, either version 3 of the License, or
|
%% it under the terms of the GNU General Public License as published by
|
||||||
% (at your option) any later version.
|
%% the Free Software Foundation, either version 3 of the License, or
|
||||||
%
|
%% (at your option) any later version.
|
||||||
% EGS is distributed in the hope that it will be useful,
|
%%
|
||||||
% but WITHOUT ANY WARRANTY; without even the implied warranty of
|
%% EGS is distributed in the hope that it will be useful,
|
||||||
% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
%% but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
% GNU General Public License for more details.
|
%% 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 EGS. If not, see <http://www.gnu.org/licenses/>.
|
%% You should have received a copy of the GNU General Public License
|
||||||
|
%% along with EGS. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
-module(egs).
|
-module(egs).
|
||||||
-compile(export_all).
|
-compile(export_all).
|
||||||
|
|
||||||
-include("include/records.hrl").
|
-include("include/records.hrl").
|
||||||
|
|
||||||
-define(MODULES, [egs, egs_cron, egs_db, egs_game, egs_login, egs_patch, egs_proto, psu_appearance, psu_characters, psu_missions, psu_parser]).
|
-define(MODULES, [egs, egs_app, egs_sup, egs_cron, egs_db, egs_game, egs_login, egs_patch, egs_proto, psu_appearance, psu_characters, psu_missions, psu_parser]).
|
||||||
|
|
||||||
%% @doc Start all the application servers. Return the PIDs of the listening processes.
|
%% @spec ensure_started(App) -> ok
|
||||||
|
%% @doc Make sure the given App is started.
|
||||||
|
ensure_started(App) ->
|
||||||
|
case application:start(App) of
|
||||||
|
ok -> ok;
|
||||||
|
{error, {already_started, App}} -> ok
|
||||||
|
end.
|
||||||
|
|
||||||
|
%% @spec start() -> ok
|
||||||
|
%% @doc Start the EGS server.
|
||||||
start() ->
|
start() ->
|
||||||
application:start(crypto),
|
ensure_started(crypto),
|
||||||
application:start(ssl),
|
ensure_started(ssl),
|
||||||
ssl:seed(crypto:rand_bytes(256)),
|
ssl:seed(crypto:rand_bytes(256)),
|
||||||
egs_db:create(),
|
ensure_started(mnesia),
|
||||||
Cron = egs_cron:start(),
|
application:start(egs).
|
||||||
Game = egs_game:start(),
|
|
||||||
Login = egs_login:start(),
|
%% @spec stop() -> ok
|
||||||
Patch = egs_patch:start(),
|
%% @doc Stop the EGS server.
|
||||||
[{patch, Patch}, {login, Login}, {game, Game}, {cron, Cron}].
|
stop() ->
|
||||||
|
Res = application:stop(egs),
|
||||||
|
application:stop(mnesia),
|
||||||
|
application:stop(ssl),
|
||||||
|
application:stop(crypto),
|
||||||
|
Res.
|
||||||
|
|
||||||
%% @doc Reload all the modules.
|
%% @doc Reload all the modules.
|
||||||
|
%% @todo Do it the OTP way.
|
||||||
reload() ->
|
reload() ->
|
||||||
[code:soft_purge(Module) || Module <- ?MODULES],
|
[code:soft_purge(Module) || Module <- ?MODULES],
|
||||||
[code:load_file(Module) || Module <- ?MODULES].
|
[code:load_file(Module) || Module <- ?MODULES].
|
||||||
|
|
||||||
%% @doc Send a global message.
|
%% @doc Send a global message.
|
||||||
|
%% @todo Move that in a psu module.
|
||||||
global(Type, Message) ->
|
global(Type, Message) ->
|
||||||
lists:foreach(fun(User) -> egs_proto:send_global(User#users.socket, Type, Message) end, egs_db:users_select_all()).
|
lists:foreach(fun(User) -> egs_proto:send_global(User#users.socket, Type, Message) end, egs_db:users_select_all()).
|
||||||
|
|
||||||
%% @doc Warp all players to a new map.
|
%% @doc Warp all players to a new map.
|
||||||
|
%% @todo Move that in a psu module.
|
||||||
warp(QuestID, ZoneID, MapID, EntryID) ->
|
warp(QuestID, ZoneID, MapID, EntryID) ->
|
||||||
lists:foreach(fun(User) -> User#users.pid ! {psu_warp, QuestID, ZoneID, MapID, EntryID} end, egs_db:users_select_all()).
|
lists:foreach(fun(User) -> User#users.pid ! {psu_warp, QuestID, ZoneID, MapID, EntryID} end, egs_db:users_select_all()).
|
||||||
|
|
||||||
%% @doc Warp one player to a new map.
|
%% @doc Warp one player to a new map.
|
||||||
|
%% @todo Move that in a psu module.
|
||||||
warp(GID, QuestID, ZoneID, MapID, EntryID) ->
|
warp(GID, QuestID, ZoneID, MapID, EntryID) ->
|
||||||
User = egs_db:users_select(GID),
|
User = egs_db:users_select(GID),
|
||||||
User#users.pid ! {psu_warp, QuestID, ZoneID, MapID, EntryID}.
|
User#users.pid ! {psu_warp, QuestID, ZoneID, MapID, EntryID}.
|
||||||
|
75
src/egs_app.erl
Normal file
75
src/egs_app.erl
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
%% @author Loïc Hoguin <essen@dev-extend.eu>
|
||||||
|
%% @copyright 2010 Loïc Hoguin.
|
||||||
|
%% @doc Callbacks for the egs application.
|
||||||
|
%%
|
||||||
|
%% This file is part of EGS.
|
||||||
|
%%
|
||||||
|
%% EGS 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 3 of the License, or
|
||||||
|
%% (at your option) any later version.
|
||||||
|
%%
|
||||||
|
%% EGS 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 EGS. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
-module(egs_app).
|
||||||
|
-behaviour(application).
|
||||||
|
-export([start/2, stop/1]).
|
||||||
|
|
||||||
|
-include("include/records.hrl").
|
||||||
|
|
||||||
|
%% @spec start(_Type, _StartArgs) -> ServerRet
|
||||||
|
%% @doc application start callback for egs.
|
||||||
|
start(_Type, _StartArgs) ->
|
||||||
|
case is_fresh_startup() of
|
||||||
|
true ->
|
||||||
|
db_init();
|
||||||
|
{exists, Tables} ->
|
||||||
|
ok = mnesia:wait_for_tables(Tables, 20000)
|
||||||
|
end,
|
||||||
|
egs_sup:start_link().
|
||||||
|
|
||||||
|
%% @spec stop(_State) -> ServerRet
|
||||||
|
%% @doc application stop callback for egs.
|
||||||
|
stop(_State) ->
|
||||||
|
ok.
|
||||||
|
|
||||||
|
%% @spec is_fresh_startup() -> true | false
|
||||||
|
%% @doc Returns true if mnesia has not been initialized with
|
||||||
|
%% the sherl schema.
|
||||||
|
%% Thanks to Dale Harvey for this function posted to
|
||||||
|
%% the erlang questions mailing list.
|
||||||
|
is_fresh_startup() ->
|
||||||
|
Node = node(),
|
||||||
|
case mnesia:system_info(tables) of
|
||||||
|
[schema] -> true;
|
||||||
|
Tables ->
|
||||||
|
case mnesia:table_info(schema, cookie) of
|
||||||
|
{_, Node} -> {exists, Tables};
|
||||||
|
_ -> true
|
||||||
|
end
|
||||||
|
end.
|
||||||
|
|
||||||
|
%% @todo doc
|
||||||
|
%% @todo Rename ids to counters. Remove objects and use a FSM instead.
|
||||||
|
db_init() ->
|
||||||
|
Nodes = [node()],
|
||||||
|
case mnesia:system_info(is_running) of
|
||||||
|
yes ->
|
||||||
|
error_logger:info_report("stopping mnesia"),
|
||||||
|
mnesia:stop();
|
||||||
|
_ -> pass
|
||||||
|
end,
|
||||||
|
mnesia:create_schema(Nodes),
|
||||||
|
error_logger:info_report("mnesia schema created"),
|
||||||
|
error_logger:info_report("starting mnesia"),
|
||||||
|
mnesia:start(),
|
||||||
|
mnesia:create_table(ids, [{attributes, record_info(fields, ids)}]),
|
||||||
|
mnesia:create_table(objects, [{attributes, record_info(fields, objects)}]),
|
||||||
|
mnesia:create_table(users, [{attributes, record_info(fields, users)}]),
|
||||||
|
error_logger:info_report("mnesia tables created").
|
@ -26,8 +26,8 @@
|
|||||||
|
|
||||||
start() ->
|
start() ->
|
||||||
KeepAlivePid = spawn_link(?MODULE, keepalive, []),
|
KeepAlivePid = spawn_link(?MODULE, keepalive, []),
|
||||||
CleanupPid = spawn_link(?MODULE, cleanup, []),
|
spawn_link(?MODULE, cleanup, []),
|
||||||
[{keepalive, KeepAlivePid}, {cleanup, CleanupPid}].
|
{ok, KeepAlivePid}.
|
||||||
|
|
||||||
%% @doc Cleanup the users table of failures to log into the game server.
|
%% @doc Cleanup the users table of failures to log into the game server.
|
||||||
|
|
||||||
|
@ -31,15 +31,6 @@ do(Q) ->
|
|||||||
{atomic, Val} = mnesia:transaction(F),
|
{atomic, Val} = mnesia:transaction(F),
|
||||||
Val.
|
Val.
|
||||||
|
|
||||||
%% @doc Create the database.
|
|
||||||
|
|
||||||
create() ->
|
|
||||||
mnesia:create_schema([node()]),
|
|
||||||
mnesia:start(),
|
|
||||||
mnesia:create_table(ids, [{attributes, record_info(fields, ids)}]),
|
|
||||||
mnesia:create_table(objects, [{attributes, record_info(fields, objects)}]),
|
|
||||||
mnesia:create_table(users, [{attributes, record_info(fields, users)}]).
|
|
||||||
|
|
||||||
%% @doc Retrieve the next unique ID.
|
%% @doc Retrieve the next unique ID.
|
||||||
|
|
||||||
next(Type) ->
|
next(Type) ->
|
||||||
|
@ -27,9 +27,9 @@
|
|||||||
%% @doc Start the game server.
|
%% @doc Start the game server.
|
||||||
|
|
||||||
start() ->
|
start() ->
|
||||||
SPid = spawn(?MODULE, supervisor_init, []),
|
SPid = spawn_link(?MODULE, supervisor_init, []),
|
||||||
LPid = spawn(?MODULE, listen, [SPid]),
|
LPid = spawn_link(?MODULE, listen, [SPid]),
|
||||||
[{listener, LPid}, {supervisor, SPid}].
|
{ok, LPid}.
|
||||||
|
|
||||||
%% @doc Game processes supervisor initialization.
|
%% @doc Game processes supervisor initialization.
|
||||||
|
|
||||||
|
@ -26,10 +26,10 @@
|
|||||||
%% @doc Start the login server. Currently AOTI JP and US only.
|
%% @doc Start the login server. Currently AOTI JP and US only.
|
||||||
|
|
||||||
start() ->
|
start() ->
|
||||||
JPPidE1 = spawn_link(?MODULE, listen, [?LOGIN_PORT_JP_ONE, 10000001]),
|
spawn_link(?MODULE, listen, [?LOGIN_PORT_JP_ONE, 10000001]),
|
||||||
JPPidE2 = spawn_link(?MODULE, listen, [?LOGIN_PORT_JP_TWO, 20000001]),
|
spawn_link(?MODULE, listen, [?LOGIN_PORT_JP_TWO, 20000001]),
|
||||||
USPid = spawn_link(?MODULE, listen, [?LOGIN_PORT_US, 30000001]),
|
USPid = spawn_link(?MODULE, listen, [?LOGIN_PORT_US, 30000001]),
|
||||||
[{jp_e1, JPPidE1}, {jp_e2, JPPidE2}, {us, USPid}].
|
{ok, USPid}.
|
||||||
|
|
||||||
%% @doc Listen for connections.
|
%% @doc Listen for connections.
|
||||||
|
|
||||||
|
@ -25,9 +25,9 @@
|
|||||||
%% @doc Start the patch server. Currently supports AOTI US and JP.
|
%% @doc Start the patch server. Currently supports AOTI US and JP.
|
||||||
|
|
||||||
start() ->
|
start() ->
|
||||||
JPPid = spawn_link(?MODULE, listen, [?PATCH_PORT_JP]),
|
spawn_link(?MODULE, listen, [?PATCH_PORT_JP]),
|
||||||
USPid = spawn_link(?MODULE, listen, [?PATCH_PORT_US]),
|
USPid = spawn_link(?MODULE, listen, [?PATCH_PORT_US]),
|
||||||
[{jp, JPPid}, {us, USPid}].
|
{ok, USPid}.
|
||||||
|
|
||||||
%% @doc Listen for connections.
|
%% @doc Listen for connections.
|
||||||
|
|
||||||
|
54
src/egs_sup.erl
Normal file
54
src/egs_sup.erl
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
%% @author Loïc Hoguin <essen@dev-extend.eu>
|
||||||
|
%% @copyright 2010 Loïc Hoguin.
|
||||||
|
%% @doc Supervisor for the egs application.
|
||||||
|
%%
|
||||||
|
%% This file is part of EGS.
|
||||||
|
%%
|
||||||
|
%% EGS 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 3 of the License, or
|
||||||
|
%% (at your option) any later version.
|
||||||
|
%%
|
||||||
|
%% EGS 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 EGS. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
-module(egs_sup).
|
||||||
|
-behaviour(supervisor).
|
||||||
|
-export([init/1]). %% Supervisor callbacks.
|
||||||
|
-export([start_link/0, upgrade/0]). %% Other functions.
|
||||||
|
|
||||||
|
%% @spec start_link() -> ServerRet
|
||||||
|
%% @doc API for starting the supervisor.
|
||||||
|
start_link() ->
|
||||||
|
supervisor:start_link({local, ?MODULE}, ?MODULE, []).
|
||||||
|
|
||||||
|
%% @spec upgrade() -> ok
|
||||||
|
%% @doc Add processes if necessary.
|
||||||
|
upgrade() ->
|
||||||
|
{ok, {_, Specs}} = init([]),
|
||||||
|
Old = sets:from_list(
|
||||||
|
[Name || {Name, _, _, _} <- supervisor:which_children(?MODULE)]),
|
||||||
|
New = sets:from_list([Name || {Name, _, _, _, _, _} <- Specs]),
|
||||||
|
Kill = sets:subtract(Old, New),
|
||||||
|
sets:fold(fun (Id, ok) ->
|
||||||
|
supervisor:terminate_child(?MODULE, Id),
|
||||||
|
supervisor:delete_child(?MODULE, Id),
|
||||||
|
ok
|
||||||
|
end, ok, Kill),
|
||||||
|
[supervisor:start_child(?MODULE, Spec) || Spec <- Specs],
|
||||||
|
ok.
|
||||||
|
|
||||||
|
%% @spec init([]) -> SupervisorTree
|
||||||
|
%% @doc supervisor callback.
|
||||||
|
init([]) ->
|
||||||
|
%% Start egs_cron, egs_game, egs_login, egs_patch. To be replaced by configurable modules.
|
||||||
|
Processes = [{egs_cron, {egs_cron, start, []}, permanent, 5000, worker, dynamic},
|
||||||
|
{egs_game, {egs_game, start, []}, permanent, 5000, worker, dynamic},
|
||||||
|
{egs_login, {egs_login, start, []}, permanent, 5000, worker, dynamic},
|
||||||
|
{egs_patch, {egs_patch, start, []}, permanent, 5000, worker, dynamic}],
|
||||||
|
{ok, {{one_for_one, 10, 10}, Processes}}.
|
Loading…
Reference in New Issue
Block a user