diff --git a/apps/egs/include/records.hrl b/apps/egs/include/records.hrl
index 680368a..ca5f35a 100644
--- a/apps/egs/include/records.hrl
+++ b/apps/egs/include/records.hrl
@@ -20,12 +20,15 @@
%% Records.
%% @doc Client state. One per connected client.
--record(client, {
+-record(egs_net, {
socket :: ssl:sslsocket(),
transport :: module(),
+ handler :: module(),
+ buffer = <<>> :: binary(),
+ keepalive = false :: boolean(),
gid = 0 :: egs:gid(),
- slot = 0 :: egs:character_slot(), %% @todo Probably should remove this one from the state.
lid = 16#ffff :: egs:lid(),
+ slot = 0 :: 0..3,
areanb = 0 :: non_neg_integer()
}).
@@ -70,7 +73,8 @@
prev_area = {0, 0, 0} :: egs:area(),
prev_entryid = 0 :: egs:entryid(),
%% To be moved or deleted later on.
- instancepid :: pid()
+ instancepid :: pid(),
+ char
}).
%% Past this point needs to be reviewed.
diff --git a/apps/egs/rebar.config b/apps/egs/rebar.config
index 6d687c8..bec3704 100644
--- a/apps/egs/rebar.config
+++ b/apps/egs/rebar.config
@@ -1,4 +1,6 @@
{deps, [
+ {erlson, ".*", {git, "https://github.com/alavrik/erlson.git", "HEAD"}},
{ex_reloader, ".*", {git, "git://github.com/extend/ex_reloader.git", "HEAD"}},
{cowboy, ".*", {git, "git://github.com/extend/cowboy.git", "HEAD"}}
]}.
+{plugins, [erlson_rebar_plugin]}.
diff --git a/apps/egs/src/egs.erl b/apps/egs/src/egs.erl
index f149065..3e29650 100644
--- a/apps/egs/src/egs.erl
+++ b/apps/egs/src/egs.erl
@@ -46,7 +46,6 @@ start() ->
ensure_started(crypto),
ensure_started(public_key),
ensure_started(ssl),
- ssl:seed(crypto:rand_bytes(256)),
ensure_started(cowboy),
application:start(egs).
diff --git a/apps/egs/src/egs_app.erl b/apps/egs/src/egs_app.erl
index f1f2071..44cfacf 100644
--- a/apps/egs/src/egs_app.erl
+++ b/apps/egs/src/egs_app.erl
@@ -34,6 +34,7 @@ start(_Type, _StartArgs) ->
{ok, Pid} = egs_sup:start_link(),
application:set_env(egs_patch, patch_ports, egs_conf:read(patch_ports)),
application:start(egs_patch),
+ application:start(egs_store),
start_login_listeners(egs_conf:read(login_ports)),
{_ServerIP, GamePort} = egs_conf:read(game_server),
{ok, _GamePid} = cowboy:start_listener({game, GamePort}, 10,
diff --git a/apps/egs/src/egs_char.erl b/apps/egs/src/egs_char.erl
new file mode 100644
index 0000000..4c79c52
--- /dev/null
+++ b/apps/egs/src/egs_char.erl
@@ -0,0 +1,64 @@
+-module(egs_char).
+
+-export([new/6]).
+
+-include_lib("erlson/include/erlson.hrl").
+
+%% @todo Add the current location for backtopreviousfield
+new(Slot, Name, Race, Gender, Class, Appearance) ->
+ #{
+ type=player,
+
+ slot=Slot,
+ name=Name,
+ race=Race,
+ gender=Gender,
+ class=Class,
+ appearance=Appearance,
+
+ level=1,
+ exp=0,
+
+ hunter_level=1,
+ hunter_exp=0,
+ ranger_level=1,
+ ranger_exp=0,
+ force_level=1,
+ force_exp=0,
+ acro_level=1,
+ acro_level=0,
+
+ blast_bar=0,
+ luck=3,
+ playtime=0,
+
+ %% current_uni,
+ %% current location + entryid
+ %% previous location + entryid
+ %% pids for such
+
+ money=1000000,
+ inventory=[],
+
+ card_comment= <<>>,
+
+ options=#{
+ brightness=4,
+ buttonhelp=0,
+ cam1stx=0,
+ cam1sty=0,
+ cam3rdx=0,
+ cam3rdy=0,
+ controller=0,
+ cursorpos=0,
+ cutin=0,
+ fnkeys=0,
+ lockon=0,
+ musicvolume=0,
+ radarmap=0,
+ sfxvolume=0,
+ sound=0,
+ textspeed=0,
+ vibration=0,
+ weaponswap=0
+ }}.
diff --git a/apps/egs/src/egs_char_select.erl b/apps/egs/src/egs_char_select.erl
index c57fa1e..1fdeadb 100644
--- a/apps/egs/src/egs_char_select.erl
+++ b/apps/egs/src/egs_char_select.erl
@@ -18,14 +18,11 @@
%% along with EGS. If not, see .
-module(egs_char_select).
--export([keepalive/1, info/2, cast/3, raw/3, event/2]).
+-export([info/2, cast/3, event/2]).
+-include_lib("erlson/include/erlson.hrl").
-include("include/records.hrl").
-%% @doc Send a keepalive.
-keepalive(Client) ->
- egs_proto:send_keepalive(Client).
-
%% @doc We don't expect any message here.
info(_Msg, _Client) ->
ok.
@@ -34,75 +31,56 @@ info(_Msg, _Client) ->
cast(_Command, _Data, _Client) ->
ok.
-%% @doc Dismiss all raw commands with a log notice.
-%% @todo Have a log event handler instead.
-raw(Command, _Data, Client) ->
- io:format("~p (~p): dismissed command ~4.16.0b~n", [?MODULE, Client#client.gid, Command]).
-
%% Events.
%% @doc Character screen selection request and delivery.
-event(char_select_request, Client=#client{gid=GID}) ->
- Folder = egs_accounts:get_folder(GID),
- egs_proto:send_0d03(data_load(Folder, 0), data_load(Folder, 1), data_load(Folder, 2), data_load(Folder, 3), Client);
+event(account_characters_request, Client) ->
+ {ok, Characters}
+ = egs_store:load_characters(egs_net:get_gid(Client), [0, 1, 2, 3]),
+ egs_net:account_characters_response(Characters, Client);
-%% @doc The options default to 0 for everything except brightness to 4.
-%% @todo Don't forget to check for the character's name.
+%% @todo Don't forget to check for the character's name (in egs_net).
%% 00F7 is the RGBA color control character.
%% 03F7 is the RGB color control character.
-event({char_select_create, Slot, CharBin}, #client{gid=GID}) ->
- %% check for valid character appearance
- %~ << _Name:512, RaceID:8, GenderID:8, _TypeID:8, AppearanceBin:776/bits, _/bits >> = CharBin,
- %~ Race = proplists:get_value(RaceID, [{0, human}, {1, newman}, {2, cast}, {3, beast}]),
- %~ Gender = proplists:get_value(GenderID, [{0, male}, {1, female}]),
- %~ Appearance = psu_appearance:binary_to_tuple(Race, AppearanceBin),
- %~ psu_appearance:validate_char_create(Race, Gender, Appearance),
- %% end of check, continue doing it wrong past that point for now
- Folder = egs_accounts:get_folder(GID),
- Dir = io_lib:format("save/~s", [Folder]),
- File = io_lib:format("~s/~b-character", [Dir, Slot]),
- _ = file:make_dir(Dir),
- file:write_file(File, CharBin),
- file:write_file(io_lib:format("~s.options", [File]), << 0:128, 4, 0:56 >>);
+event({account_create_character, Slot, Name, Race, Gender, Class, Appearance},
+ Client) ->
+ Character = egs_char:new(Slot, Name, Race, Gender, Class, Appearance),
+ egs_store:save_character(egs_net:get_gid(Client), Slot, Character);
%% @doc Load the selected character into the game's default universe.
%% @todo Isn't very pretty to call egs_game from here but that will do for now.
-event({char_select_enter, Slot, _BackToPreviousField}, Client=#client{gid=GID}) ->
- Folder = egs_accounts:get_folder(GID),
- [{status, 1}, {char, CharBin}, {options, OptionsBin}] = data_load(Folder, Slot),
- << Name:512/bits, RaceBin:8, GenderBin:8, ClassBin:8, AppearanceBin:776/bits, _/bits >> = CharBin,
- Race = psu_characters:race_binary_to_atom(RaceBin),
- Gender = psu_characters:gender_binary_to_atom(GenderBin),
- Class = psu_characters:class_binary_to_atom(ClassBin),
- Appearance = psu_appearance:binary_to_tuple(Race, AppearanceBin),
- Options = psu_characters:options_binary_to_tuple(OptionsBin),
+event({system_character_select, Slot, _BackToPreviousField}, Client) ->
+ GID = egs_net:get_gid(Client),
+ {ok, 1, Char} = egs_store:load_character(GID, Slot),
UniID = egs_universes:defaultid(),
egs_universes:enter(UniID),
- User = #users{gid=GID, pid=self(), uni=UniID, slot=Slot, name=Name, race=Race, gender=Gender,
- class=Class, appearance=Appearance, options=Options, area={1100000, 0, 4}, entryid=0},
+ %% @todo Handle users properly, just giving Character directly.
+ Name = Char.name,
+ NameBin = << Name/binary, 0:(512 - bit_size(Name)) >>,
+ Race = Char.race,
+ Gender = Char.gender,
+ Class = Char.class,
+ Appearance = Char.appearance,
+ Options = Char.options,
+ User = #users{gid=GID, pid=self(), uni=UniID, slot=Slot,
+ name=NameBin, race=Race, gender=Gender, class=Class,
+ appearance=Appearance, options=Options,
+ area={1100000, 0, 4}, entryid=0, char=Char},
egs_users:write(User),
egs_users:item_add(GID, 16#11010000, #psu_special_item_variables{}),
egs_users:item_add(GID, 16#11020000, #psu_special_item_variables{}),
egs_users:item_add(GID, 16#11020100, #psu_special_item_variables{}),
egs_users:item_add(GID, 16#11020200, #psu_special_item_variables{}),
- egs_users:item_add(GID, 16#01010900, #psu_striking_weapon_item_variables{current_pp=99, max_pp=100, element=#psu_element{type=1, percent=50}}),
- egs_users:item_add(GID, 16#01010a00, #psu_striking_weapon_item_variables{current_pp=99, max_pp=100, element=#psu_element{type=2, percent=50}}),
- egs_users:item_add(GID, 16#01010b00, #psu_striking_weapon_item_variables{current_pp=99, max_pp=100, element=#psu_element{type=3, percent=50}}),
+ egs_users:item_add(GID, 16#01010900, #psu_striking_weapon_item_variables{
+ current_pp=99, max_pp=100, element=#psu_element{type=1, percent=50}}),
+ egs_users:item_add(GID, 16#01010a00, #psu_striking_weapon_item_variables{
+ current_pp=99, max_pp=100, element=#psu_element{type=2, percent=50}}),
+ egs_users:item_add(GID, 16#01010b00, #psu_striking_weapon_item_variables{
+ current_pp=99, max_pp=100, element=#psu_element{type=3, percent=50}}),
{ok, User2} = egs_users:read(GID),
- Client2 = Client#client{slot=Slot},
- egs_game:char_load(User2, Client2),
- {ok, egs_game, Client2}.
+ egs_game:char_load(User2, Client),
+ Client2 = egs_net:set_handler(egs_game, Client),
+ {ok, Client2};
-%% Internal.
-
-%% @doc Load the given character's data.
-%% @todo This function is temporary until we get permanent accounts.
-data_load(Folder, Number) ->
- Filename = io_lib:format("save/~s/~b-character", [Folder, Number]),
- case file:read_file(Filename) of
- {ok, Char} ->
- {ok, Options} = file:read_file(io_lib:format("~s.options", [Filename])),
- [{status, 1}, {char, Char}, {options, Options}];
- {error, _Reason} ->
- [{status, 0}, {char, << 0:2208 >>}]
- end.
+event({client_hardware, GPU, CPU}, Client) ->
+ ok.
diff --git a/apps/egs/src/egs_game.erl b/apps/egs/src/egs_game.erl
index 4644a4e..d873b63 100644
--- a/apps/egs/src/egs_game.erl
+++ b/apps/egs/src/egs_game.erl
@@ -18,17 +18,13 @@
%% along with EGS. If not, see .
-module(egs_game).
--export([keepalive/1, info/2, cast/3, raw/3, event/2]).
+-export([info/2, cast/2, raw/3, event/2]).
-export([char_load/2]). %% Hopefully temporary export.
-include("include/records.hrl").
-%% @doc Send a keepalive.
-keepalive(Client) ->
- egs_proto:send_keepalive(Client).
-
%% @doc Forward the broadcasted command to the client.
-info({egs, cast, Command}, Client=#client{gid=GID}) ->
+info({egs, cast, Command}, Client=#egs_net{gid=GID}) ->
<< A:64/bits, _:32, B:96/bits, _:64, C/bits >> = Command,
egs_proto:packet_send(Client, << A/binary, 16#00011300:32, B/binary, 16#00011300:32, GID:32/little, C/binary >>);
@@ -60,30 +56,30 @@ info({egs, warp, QuestID, ZoneID, MapID, EntryID}, Client) ->
%% @todo Handle broadcasting better than that. Review the commands at the same time.
%% @doc Position change. Save the position and then dispatch it.
-cast(16#0503, Data, Client=#client{gid=GID}) ->
+cast({16#0503, Data}, Client=#egs_net{gid=GID}) ->
<< _:424, Dir:24/little, _PrevCoords:96, X:32/little-float, Y:32/little-float, Z:32/little-float,
QuestID:32/little, ZoneID:32/little, MapID:32/little, EntryID:32/little, _:32 >> = Data,
FloatDir = Dir / 46603.375,
{ok, User} = egs_users:read(GID),
NewUser = User#users{pos={X, Y, Z, FloatDir}, area={QuestID, ZoneID, MapID}, entryid=EntryID},
egs_users:write(NewUser),
- cast(valid, Data, Client);
+ cast({valid, Data}, Client);
%% @doc Stand still. Save the position and then dispatch it.
-cast(16#0514, Data, Client=#client{gid=GID}) ->
+cast({16#0514, Data}, Client=#egs_net{gid=GID}) ->
<< _:424, Dir:24/little, X:32/little-float, Y:32/little-float, Z:32/little-float,
QuestID:32/little, ZoneID:32/little, MapID:32/little, EntryID:32/little, _/bits >> = Data,
FloatDir = Dir / 46603.375,
{ok, User} = egs_users:read(GID),
NewUser = User#users{pos={X, Y, Z, FloatDir}, area={QuestID, ZoneID, MapID}, entryid=EntryID},
egs_users:write(NewUser),
- cast(valid, Data, Client);
+ cast({valid, Data}, Client);
%% @doc Default broadcast handler. Dispatch the command to everyone.
%% We clean up the command and use the real GID and LID of the user, disregarding what was sent and possibly tampered with.
%% Only a handful of commands are allowed to broadcast. An user tampering with it would get disconnected instantly.
%% @todo Don't query the user data everytime! Keep the needed information in the Client.
-cast(Command, Data, #client{gid=GID, lid=LID})
+cast({Command, Data}, #egs_net{gid=GID, lid=LID})
when Command =:= 16#0101;
Command =:= 16#0102;
Command =:= 16#0104;
@@ -101,7 +97,7 @@ cast(Command, Data, #client{gid=GID, lid=LID})
%% @todo Handle this packet properly.
%% @todo Spawn cleared response event shouldn't be handled following this packet but when we see the spawn actually dead HP-wise.
%% @todo Type shouldn't be :32 but it seems when the later 16 have something it's not a spawn event.
-raw(16#0402, << _:352, Data/bits >>, Client=#client{gid=GID}) ->
+raw(16#0402, << _:352, Data/bits >>, Client=#egs_net{gid=GID}) ->
<< SpawnID:32/little, _:64, Type:32/little, _:64 >> = Data,
case Type of
7 -> % spawn cleared @todo 1201 sent back with same values apparently, but not always
@@ -119,23 +115,23 @@ raw(16#0402, << _:352, Data/bits >>, Client=#client{gid=GID}) ->
%% @todo 3rd Unsafe Passage C, EventID 10 BlockID 2 = mission cleared?
raw(16#0404, << _:352, Data/bits >>, Client) ->
<< EventID:8, BlockID:8, _:16, Value:8, _/bits >> = Data,
- io:format("~p: unknown command 0404: eventid ~b blockid ~b value ~b~n", [Client#client.gid, EventID, BlockID, Value]),
+ io:format("~p: unknown command 0404: eventid ~b blockid ~b value ~b~n", [Client#egs_net.gid, EventID, BlockID, Value]),
egs_proto:send_1205(EventID, BlockID, Value, Client);
%% @todo Used in the tutorial. Not sure what it does. Give an item (the PA) maybe?
%% @todo Probably should ignore that until more is known.
-raw(16#0a09, _Data, Client=#client{gid=GID}) ->
+raw(16#0a09, _Data, Client=#egs_net{gid=GID}) ->
egs_proto:packet_send(Client, << 16#0a090300:32, 0:32, 16#00011300:32, GID:32/little, 0:64, 16#00011300:32, GID:32/little, 0:64, 16#00003300:32, 0:32 >>);
%% @todo Figure out this command.
-raw(16#0c11, << _:352, A:32/little, B:32/little >>, Client=#client{gid=GID}) ->
+raw(16#0c11, << _:352, A:32/little, B:32/little >>, Client=#egs_net{gid=GID}) ->
io:format("~p: 0c11 ~p ~p~n", [GID, A, B]),
egs_proto:packet_send(Client, << 16#0c120300:32, 0:160, 16#00011300:32, GID:32/little, 0:64, A:32/little, 1:32/little >>);
%% @doc Set flag handler. Associate a new flag with the character.
%% Just reply with a success value for now.
%% @todo God save the flags.
-raw(16#0d04, << _:352, Data/bits >>, Client=#client{gid=GID}) ->
+raw(16#0d04, << _:352, Data/bits >>, Client=#egs_net{gid=GID}) ->
<< Flag:128/bits, A:16/bits, _:8, B/bits >> = Data,
io:format("~p: flag handler for ~s~n", [GID, re:replace(Flag, "\\0+", "", [global, {return, binary}])]),
egs_proto:packet_send(Client, << 16#0d040300:32, 0:160, 16#00011300:32, GID:32/little, 0:64, Flag/binary, A/binary, 1, B/binary >>);
@@ -143,7 +139,7 @@ raw(16#0d04, << _:352, Data/bits >>, Client=#client{gid=GID}) ->
%% @doc Initialize a vehicle object.
%% @todo Find what are the many values, including the odd Whut value (and whether it's used in the reply).
%% @todo Separate the reply.
-raw(16#0f00, << _:352, Data/bits >>, Client=#client{gid=GID}) ->
+raw(16#0f00, << _:352, Data/bits >>, Client=#egs_net{gid=GID}) ->
<< A:32/little, 0:16, B:16/little, 0:16, C:16/little, 0, Whut:8, D:16/little, 0:16,
E:16/little, 0:16, F:16/little, G:16/little, H:16/little, I:32/little >> = Data,
io:format("~p: init vehicle: ~b ~b ~b ~b ~b ~b ~b ~b ~b ~b~n", [GID, A, B, C, Whut, D, E, F, G, H, I]),
@@ -154,7 +150,7 @@ raw(16#0f00, << _:352, Data/bits >>, Client=#client{gid=GID}) ->
%% @doc Enter vehicle.
%% @todo Separate the reply.
-raw(16#0f02, << _:352, Data/bits >>, Client=#client{gid=GID}) ->
+raw(16#0f02, << _:352, Data/bits >>, Client=#egs_net{gid=GID}) ->
<< A:32/little, B:32/little, C:32/little >> = Data,
io:format("~p: enter vehicle: ~b ~b ~b~n", [GID, A, B, C]),
HP = 100,
@@ -162,7 +158,7 @@ raw(16#0f02, << _:352, Data/bits >>, Client=#client{gid=GID}) ->
%% @doc Sent right after entering the vehicle. Can't move without it.
%% @todo Separate the reply.
-raw(16#0f07, << _:352, Data/bits >>, Client=#client{gid=GID}) ->
+raw(16#0f07, << _:352, Data/bits >>, Client=#egs_net{gid=GID}) ->
<< A:32/little, B:32/little >> = Data,
io:format("~p: after enter vehicle: ~b ~b~n", [GID, A, B]),
egs_proto:packet_send(Client, << 16#120f0300:32, 0:160, 16#00011300:32, GID:32/little, 0:64, A:32/little, B:32/little >>);
@@ -183,13 +179,13 @@ raw(16#1112, << _:352, Data/bits >>, Client) ->
%% @todo Not sure yet. Value is probably a TargetID. Used in Airboard Rally. Replying with the same value starts the race.
raw(16#1216, << _:352, Data/bits >>, Client) ->
<< Value:32/little >> = Data,
- io:format("~p: command 1216 with value ~b~n", [Client#client.gid, Value]),
+ io:format("~p: command 1216 with value ~b~n", [Client#egs_net.gid, Value]),
egs_proto:send_1216(Value, Client);
%% @doc Dismiss all unknown raw commands with a log notice.
%% @todo Have a log event handler instead.
raw(Command, _Data, Client) ->
- io:format("~p (~p): dismissed command ~4.16.0b~n", [?MODULE, Client#client.gid, Command]).
+ io:format("~p (~p): dismissed command ~4.16.0b~n", [?MODULE, Client#egs_net.gid, Command]).
%% Events.
@@ -199,8 +195,8 @@ raw(Command, _Data, Client) ->
event({area_change, QuestID, ZoneID, MapID, EntryID}, Client) ->
event({area_change, QuestID, ZoneID, MapID, EntryID, 16#ffffffff}, Client);
event({area_change, QuestID, ZoneID, MapID, EntryID, PartyPos=16#ffffffff}, Client) ->
- io:format("~p: area change (~b,~b,~b,~b,~b)~n", [Client#client.gid, QuestID, ZoneID, MapID, EntryID, PartyPos]),
- {ok, OldUser} = egs_users:read(Client#client.gid),
+ io:format("~p: area change (~b,~b,~b,~b,~b)~n", [Client#egs_net.gid, QuestID, ZoneID, MapID, EntryID, PartyPos]),
+ {ok, OldUser} = egs_users:read(Client#egs_net.gid),
{OldQuestID, OldZoneID, _OldMapID} = OldUser#users.area,
QuestChange = OldQuestID /= QuestID,
ZoneChange = if OldQuestID =:= QuestID, OldZoneID =:= ZoneID -> false; true -> true end,
@@ -221,7 +217,7 @@ event({area_change, QuestID, ZoneID, MapID, EntryID, PartyPos=16#ffffffff}, Clie
ZonePid = egs_quests:zone_pid(User2#users.questpid, ZoneID),
egs_zones:leave(User2#users.zonepid, User2#users.gid),
NewLID = egs_zones:enter(ZonePid, User2#users.gid),
- NewClient = Client#client{lid=NewLID},
+ NewClient = Client#egs_net{lid=NewLID},
{ok, User3} = egs_users:read(User2#users.gid),
egs_proto:send_0a05(NewClient),
egs_proto:send_0111(User3, 6, NewClient),
@@ -236,7 +232,7 @@ event({area_change, QuestID, ZoneID, MapID, EntryID, PartyPos=16#ffffffff}, Clie
%% Save the user.
egs_users:write(User3),
%% Load the player location.
- Client2 = Client1#client{areanb=Client#client.areanb + 1},
+ Client2 = Client1#egs_net{areanb=Client#egs_net.areanb + 1},
egs_proto:send_0205(User3, IsSeasonal, Client2),
egs_proto:send_100e(User3#users.area, User3#users.entryid, AreaShortName, Client2),
%% Load the zone objects.
@@ -258,19 +254,19 @@ event({area_change, QuestID, ZoneID, MapID, EntryID, PartyPos=16#ffffffff}, Clie
true -> ignore
end,
%% End of loading.
- Client3 = Client2#client{areanb=Client2#client.areanb + 1},
+ Client3 = Client2#egs_net{areanb=Client2#egs_net.areanb + 1},
egs_proto:send_0208(Client3),
egs_proto:send_0236(Client3),
%% @todo Load APC characters.
{ok, Client3};
event({area_change, QuestID, ZoneID, MapID, EntryID, PartyPos}, Client) ->
- io:format("~p: area change (~b,~b,~b,~b,~b)~n", [Client#client.gid, QuestID, ZoneID, MapID, EntryID, PartyPos]),
+ io:format("~p: area change (~b,~b,~b,~b,~b)~n", [Client#egs_net.gid, QuestID, ZoneID, MapID, EntryID, PartyPos]),
ignore;
%% @doc After the character has been (re)loaded, change the area he's in.
%% @todo The area_change event should probably not change the user's values.
%% @todo Remove that ugly code when the above is done.
-event(char_load_complete, Client=#client{gid=GID}) ->
+event(system_character_load_complete, Client=#egs_net{gid=GID}) ->
{ok, User=#users{area={QuestID, ZoneID, MapID}, entryid=EntryID}} = egs_users:read(GID),
egs_users:write(User#users{area={0, 0, 0}, entryid=0}),
event({area_change, QuestID, ZoneID, MapID, EntryID}, Client);
@@ -281,7 +277,7 @@ event(char_load_complete, Client=#client{gid=GID}) ->
%% @todo In the case of NPC characters, when FromTypeID is 00001d00, check that the NPC is in the party and broadcast only to the party (probably).
%% @todo When the game doesn't find an NPC (probably) and forces it to talk like in the tutorial mission it seems FromTypeID, FromGID and Name are all 0.
%% @todo Make sure modifiers have correct values.
-event({chat, _FromTypeID, FromGID, _FromName, Modifiers, ChatMsg}, #client{gid=UserGID}) ->
+event({chat, _, FromGID, _, Modifiers, _, ChatMsg}, #egs_net{gid=UserGID}) ->
[BcastTypeID, BcastGID, BcastName] = case FromGID of
0 -> %% This probably shouldn't happen. Just make it crash on purpose.
io:format("~p: chat FromGID=0~n", [UserGID]),
@@ -307,7 +303,7 @@ event(counter_background_locations_request, Client) ->
%% @todo Make sure non-mission counters follow the same loading process.
%% @todo Probably validate the From* values, to not send the player back inside a mission.
%% @todo Handle the LID change when entering counters.
-event({counter_enter, CounterID, FromZoneID, FromMapID, FromEntryID}, Client=#client{gid=GID}) ->
+event({counter_enter, CounterID, FromZoneID, FromMapID, FromEntryID}, Client=#egs_net{gid=GID}) ->
io:format("~p: counter load ~b~n", [GID, CounterID]),
{ok, OldUser} = egs_users:read(GID),
FromArea = {element(1, OldUser#users.area), FromZoneID, FromMapID},
@@ -324,7 +320,7 @@ event({counter_enter, CounterID, FromZoneID, FromMapID, FromEntryID}, Client=#cl
egs_proto:send_010d(User, Client),
egs_proto:send_0200(0, mission, Client),
egs_proto:send_020f(ZoneData, 0, 255, Client),
- Client2 = Client#client{areanb=Client#client.areanb + 1},
+ Client2 = Client#egs_net{areanb=Client#egs_net.areanb + 1},
egs_proto:send_0205(User, 0, Client2),
egs_proto:send_100e(CounterID, "Counter", Client2),
egs_proto:send_0215(0, Client2),
@@ -341,7 +337,7 @@ event({counter_enter, CounterID, FromZoneID, FromMapID, FromEntryID}, Client=#cl
undefined -> ignore;
_ -> egs_proto:send_022c(0, 16#12, Client)
end,
- Client3 = Client2#client{areanb=Client2#client.areanb + 1},
+ Client3 = Client2#egs_net{areanb=Client2#egs_net.areanb + 1},
egs_proto:send_0208(Client3),
egs_proto:send_0236(Client3),
{ok, Client3};
@@ -351,7 +347,7 @@ event(counter_join_party_request, Client) ->
egs_proto:send_1701(Client);
%% @doc Leave mission counter handler.
-event(counter_leave, Client=#client{gid=GID}) ->
+event(counter_leave, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
PrevArea = User#users.prev_area,
event({area_change, element(1, PrevArea), element(2, PrevArea), element(3, PrevArea), User#users.prev_entryid}, Client);
@@ -360,11 +356,11 @@ event(counter_leave, Client=#client{gid=GID}) ->
%% @todo Apparently background values 1 2 3 are never used on official servers. Find out why.
%% @todo Rename to counter_bg_request.
event({counter_options_request, CounterID}, Client) ->
- io:format("~p: counter options request ~p~n", [Client#client.gid, CounterID]),
+ io:format("~p: counter options request ~p~n", [Client#egs_net.gid, CounterID]),
egs_proto:send_1711(egs_counters_db:bg(CounterID), Client);
%% @todo Handle when the party already exists! And stop doing it wrong.
-event(counter_party_info_request, Client=#client{gid=GID}) ->
+event(counter_party_info_request, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
egs_proto:send_1706(User#users.name, Client);
@@ -374,16 +370,16 @@ event(counter_party_options_request, Client) ->
%% @doc Request the counter's quest files.
event({counter_quest_files_request, CounterID}, Client) ->
- io:format("~p: counter quest files request ~p~n", [Client#client.gid, CounterID]),
+ io:format("~p: counter quest files request ~p~n", [Client#egs_net.gid, CounterID]),
egs_proto:send_0c06(egs_counters_db:pack(CounterID), Client);
%% @doc Counter available mission list request handler.
event({counter_quest_options_request, CounterID}, Client) ->
- io:format("~p: counter quest options request ~p~n", [Client#client.gid, CounterID]),
+ io:format("~p: counter quest options request ~p~n", [Client#egs_net.gid, CounterID]),
egs_proto:send_0c10(egs_counters_db:opts(CounterID), Client);
%% @todo A and B are mostly unknown. Like most of everything else from the command 0e00...
-event({hit, FromTargetID, ToTargetID, A, B}, Client=#client{gid=GID}) ->
+event({hit, FromTargetID, ToTargetID, A, B}, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
%% hit!
#hit_response{type=Type, user=NewUser, exp=HasEXP, damage=Damage, targethp=TargetHP, targetse=TargetSE, events=Events} = psu_instance:hit(User, FromTargetID, ToTargetID),
@@ -427,7 +423,7 @@ event({item_description_request, ItemID}, Client) ->
%% @todo Currently use a separate file for the data sent for the weapons.
%% @todo TargetGID and TargetLID must be validated, they're either the player's or his NPC characters.
%% @todo Handle NPC characters properly.
-event({item_equip, ItemIndex, TargetGID, TargetLID, A, B}, Client=#client{gid=GID}) ->
+event({item_equip, ItemIndex, TargetGID, TargetLID, A, B}, Client=#egs_net{gid=GID}) ->
case egs_users:item_nth(GID, ItemIndex) of
{ItemID, Variables} when element(1, Variables) =:= psu_special_item_variables ->
<< Category:8, _:24 >> = << ItemID:32 >>,
@@ -462,7 +458,7 @@ event({item_equip, ItemIndex, TargetGID, TargetLID, A, B}, Client=#client{gid=GI
ignore
end;
-event({item_set_trap, ItemIndex, TargetGID, TargetLID, A, B}, Client=#client{gid=GID}) ->
+event({item_set_trap, ItemIndex, TargetGID, TargetLID, A, B}, Client=#egs_net{gid=GID}) ->
{ItemID, _Variables} = egs_users:item_nth(GID, ItemIndex),
egs_users:item_qty_add(GID, ItemIndex, -1),
<< Category:8, _:24 >> = << ItemID:32 >>,
@@ -471,7 +467,7 @@ event({item_set_trap, ItemIndex, TargetGID, TargetLID, A, B}, Client=#client{gid
%% @todo A and B are unknown.
%% @see item_equip
-event({item_unequip, ItemIndex, TargetGID, TargetLID, A, B}, Client=#client{gid=GID}) ->
+event({item_unequip, ItemIndex, TargetGID, TargetLID, A, B}, Client=#egs_net{gid=GID}) ->
Category = case ItemIndex of
% units would be 8, traps would be 12
19 -> 2; % armor
@@ -485,12 +481,12 @@ event({item_unequip, ItemIndex, TargetGID, TargetLID, A, B}, Client=#client{gid=
event(lobby_transport_request, Client) ->
egs_proto:send_0c08(Client);
-event(lumilass_options_request, Client=#client{gid=GID}) ->
+event(lumilass_options_request, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
egs_proto:send_1a03(User, Client);
%% @todo Probably replenish the player HP when entering a non-mission area rather than when aborting the mission?
-event(mission_abort, Client=#client{gid=GID}) ->
+event(mission_abort, Client=#egs_net{gid=GID}) ->
egs_proto:send_1006(11, Client),
{ok, User} = egs_users:read(GID),
%% delete the mission
@@ -509,14 +505,14 @@ event(mission_abort, Client=#client{gid=GID}) ->
%% @todo Forward the mission start to other players of the same party, whatever their location is.
event({mission_start, QuestID}, Client) ->
- io:format("~p: mission start ~b~n", [Client#client.gid, QuestID]),
+ io:format("~p: mission start ~b~n", [Client#egs_net.gid, QuestID]),
egs_proto:send_1020(Client),
egs_proto:send_1015(QuestID, Client),
egs_proto:send_0c02(Client);
%% @doc Force the invite of an NPC character while inside a mission. Mostly used by story missions.
%% Note that the NPC is often removed and reinvited between block/cutscenes.
-event({npc_force_invite, NPCid}, Client=#client{gid=GID}) ->
+event({npc_force_invite, NPCid}, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
%% Create NPC.
io:format("~p: npc force invite ~p~n", [GID, NPCid]),
@@ -544,7 +540,7 @@ event({npc_force_invite, NPCid}, Client=#client{gid=GID}) ->
egs_proto:send_1601(PartyPos, Client);
%% @todo Also at the end send a 101a (NPC:16, PartyPos:16, ffffffff). Not sure about PartyPos.
-event({npc_invite, NPCid}, Client=#client{gid=GID}) ->
+event({npc_invite, NPCid}, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
%% Create NPC.
io:format("~p: invited npcid ~b~n", [GID, NPCid]),
@@ -566,7 +562,7 @@ event({npc_invite, NPCid}, Client=#client{gid=GID}) ->
egs_proto:send_101a(NPCid, PartyPos, Client);
%% @todo Should be 0115(money) 010a03(confirm sale).
-event({npc_shop_buy, ShopItemIndex, QuantityOrColor}, Client=#client{gid=GID}) ->
+event({npc_shop_buy, ShopItemIndex, QuantityOrColor}, Client=#egs_net{gid=GID}) ->
ShopID = egs_users:shop_get(GID),
ItemID = egs_shops_db:nth(ShopID, ShopItemIndex + 1),
io:format("~p: npc shop ~p buy itemid ~8.16.0b quantity/color+1 ~p~n", [GID, ShopID, ItemID, QuantityOrColor]),
@@ -602,12 +598,12 @@ event({npc_shop_buy, ShopItemIndex, QuantityOrColor}, Client=#client{gid=GID}) -
UCS2Name/binary, 0:NamePadding, RarityInt:8, Category:8, SellPrice:32/little, (egs_proto:build_item_constants(Constants))/binary >>);
%% @todo Currently send the normal items shop for all shops, differentiate.
-event({npc_shop_enter, ShopID}, Client=#client{gid=GID}) ->
+event({npc_shop_enter, ShopID}, Client=#egs_net{gid=GID}) ->
io:format("~p: npc shop enter ~p~n", [GID, ShopID]),
egs_users:shop_enter(GID, ShopID),
egs_proto:send_010a(egs_shops_db:read(ShopID), Client);
-event({npc_shop_leave, ShopID}, Client=#client{gid=GID}) ->
+event({npc_shop_leave, ShopID}, Client=#egs_net{gid=GID}) ->
io:format("~p: npc shop leave ~p~n", [GID, ShopID]),
egs_users:shop_leave(GID),
egs_proto:packet_send(Client, << 16#010a0300:32, 0:64, GID:32/little, 0:64, 16#00011300:32,
@@ -615,13 +611,13 @@ event({npc_shop_leave, ShopID}, Client=#client{gid=GID}) ->
%% @todo Should be 0115(money) 010a03(confirm sale).
event({npc_shop_sell, InventoryItemIndex, Quantity}, Client) ->
- io:format("~p: npc shop sell itemindex ~p quantity ~p~n", [Client#client.gid, InventoryItemIndex, Quantity]);
+ io:format("~p: npc shop sell itemindex ~p quantity ~p~n", [Client#egs_net.gid, InventoryItemIndex, Quantity]);
%% @todo First 1a02 value should be non-0.
%% @todo Could the 2nd 1a02 parameter simply be the shop type or something?
%% @todo Although the values replied should be right, they seem mostly ignored by the client.
event({npc_shop_request, ShopID}, Client) ->
- io:format("~p: npc shop request ~p~n", [Client#client.gid, ShopID]),
+ io:format("~p: npc shop request ~p~n", [Client#egs_net.gid, ShopID]),
case ShopID of
80 -> egs_proto:send_1a02(17, 17, 3, 9, Client); %% lumilass
90 -> egs_proto:send_1a02(5, 1, 4, 5, Client); %% parum weapon grinding
@@ -665,14 +661,14 @@ event({object_crystal_activate, ObjectID}, Client) ->
event({object_event_trigger, BlockID, EventID}, Client) ->
egs_proto:send_1205(EventID, BlockID, 0, Client);
-event({object_goggle_target_activate, ObjectID}, Client=#client{gid=GID}) ->
+event({object_goggle_target_activate, ObjectID}, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
{BlockID, EventID} = psu_instance:std_event(User#users.instancepid, element(2, User#users.area), ObjectID),
egs_proto:send_1205(EventID, BlockID, 0, Client),
egs_proto:send_1213(ObjectID, 8, Client);
%% @todo Make NPC characters heal too.
-event({object_healing_pad_tick, [_PartyPos]}, Client=#client{gid=GID}) ->
+event({object_healing_pad_tick, [_PartyPos]}, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
if User#users.currenthp =:= User#users.maxhp -> ignore;
true ->
@@ -684,38 +680,38 @@ event({object_healing_pad_tick, [_PartyPos]}, Client=#client{gid=GID}) ->
egs_proto:send_0111(User2, 4, Client)
end;
-event({object_key_console_enable, ObjectID}, Client=#client{gid=GID}) ->
+event({object_key_console_enable, ObjectID}, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
{BlockID, [EventID|_]} = psu_instance:std_event(User#users.instancepid, element(2, User#users.area), ObjectID),
egs_proto:send_1205(EventID, BlockID, 0, Client),
egs_proto:send_1213(ObjectID, 1, Client);
-event({object_key_console_init, ObjectID}, Client=#client{gid=GID}) ->
+event({object_key_console_init, ObjectID}, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
{BlockID, [_, EventID, _]} = psu_instance:std_event(User#users.instancepid, element(2, User#users.area), ObjectID),
egs_proto:send_1205(EventID, BlockID, 0, Client);
-event({object_key_console_open_gate, ObjectID}, Client=#client{gid=GID}) ->
+event({object_key_console_open_gate, ObjectID}, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
{BlockID, [_, _, EventID]} = psu_instance:std_event(User#users.instancepid, element(2, User#users.area), ObjectID),
egs_proto:send_1205(EventID, BlockID, 0, Client),
egs_proto:send_1213(ObjectID, 1, Client);
%% @todo Now that it's separate from object_key_console_enable, handle it better than that, don't need a list of events.
-event({object_key_enable, ObjectID}, Client=#client{gid=GID}) ->
+event({object_key_enable, ObjectID}, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
{BlockID, [EventID|_]} = psu_instance:std_event(User#users.instancepid, element(2, User#users.area), ObjectID),
egs_proto:send_1205(EventID, BlockID, 0, Client),
egs_proto:send_1213(ObjectID, 1, Client);
%% @todo Some switch objects apparently work differently, like the light switch in Mines in MAG'.
-event({object_switch_off, ObjectID}, Client=#client{gid=GID}) ->
+event({object_switch_off, ObjectID}, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
{BlockID, EventID} = psu_instance:std_event(User#users.instancepid, element(2, User#users.area), ObjectID),
egs_proto:send_1205(EventID, BlockID, 1, Client),
egs_proto:send_1213(ObjectID, 0, Client);
-event({object_switch_on, ObjectID}, Client=#client{gid=GID}) ->
+event({object_switch_on, ObjectID}, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
{BlockID, EventID} = psu_instance:std_event(User#users.instancepid, element(2, User#users.area), ObjectID),
egs_proto:send_1205(EventID, BlockID, 0, Client),
@@ -728,7 +724,7 @@ event({object_vehicle_boost_respawn, ObjectID}, Client) ->
egs_proto:send_1213(ObjectID, 0, Client);
%% @todo Second send_1211 argument should be User#users.lid. Fix when it's correctly handled.
-event({object_warp_take, BlockID, ListNb, ObjectNb}, Client=#client{gid=GID}) ->
+event({object_warp_take, BlockID, ListNb, ObjectNb}, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
Pos = psu_instance:warp_event(User#users.instancepid, element(2, User#users.area), BlockID, ListNb, ObjectNb),
NewUser = User#users{pos=Pos},
@@ -737,7 +733,7 @@ event({object_warp_take, BlockID, ListNb, ObjectNb}, Client=#client{gid=GID}) ->
egs_proto:send_1211(16#ffffffff, 0, 14, 0, Client);
%% @todo Don't send_0204 if the player is removed from the party while in the lobby I guess.
-event({party_remove_member, PartyPos}, Client=#client{gid=GID}) ->
+event({party_remove_member, PartyPos}, Client=#egs_net{gid=GID}) ->
io:format("~p: party remove member ~b~n", [GID, PartyPos]),
{ok, DestUser} = egs_users:read(GID),
{ok, RemovedGID} = psu_party:get_member(DestUser#users.partypid, PartyPos),
@@ -751,13 +747,13 @@ event({party_remove_member, PartyPos}, Client=#client{gid=GID}) ->
egs_proto:send_0204(RemovedUser, Client),
egs_proto:send_0215(0, Client);
-event({player_options_change, Options}, #client{gid=GID, slot=Slot}) ->
+event({player_options_change, Options}, #egs_net{gid=GID, slot=Slot}) ->
Folder = egs_accounts:get_folder(GID),
file:write_file(io_lib:format("save/~s/~b-character.options", [Folder, Slot]), Options);
%% @todo If the player has a scape, use it! Otherwise red screen.
%% @todo Right now we force revive with a dummy HP value.
-event(player_death, Client=#client{gid=GID}) ->
+event(player_death, Client=#egs_net{gid=GID}) ->
% @todo send_0115(GID, 16#ffffffff, LV=1, EXP=idk, Money=1000), % apparently sent everytime you die...
%% use scape:
NewHP = 10,
@@ -770,7 +766,7 @@ event(player_death, Client=#client{gid=GID}) ->
%~ egs_proto:send_0111(User2, 3, 1, Client);
%% @todo Refill the player's HP to maximum, remove SEs etc.
-event(player_death_return_to_lobby, Client=#client{gid=GID}) ->
+event(player_death_return_to_lobby, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
PrevArea = User#users.prev_area,
event({area_change, element(1, PrevArea), element(2, PrevArea), element(3, PrevArea), User#users.prev_entryid}, Client);
@@ -778,7 +774,7 @@ event(player_death_return_to_lobby, Client=#client{gid=GID}) ->
event(player_type_availability_request, Client) ->
egs_proto:send_1a07(Client);
-event(player_type_capabilities_request, Client) ->
+event(character_type_capabilities_request, Client) ->
egs_proto:send_0113(Client);
event(ppcube_request, Client) ->
@@ -790,7 +786,7 @@ event(unicube_request, Client) ->
%% @todo When selecting 'Your room', don't load a default room that's not yours.
event({unicube_select, cancel, _EntryID}, _Client) ->
ignore;
-event({unicube_select, Selection, EntryID}, Client=#client{gid=GID}) ->
+event({unicube_select, Selection, EntryID}, Client=#egs_net{gid=GID}) ->
{ok, User} = egs_users:read(GID),
case Selection of
16#ffffffff ->
@@ -825,7 +821,7 @@ events(Events, Client) ->
%% @doc Load and send the character information to the client.
char_load(User, Client) ->
- egs_proto:send_0d01(User, Client),
+ egs_net:account_character(User#users.char, Client),
%% 0246
egs_proto:send_0a0a(User#users.inventory, Client),
egs_proto:send_1006(5, 0, Client), %% @todo The 0 here is PartyPos, save it in User.
@@ -833,7 +829,7 @@ char_load(User, Client) ->
egs_proto:send_1006(12, Client),
egs_proto:send_0210(Client),
egs_proto:send_0222(User#users.uni, Client),
- egs_proto:send_1500(User, Client),
+ egs_net:comm_own_card(User#users.char, Client),
egs_proto:send_1501(Client),
egs_proto:send_1512(Client),
%% 0303
diff --git a/apps/egs/src/egs_game_protocol.erl b/apps/egs/src/egs_game_protocol.erl
index 9e4c0aa..8279645 100644
--- a/apps/egs/src/egs_game_protocol.erl
+++ b/apps/egs/src/egs_game_protocol.erl
@@ -31,16 +31,14 @@ start_link(_ListenerPid, Socket, Transport, []) ->
%% @todo Handle keepalive messages globally?
init(Socket, Transport) ->
{ok, _TRef} = timer:send_interval(5000, {egs, keepalive}),
- Client = #client{socket=Socket, transport=Transport,
- gid=egs_accounts:tmp_gid()},
- egs_proto:send_0202(Client),
- try
- egs_network:recv(<<>>, egs_login, Client)
- after
- terminate()
- end.
+ Client = egs_net:init(Socket, Transport, egs_login,
+ egs_accounts:tmp_gid()),
+ egs_net:system_hello(Client),
+ catch egs_net:loop(Client),
+ terminate().
-spec terminate() -> ok.
+%% @todo Just use monitors to handle cleanups.
%% @todo Cleanup the instance process if there's nobody in it anymore.
%% @todo Leave party instead of stopping it.
%% @todo Fix the crash when user isn't in egs_users yet.
diff --git a/apps/egs/src/egs_login.erl b/apps/egs/src/egs_login.erl
index 7beb398..1ca0482 100644
--- a/apps/egs/src/egs_login.erl
+++ b/apps/egs/src/egs_login.erl
@@ -18,14 +18,10 @@
%% along with EGS. If not, see .
-module(egs_login).
--export([keepalive/1, info/2, cast/3, raw/3, event/2]).
+-export([info/2, cast/3, event/2]).
-include("include/records.hrl").
-%% @doc Don't keep alive here, authentication should go fast.
-keepalive(_Client) ->
- ok.
-
%% @doc We don't expect any message here.
info(_Msg, _Client) ->
ok.
@@ -34,55 +30,52 @@ info(_Msg, _Client) ->
cast(_Command, _Data, _Client) ->
ok.
-%% Raw commands.
-
-%% @doc Dismiss all raw commands with a log notice.
-%% @todo Have a log event handler instead.
-raw(Command, _Data, _Client) ->
- io:format("~p: dismissed command ~4.16.0b~n", [?MODULE, Command]).
-
%% Events.
%% @doc Reject version < 2.0009.2.
%% @todo Reject wrong platforms too.
-event({system_client_version_info, _Entrance, _Language, _Platform, Version},
- Client=#client{socket=Socket, transport=Transport})
+%% @todo Put the URL in a configuration file.
+event({client_version, _Entrance, _Language, _Platform, Version}, Client)
when Version < 2009002 ->
- egs_proto:send_0231("http://psumods.co.uk/forums/comments.php?DiscussionID=40#Item_1", Client),
- {ok, ErrorMsg} = file:read_file("priv/login/error_version.txt"),
- egs_proto:send_0223(ErrorMsg, Client),
- Transport:close(Socket),
+ egs_net:system_open_url(<<"http://psumods.co.uk/forums/comments.php?DiscussionID=40#Item_1">>, Client),
+ {ok, Error} = file:read_file("priv/login/error_version.txt"),
+ egs_net:system_auth_error(Error, Client),
+ egs_net:terminate(Client),
closed;
-event({system_client_version_info, _Entrance, _Language, _Platform, _Version}, _Client) ->
+event({client_version, _Entrance, _Language, _Platform, _Version}, _Client) ->
ok;
%% @doc Game server info request handler.
-event(system_game_server_request,
- Client=#client{socket=Socket, transport=Transport}) ->
+event(system_game_server_request, Client) ->
{ServerIP, ServerPort} = egs_conf:read(game_server),
- egs_proto:send_0216(ServerIP, ServerPort, Client),
- Transport:close(Socket),
+ egs_net:system_game_server_response(ServerIP, ServerPort, Client),
+ egs_net:terminate(Client),
closed;
%% @doc Authenticate the user by pattern matching its saved state against the key received.
%% If the user is authenticated, send him the character flags list.
-event({system_key_auth_request, AuthGID, AuthKey}, Client) ->
+event({system_key_auth, AuthGID, AuthKey}, Client) ->
egs_accounts:key_auth(AuthGID, AuthKey),
- Client2 = Client#client{gid=AuthGID},
- egs_proto:send_0d05(Client2),
- {ok, egs_char_select, Client2};
+ Client2 = egs_net:set_gid(AuthGID, Client),
+ ValueFlags = egs_conf:read(value_flags),
+ BoolFlags = egs_conf:read(bool_flags),
+ TempFlags = egs_conf:read(temp_flags),
+ egs_net:account_flags(ValueFlags, BoolFlags, TempFlags, Client2),
+ Client3 = egs_net:set_handler(egs_char_select, Client2),
+ Client4 = egs_net:set_keepalive(Client3),
+ {ok, Client4};
%% @doc Authentication request handler. Currently always succeed.
%% @todo Handle real GIDs whenever there's real authentication. GID is the second SessionID in the reply.
%% @todo Apparently it's possible to ask a question in the reply here. Used for free course on JP.
-event({system_login_auth_request, Username, Password}, Client) ->
- {ok, GID} = egs_accounts:login_auth(Username, Password),
- {ok, AuthKey} = egs_accounts:key_auth_init(GID),
+event({system_login_auth, Username, Password}, Client) ->
+ {ok, AuthGID} = egs_accounts:login_auth(Username, Password),
+ {ok, AuthKey} = egs_accounts:key_auth_init(AuthGID),
io:format("auth success for ~s ~s~n", [Username, Password]),
- egs_proto:send_0223(GID, AuthKey, Client);
+ egs_net:system_key_auth_info(AuthGID, AuthKey, Client);
%% @doc MOTD request handler. Page number starts at 0.
%% @todo Currently ignore the language and send the same MOTD file to everyone.
event({system_motd_request, Page, _Language}, Client) ->
{ok, MOTD} = file:read_file("priv/login/motd.txt"),
- egs_proto:send_0225(MOTD, Page, Client).
+ egs_net:system_motd_response(MOTD, Page, Client).
diff --git a/apps/egs/src/egs_login_protocol.erl b/apps/egs/src/egs_login_protocol.erl
index 2a21ccc..7ce4a45 100644
--- a/apps/egs/src/egs_login_protocol.erl
+++ b/apps/egs/src/egs_login_protocol.erl
@@ -29,7 +29,7 @@ start_link(_ListenerPid, Socket, Transport, []) ->
-spec init(ssl:sslsocket(), module()) -> ok | closed.
init(Socket, Transport) ->
- Client = #client{socket=Socket, transport=Transport,
- gid=egs_accounts:tmp_gid()},
- egs_proto:send_0202(Client),
- egs_network:recv(<<>>, egs_login, Client).
+ Client = egs_net:init(Socket, Transport, egs_login,
+ egs_accounts:tmp_gid()),
+ egs_net:system_hello(Client),
+ egs_net:loop(Client).
diff --git a/apps/egs/src/egs_network.erl b/apps/egs/src/egs_network.erl
deleted file mode 100644
index 845a6cc..0000000
--- a/apps/egs/src/egs_network.erl
+++ /dev/null
@@ -1,86 +0,0 @@
-%% @author Loïc Hoguin
-%% @copyright 2010-2011 Loïc Hoguin.
-%% @doc Login and game servers low-level network handling.
-%%
-%% This file is part of EGS.
-%%
-%% EGS is free software: you can redistribute it and/or modify
-%% it under the terms of the GNU Affero 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 Affero General Public License for more details.
-%%
-%% You should have received a copy of the GNU Affero General Public License
-%% along with EGS. If not, see .
-
--module(egs_network).
--export([recv/3]). %% API.
-
--include("include/records.hrl").
-
-%% API.
-
-%% @doc Main loop for the network stack. Receive and handle messages.
-recv(SoFar, CallbackMod, Client=#client{socket=Socket, transport=Transport}) ->
- Transport:setopts(Socket, [{active, once}]),
- {OK, Closed, Error} = Transport:messages(),
- receive
- {OK, _Any, Data} ->
- {Commands, Rest} = split(<< SoFar/bits, Data/bits >>, []),
- case dispatch(Commands, CallbackMod, CallbackMod, Client) of
- {ok, NextCallbackMod, NewClient} ->
- ?MODULE:recv(Rest, NextCallbackMod, NewClient);
- closed -> closed
- end;
- {Closed, _} -> closed;
- {Error, _, _} -> closed;
- {egs, keepalive} ->
- CallbackMod:keepalive(Client),
- ?MODULE:recv(SoFar, CallbackMod, Client);
- Tuple when element(1, Tuple) =:= egs ->
- case CallbackMod:info(Tuple, Client) of
- {ok, NewClient} -> ?MODULE:recv(SoFar, CallbackMod, NewClient);
- _Any -> ?MODULE:recv(SoFar, CallbackMod, Client)
- end
- end.
-
-%% Internal.
-
-%% @doc Dispatch the commands received to the right handler.
-dispatch([], _CallbackMod, NextMod, Client) ->
- {ok, NextMod, Client};
-dispatch([Data|Tail], CallbackMod, NextMod, Client) ->
- Ret = case egs_proto:parse(Data) of
- {command, Command, Channel} ->
- case Channel of
- 1 -> CallbackMod:cast(Command, Data, Client);
- _ -> CallbackMod:raw(Command, Data, Client)
- end;
- ignore ->
- ignore;
- Event ->
- CallbackMod:event(Event, Client)
- end,
- case Ret of
- {ok, NewMod, NewClient} ->
- dispatch(Tail, CallbackMod, NewMod, NewClient);
- {ok, NewClient} ->
- dispatch(Tail, CallbackMod, NextMod, NewClient);
- closed ->
- closed;
- _Any ->
- dispatch(Tail, CallbackMod, NextMod, Client)
- end.
-
-%% @doc Split the network data received into commands.
-split(Data, Acc) when byte_size(Data) < 4 ->
- {lists:reverse(Acc), Data};
-split(<< Size:32/little, _/bits >> = Data, Acc) when Size > byte_size(Data) ->
- {lists:reverse(Acc), Data};
-split(<< Size:32/little, _/bits >> = Data, Acc) ->
- << Split:Size/binary, Rest/bits >> = Data,
- split(Rest, [Split|Acc]).
diff --git a/apps/egs/src/egs_proto.erl b/apps/egs/src/egs_proto.erl
index e1a294a..f95265a 100644
--- a/apps/egs/src/egs_proto.erl
+++ b/apps/egs/src/egs_proto.erl
@@ -30,1163 +30,9 @@
%% @doc Log a detailed message when the assertion A =:= B fails.
-define(ASSERT_EQ(A, B), if A =:= B -> ok; true -> io:format("assert error in module ~p on line ~p~n", [?MODULE, ?LINE]) end).
-%% @spec parse(Packet) -> Result
-%% @doc Parse the packet and return a result accordingly.
-parse(<< Size:32/little, Command:16, Channel:8, _Unknown:8, Data/bits >>) ->
- parse(Size, Command, Channel, Data).
-
-%% @todo Maybe we shouldn't ignore it?
-%% @todo VarI is probably animation state related and defines what the player is doing.
-parse(Size, 16#0102, 2, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, _FromGID:32/little, VarC:32/little, VarD:32/little,
- VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, _TargetGID:32/little, _TargetLID:32/little,
- _VarI:8, _IntDir:24/little, VarJ:32/little, _X:32/little-float, _Y:32/little-float, _Z:32/little-float,
- _QuestID:32/little, _ZoneID:32/little, _MapID:32/little, _EntryID:32/little, VarK:32/little >> = Data,
- ?ASSERT_EQ(Size, 92),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarJ, 0),
- ?ASSERT_EQ(VarK, 0),
- ignore;
-
-%% @todo One of the missing events is probably learning a new PA.
-parse(Size, 16#0105, Channel, Data) ->
- << _LID:16/little, _VarB:16/little, VarC:32/little, _FromGID:32/little, VarD:32/little, VarE:32/little, TypeID:32/little, GID:32/little,
- VarF:32/little, VarG:32/little, TargetGID:32/little, TargetLID:32/little, ItemIndex:8, EventID:8, _PAIndex:8, VarH:8, VarI:32/little, Rest/bits >> = Data,
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(TypeID, 0),
- ?ASSERT_EQ(GID, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- Event = case EventID of
- 1 -> item_equip;
- 2 -> item_unequip;
- 3 -> ignore; %% @todo item_link_pa;
- 4 -> ignore; %% @todo item_unlink_pa;
- 5 -> item_drop;
- 7 -> ignore; %% @todo item_learn_pa;
- 8 -> ignore; %% @todo item_use;
- 9 -> item_set_trap;
- 18 -> ignore; %% @todo item_unlearn_pa;
- _ -> io:format("unknown 0105 EventID ~p~n", [EventID])
- end,
- case Event of
- item_drop ->
- ?ASSERT_EQ(Size, 76),
- << _Quantity:32/little, _PosX:32/little-float, _PosY:32/little-float, _PosZ:32/little-float >> = Rest,
- %~ {Event, ItemIndex, Quantity, ...};
- ignore;
- ignore ->
- ?ASSERT_EQ(Size, 60),
- ignore;
- _ ->
- ?ASSERT_EQ(Size, 60),
- {Event, ItemIndex, TargetGID, TargetLID, VarH, VarI}
- end;
-
-parse(Size, 16#010a, Channel, Data) ->
- << HeaderLID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little,
- _GID:32/little, BodyLID:32/little, EventID:16/little, QuantityOrColor:8, VarK:8, Param:16/bits, VarL:16 >> = Data,
- ?ASSERT_EQ(Size, 60),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(HeaderLID, BodyLID),
- case EventID of
- 1 ->
- << ShopID:16/little >> = Param,
- ?ASSERT_EQ(QuantityOrColor, 0),
- ?ASSERT_EQ(VarK, 0),
- ?ASSERT_EQ(VarL, 0),
- {npc_shop_enter, ShopID};
- 2 ->
- << ShopItemIndex:16/little >> = Param,
- ?ASSERT_EQ(QuantityOrColor, VarK),
- ?ASSERT_EQ(VarL, 0),
- {npc_shop_buy, ShopItemIndex, QuantityOrColor};
- 3 ->
- << InventoryItemIndex:8, _Unknown:8 >> = Param,
- ?ASSERT_EQ(VarK, 0),
- ?ASSERT_EQ(VarL, 0),
- {npc_shop_sell, InventoryItemIndex, QuantityOrColor};
- 4 -> ignore; %% @todo npc_shop_gift_wrap
- 5 ->
- << ShopID:16/little >> = Param,
- ?ASSERT_EQ(QuantityOrColor, 0),
- ?ASSERT_EQ(VarK, 0),
- ?ASSERT_EQ(VarL, 0),
- {npc_shop_leave, ShopID};
- 6 -> ?ASSERT(), ignore
- end;
-
-%% @todo We probably want to check some of those values and save the others. It's mostly harmless though, ignore for now.
-%% @todo We also probably should send the spawn to everyone in response to this command rather than on area_change.
-parse(Size, 16#010b, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, HeaderGID:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little,
- BodyGID:32/little, _PartyPosOrLID:32/little, VarJ:16/little, _IntDir:16/little, _X:32/little-float, _Y:32/little-float, _Z:32/little-float,
- VarK:32/little, VarL:32/little, _QuestID:32/little, _ZoneID:32/little, _MapID:32/little, _EntryID:32/little >> = Data,
- ?ASSERT_EQ(Size, 92),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarJ, 0),
- ?ASSERT_EQ(VarK, 0),
- ?ASSERT_EQ(VarL, 0),
- ?ASSERT_EQ(HeaderGID, BodyGID),
- ignore; %% @todo player_enter_area
-
-parse(Size, 16#0110, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, HeaderGID:32/little, VarC:32/little, VarD:32/little, VarE:32/little,
- VarF:32/little, VarG:32/little, VarH:32/little, BodyGID:32/little, _PartyPosOrLID:32/little, EventID:32/little, Param:32/little >> = Data,
- ?ASSERT_EQ(Size, 60),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(HeaderGID, BodyGID),
- case EventID of
- 1 -> ?ASSERT_EQ(Param, 0), ?ASSERT(), ignore;
- 2 -> ?ASSERT_EQ(Param, 0), player_type_capabilities_request;
- 3 -> ignore; %% @todo {player_type_change, Param};
- 4 -> ?ASSERT_EQ(Param, 0), ignore; %% @todo (related to npc death)
- 6 -> ?ASSERT_EQ(Param, 0), ignore; %% @todo
- 7 -> ?ASSERT_EQ(Param, 0), player_death;
- 8 -> ?ASSERT_EQ(Param, 0), player_death_return_to_lobby;
- 9 -> ?ASSERT_EQ(Param, 10), ignore; %% @todo
- 10 -> ignore; %% @todo {player_online_status_change, Param};
- _ -> io:format("unknown 0110 EventID ~p~n", [EventID])
- end;
-
-parse(Size, 16#020b, Channel, Data) ->
- << LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little,
- VarG:32/little, VarH:32/little, VarI:32/little, Slot:32/little, VarJ:8, BackToPreviousField:8, VarK:16/little >> = Data,
- ?ASSERT_EQ(Size, 52),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(LID, 16#ffff),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarJ, 0),
- ?ASSERT_EQ(VarK, 0),
- AtomBackToPreviousField = if BackToPreviousField =:= 0 -> false; true -> true end,
- {char_select_enter, Slot, AtomBackToPreviousField};
-
-parse(Size, 16#020d, Channel, Data) ->
- << LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little,
- VarG:32/little, VarH:32/little, VarI:32/little, AuthGID:32/little, AuthKey:32/bits, VarJ:32/little, VarK:32/little >> = Data,
- ?ASSERT_EQ(Size, 60),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(LID, 16#ffff),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarJ, 0),
- ?ASSERT_EQ(VarK, 0),
- {system_key_auth_request, AuthGID, AuthKey};
-
-parse(Size, 16#0217, Channel, Data) ->
- << LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little >> = Data,
- ?ASSERT_EQ(Size, 44),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(LID, 16#ffff),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- system_game_server_request;
-
-parse(Size, 16#0219, Channel, Data) ->
- << LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little,
- VarG:32/little, VarH:32/little, VarI:32/little, UsernameBlob:192/bits, PasswordBlob:192/bits, _VarJ:32/little, _VarK:32/little >> = Data,
- ?ASSERT_EQ(Size, 100),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(LID, 16#ffff),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- [Username|_] = re:split(UsernameBlob, "\\0"),
- [Password|_] = re:split(PasswordBlob, "\\0"),
- {system_login_auth_request, Username, Password};
-
-parse(Size, 16#021c, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little >> = Data,
- ?ASSERT_EQ(Size, 44),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- char_load_complete;
-
-parse(Size, 16#021d, Channel, Data) ->
- << _LID:16/little, VarB:16/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little,
- VarG:32/little, VarH:32/little, VarI:32/little, VarJ:32/little, _EntryID:32/little >> = Data,
- ?ASSERT_EQ(Size, 48),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarJ, 0),
- unicube_request;
-
-parse(Size, 16#021f, Channel, Data) ->
- << _LID:16/little, VarB:16/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little,
- VarG:32/little, VarH:32/little, VarI:32/little, VarJ:32/little, UniID:32/little, EntryID:32/little >> = Data,
- ?ASSERT_EQ(Size, 52),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarJ, 0),
- Selection = case UniID of
- 0 -> cancel;
- _ -> UniID
- end,
- {unicube_select, Selection, EntryID};
-
-%% @doc Seems to be exactly the same as 023f, except Channel, and that it's used for JP clients.
-parse(Size, 16#0226, Channel, Data) ->
- << LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little,
- VarG:32/little, VarH:32/little, VarI:32/little, Page:8, Language:8, VarJ:16/little >> = Data,
- ?ASSERT_EQ(Size, 48),
- ?ASSERT_EQ(Channel, 3),
- ?ASSERT_EQ(LID, 16#ffff),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarJ, 0),
- {system_motd_request, Page, language_integer_to_atom(Language)};
-
-%% @doc Whether the MOTD was accepted. Safely ignored.
-parse(Size, 16#0227, Channel, Data) ->
- << LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little,
- VarG:32/little, VarH:32/little, VarI:32/little, _AcceptMOTD:32/little >> = Data,
- ?ASSERT_EQ(Size, 48),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(LID, 16#ffff),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ignore; %% {system_motd_accept, true|false (AcceptMOTD)};
-
-%% @doc Seems to be exactly the same as 0226, except Channel, and that it's used for US clients.
-parse(Size, 16#023f, Channel, Data) ->
- << LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little,
- VarG:32/little, VarH:32/little, VarI:32/little, Page:8, Language:8, VarJ:16/little >> = Data,
- ?ASSERT_EQ(Size, 48),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(LID, 16#ffff),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarJ, 0),
- {system_motd_request, Page, language_integer_to_atom(Language)};
-
-parse(_Size, 16#0304, Channel, Data) ->
- << _LID:16/little, VarB:16/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little,
- VarG:32/little, VarH:32/little, VarI:32/little, VarJ:32/little, FromTypeID:32, FromGID:32/little,
- VarK:32/little, VarL:32/little, ChatType:8, ChatCutIn:8, ChatCutInAngle:8, ChatMsgLength:8,
- ChatChannel:8, ChatCharacterType:8, VarN:8, _VarO:8, FromName:512/bits, ChatMsg/bits >> = Data,
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarJ, 0),
- ?ASSERT_EQ(VarK, 0),
- ?ASSERT_EQ(VarL, 0),
- ?ASSERT_EQ(VarN, 0),
- Modifiers = {chat_modifiers, ChatType, ChatCutIn, ChatCutInAngle, ChatMsgLength, ChatChannel, ChatCharacterType},
- {chat, FromTypeID, FromGID, FromName, Modifiers, ChatMsg};
-
-%% @doc Probably safely ignored. _AreaNb is apparently replied with the same value sent by 0205, the one after EntryID.
-parse(Size, 16#0806, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little,
- VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little, _AreaNb:32/little >> = Data,
- ?ASSERT_EQ(Size, 48),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ignore;
-
-parse(Size, 16#0807, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little,
- QuestID:32/little, ZoneID:16/little, MapID:16/little, EntryID:16/little, _AreaChangeNb:16/little, PartyPos:32/little >> = Data,
- ?ASSERT_EQ(Size, 60),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- {area_change, QuestID, ZoneID, MapID, EntryID, PartyPos};
-
-%% @doc Probably safely ignored. _AreaNb is apparently replied with the same value sent by 0208.
-parse(Size, 16#0808, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little,
- VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little, _AreaNb:32/little >> = Data,
- ?ASSERT_EQ(Size, 48),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ignore;
-
-%% @todo Check that _Rest is full of 0s.
-parse(Size, 16#080c, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little,
- VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little, VarJ:32/little, NPCid:16/little,
- _VarK:16/little, VarL:32/little, VarM:32/little, VarN:16/little, _Var0:16/little, _Rest/bits >> = Data,
- ?ASSERT_EQ(Size, 648),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarJ, 16#ffffffff),
- ?ASSERT_EQ(VarL, 16#ffffffff),
- ?ASSERT_EQ(VarM, 16#ffffffff),
- ?ASSERT_EQ(VarN, 0),
- {npc_force_invite, NPCid};
-
-%% @doc This command should be safely ignored. Probably indicates that a non-mission area change was successful.
-parse(Size, 16#080d, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little >> = Data,
- ?ASSERT_EQ(Size, 44),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ignore;
-
-%% @todo Make sure the Language field is the right one.
-parse(Size, 16#080e, Channel, Data) ->
- << LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little,
- VarJ:8, Language:8, VarK:8, Entrance:8, Platform:8, VarM:24/little, Revision:8, Minor:4, _VarN:12, Major:4, _VarO:4, VarP:32/little, VarQ:32/little, VarR:32/little >> = Data,
- ?ASSERT_EQ(Size, 68),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(LID, 16#ffff),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarJ, 0),
- ?ASSERT_EQ(VarK, 1),
- ?ASSERT_EQ(VarM, 0),
- ?ASSERT_EQ(VarP, 0),
- ?ASSERT_EQ(VarQ, 0),
- ?ASSERT_EQ(VarR, 0),
- AtomPlatform = case Platform of
- 0 -> ps2;
- 1 -> pc;
- _ -> io:format("unknown 080e Platform ~p~n", [Platform]), unknown
- end,
- Version = Major * 1000000 + Minor * 1000 + Revision,
- {system_client_version_info, Entrance, language_integer_to_atom(Language), AtomPlatform, Version};
-
-%% @todo Find out what it's really doing!
-parse(Size, 16#080f, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little,
- VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little, _PartyPos:32/little >> = Data,
- ?ASSERT_EQ(Size, 48),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ignore;
-
-parse(Size, 16#0811, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little,
- _CounterType:8, VarJ:8, FromZoneID:16/little, FromMapID:16/little, FromEntryID:16/little, CounterID:32/little, VarK:32/little >> = Data,
- ?ASSERT_EQ(Size, 60),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarJ, 41),
- ?ASSERT_EQ(VarK, 16#ffffffff),
- {counter_enter, CounterID, FromZoneID, FromMapID, FromEntryID};
-
-parse(Size, 16#0812, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little >> = Data,
- ?ASSERT_EQ(Size, 44),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- counter_leave;
-
-parse(Size, 16#0813, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little,
- VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little, VarJ:32/little, NPCid:32/little >> = Data,
- ?ASSERT_EQ(Size, 52),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarJ, 16#ffffffff),
- {npc_invite, NPCid};
-
-%% @doc This command should be safely ignored. Probably indicates that a mission area change was successful.
-parse(Size, 16#0814, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little >> = Data,
- ?ASSERT_EQ(Size, 44),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ignore;
-
-%% @doc This command should be safely ignored. Probably indicates that a non-mission area change was successful.
-parse(Size, 16#0815, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little >> = Data,
- ?ASSERT_EQ(Size, 44),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ignore;
-
-parse(Size, 16#0818, Channel, Data) ->
- << LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little,
- VarH:32/little, VarI:32/little, BinGPU:512/bits, BinCPU:384/bits, _VarJ:32/little >> = Data,
- ?ASSERT_EQ(Size, 160),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(LID, 16#ffff),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- [_StringCPU|_] = re:split(BinCPU, "\\0", [{return, binary}]),
- [_StringGPU|_] = re:split(BinGPU, "\\0", [{return, binary}]),
- ignore; %% @todo {system_client_hardware_info, StringGPU, StringCPU}; worth logging?
-
-parse(Size, 16#0a10, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little,
- VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little, ItemID:32 >> = Data,
- ?ASSERT_EQ(Size, 48),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- {item_description_request, ItemID};
-
-parse(Size, 16#0b05, _Channel, _Data) ->
- ?ASSERT_EQ(Size, 8),
- ignore;
-
-parse(Size, 16#0c01, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little,
- VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little, QuestID:32/little >> = Data,
- ?ASSERT_EQ(Size, 48),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- {mission_start, QuestID};
-
-parse(Size, 16#0c05, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little,
- VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little, CounterID:32/little >> = Data,
- ?ASSERT_EQ(Size, 48),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- {counter_quest_files_request, CounterID};
-
-%% @doc On official, Price = Rate x 200.
-parse(Size, 16#0c07, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little,
- VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little, _QuestID:32/little, _Rate:32/little >> = Data,
- ?ASSERT_EQ(Size, 52),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- lobby_transport_request;
-
-%% @doc This command should be safely ignored. Probably indicates that a mission area change was successful.
-parse(Size, 16#0c0d, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little >> = Data,
- ?ASSERT_EQ(Size, 44),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ignore;
-
-parse(Size, 16#0c0e, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little >> = Data,
- ?ASSERT_EQ(Size, 44),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- mission_abort;
-
-parse(Size, 16#0c0f, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little,
- VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little, CounterID:32/little >> = Data,
- ?ASSERT_EQ(Size, 48),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- {counter_quest_options_request, CounterID};
-
-%% @todo Return a tuple rather than a binary!
-%% @todo Parse and validate the data here rather than in psu_game.
-parse(Size, 16#0d02, Channel, Data) ->
- << VarA:32/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little,
- VarG:32/little, VarH:32/little, VarI:32/little, Slot:32/little, CharBin/bits >> = Data,
- ?ASSERT_EQ(Size, 324),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- {char_select_create, Slot, CharBin};
-
-parse(Size, 16#0d06, Channel, Data) ->
- << VarA:32/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little >> = Data,
- ?ASSERT_EQ(Size, 44),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- char_select_request;
-
-%% @todo Return a tuple rather than a binary!
-parse(Size, 16#0d07, Channel, Data) ->
- << VarA:32/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little,
- TextDisplaySpeed:8, Sound:8, MusicVolume:8, SoundEffectVolume:8, Vibration:8, RadarMapDisplay:8,
- CutInDisplay:8, MainMenuCursorPosition:8, VarJ:8, Camera3rdY:8, Camera3rdX:8, Camera1stY:8, Camera1stX:8,
- Controller:8, WeaponSwap:8, LockOn:8, Brightness:8, FunctionKeySetting:8, _VarK:8, ButtonDetailDisplay:8, VarL:32/little >> = Data,
- ?ASSERT_EQ(Size, 68),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarJ, 0),
- ?ASSERT_EQ(VarL, 0),
- %% Make sure the options are valid.
- true = TextDisplaySpeed =< 1,
- true = Sound =< 1,
- true = MusicVolume =< 9,
- true = SoundEffectVolume =< 9,
- true = Vibration =< 1,
- true = RadarMapDisplay =< 1,
- true = CutInDisplay =< 1,
- true = MainMenuCursorPosition =< 1,
- true = Camera3rdY =< 1,
- true = Camera3rdX =< 1,
- true = Camera1stY =< 1,
- true = Camera1stX =< 1,
- true = Controller =< 1,
- true = WeaponSwap =< 1,
- true = LockOn =< 1,
- true = Brightness =< 4,
- true = FunctionKeySetting =< 1,
- true = ButtonDetailDisplay =< 2,
- %% Options are considered safe past this point.
- Options = {options, TextDisplaySpeed, Sound, MusicVolume, SoundEffectVolume, Vibration, RadarMapDisplay,
- CutInDisplay, MainMenuCursorPosition, Camera3rdY, Camera3rdX, Camera1stY, Camera1stX,
- Controller, WeaponSwap, LockOn, Brightness, FunctionKeySetting, ButtonDetailDisplay},
- {player_options_change, psu_characters:options_tuple_to_binary(Options)}; %% @todo {player_options_change, Options};
-
-%% @todo Many unknown vars in the command header.
-parse(Size, 16#0e00, Channel, Data) ->
- << _UnknownVars:288/bits, NbHits:32/little, _PartyPosOrLID:32/little, _HitCommandNb:32/little, Hits/bits >> = Data,
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(Size, 56 + NbHits * 80),
- {hits, parse_hits(Hits, [])};
-
-parse(Size, 16#0f0a, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little,
- VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little, BlockID:16/little, ListNb:16/little,
- ObjectNb:16/little, _MapID:16/little, ObjectID:16/little, VarJ:16/little, ObjectTargetID:32/little,
- ObjectType:16/little, VarK:16/little, ObjectBaseTargetID:16/little, VarL:16/little, PartyPosOrLID:32/little,
- VarN:32/little, VarO:32/little, VarP:32/little, VarQ:32/little, VarR:32/little, VarS:32/little,
- VarT:32/little, VarU:32/little, ObjectType2:16/little, EventID:8, VarV:8, VarW:32/little >> = Data,
- ?ASSERT_EQ(Size, 112),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarK, 0),
- ?ASSERT_EQ(VarP, 16#ffffffff),
- ?ASSERT_EQ(VarQ, 16#ffffffff),
- ?ASSERT_EQ(VarR, 16#ffffffff),
- ?ASSERT_EQ(VarS, 0),
- ?ASSERT_EQ(VarT, 0),
- ?ASSERT_EQ(VarU, 0),
- ?ASSERT_EQ(ObjectType, ObjectType2),
- case [ObjectType, EventID] of
- [ 5, 13] ->
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_switch_on, ObjectID};
- [ 5, 14] ->
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_switch_off, ObjectID};
- [ 9, 20] ->
- %% @todo We probably need to handle it for Airboard Rally.
- ignore; %% object_sensor_trigger
- [14, 0] ->
- ?ASSERT_EQ(ObjectID, 16#ffff),
- ?ASSERT_EQ(ObjectTargetID, 16#ffffffff),
- ?ASSERT_EQ(ObjectBaseTargetID, 16#ffff),
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- {object_warp_take, BlockID, ListNb, ObjectNb};
- [22, 12] ->
- ?ASSERT_EQ(VarJ, 134),
- ?ASSERT_EQ(ObjectTargetID, 16#ffffffff),
- ?ASSERT_EQ(ObjectBaseTargetID, 16#ffff),
- ?ASSERT_EQ(VarL, 116),
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_key_console_enable, ObjectID};
- [22, 23] ->
- ?ASSERT_EQ(VarJ, 134),
- ?ASSERT_EQ(ObjectTargetID, 16#ffffffff),
- ?ASSERT_EQ(ObjectBaseTargetID, 16#ffff),
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_key_console_init, ObjectID};
- [22, 24] ->
- ?ASSERT_EQ(VarJ, 134),
- ?ASSERT_EQ(ObjectTargetID, 16#ffffffff),
- ?ASSERT_EQ(ObjectBaseTargetID, 16#ffff),
- ?ASSERT_EQ(VarL, 116),
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_key_console_open_gate, ObjectID};
- [31, 12] ->
- ?ASSERT_EQ(VarJ, 134),
- ?ASSERT_EQ(ObjectTargetID, 16#ffffffff),
- ?ASSERT_EQ(ObjectBaseTargetID, 16#ffff),
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_key_enable, ObjectID};
- [48, 4] ->
- ?ASSERT_EQ(VarJ, 134),
- ?ASSERT_EQ(VarL, 116),
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_boss_gate_enter, ObjectID};
- [48, 5] ->
- ?ASSERT_EQ(VarJ, 134),
- ?ASSERT_EQ(VarL, 116),
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_boss_gate_leave, ObjectID};
- [48, 6] ->
- ?ASSERT_EQ(VarJ, 134),
- ?ASSERT_EQ(VarL, 116),
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_boss_gate_activate, ObjectID};
- [48, 7] ->
- ?ASSERT_EQ(VarJ, 134),
- ?ASSERT_EQ(VarL, 116),
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- ?ASSERT(),
- ignore; %% @todo object_boss_gate_???
- [49, 3] ->
- ?ASSERT_EQ(VarJ, 134),
- ?ASSERT_EQ(ObjectTargetID, 16#ffffffff),
- ?ASSERT_EQ(ObjectBaseTargetID, 16#ffff),
- ?ASSERT_EQ(VarL, 116),
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_crystal_activate, ObjectID};
- [50, 9] ->
- %% @todo Make NPC characters be healed too. This would use VarN and VarO as PartyPosOrLID, and VarV would be > 1.
- ?ASSERT_EQ(VarJ, 134),
- ?ASSERT_EQ(ObjectTargetID, 16#ffffffff),
- ?ASSERT_EQ(ObjectBaseTargetID, 16#ffff),
- ?ASSERT_EQ(VarL, 116),
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_healing_pad_tick, [PartyPosOrLID]};
- [51, 1] ->
- ?ASSERT_EQ(VarL, 116),
- ?ASSERT_EQ(ObjectTargetID, VarN),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_goggle_target_activate, ObjectID};
- [56, 25] ->
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_chair_sit, ObjectTargetID};
- [56, 26] ->
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_chair_stand, ObjectTargetID};
- [57, 12] ->
- ?ASSERT_EQ(VarJ, 134),
- ?ASSERT_EQ(ObjectTargetID, 16#ffffffff),
- ?ASSERT_EQ(ObjectBaseTargetID, 16#ffff),
- ?ASSERT_EQ(VarL, 116),
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- {object_vehicle_boost_enable, ObjectID};
- [57, 28] ->
- ?ASSERT_EQ(VarJ, 134),
- ?ASSERT_EQ(ObjectTargetID, 16#ffffffff),
- ?ASSERT_EQ(ObjectBaseTargetID, 16#ffff),
- ?ASSERT_EQ(VarL, 116),
- ?ASSERT_EQ(VarN, 16#ffffffff),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- {object_vehicle_boost_respawn, ObjectID};
- [71, 27] ->
- ?ASSERT_EQ(VarJ, 134),
- ?ASSERT_EQ(ObjectTargetID, VarN),
- ?ASSERT_EQ(VarO, 16#ffffffff),
- ?ASSERT_EQ(VarV, 1),
- ?ASSERT_EQ(VarW, 0),
- ?ASSERT(),
- ignore; %% @todo object_trap(3rd)_???
- _ -> %% Unhandled actions.
- io:format("unknown 0f0a ObjectType ~p EventID ~p~n", [ObjectType, EventID]),
- ignore
- end;
-
-parse(Size, 16#1007, Channel, Data) ->
- << VarA:32/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little,
- VarG:32/little, VarH:32/little, VarI:32/little, PartyPos:32/little, _Name:512/bits >> = Data,
- ?ASSERT_EQ(Size, 112),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- {party_remove_member, PartyPos};
-
-parse(Size, 16#1701, Channel, Data) ->
- << VarA:32/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little,
- VarG:32/little, VarH:32/little, VarI:32/little, VarJ:32/little, VarK:32/little >> = Data,
- ?ASSERT_EQ(Size, 52),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(VarJ, 0),
- ?ASSERT_EQ(VarK, 16#ffffffff),
- counter_join_party_request;
-
-parse(Size, 16#1705, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little >> = Data,
- ?ASSERT_EQ(Size, 44),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- counter_party_info_request;
-
-%% @todo Currently selected quest. Probably need to broadcast it to other players in the party.
-parse(Size, 16#1707, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little,
- VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little, _QuestID:32/little >> = Data,
- ?ASSERT_EQ(Size, 48),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ignore; %% @todo {counter_quest_selection, QuestID}
-
-parse(Size, 16#1709, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little >> = Data,
- ?ASSERT_EQ(Size, 44),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- counter_party_options_request;
-
-parse(Size, 16#170b, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little >> = Data,
- ?ASSERT_EQ(Size, 44),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- counter_background_locations_request;
-
-parse(Size, 16#1710, Channel, Data) ->
- << _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little,
- VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little, CounterID:32/little >> = Data,
- ?ASSERT_EQ(Size, 48),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- {counter_options_request, CounterID};
-
-parse(Size, 16#1a01, Channel, Data) ->
- << HeaderLID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little,
- BodyLID:32/little, ShopID:32/little, EventID:32/little, VarJ:32/little, VarK:32/little >> = Data,
- ?ASSERT_EQ(Size, 64),
- ?ASSERT_EQ(Channel, 2),
- ?ASSERT_EQ(VarA, 0),
- ?ASSERT_EQ(VarB, 0),
- ?ASSERT_EQ(VarC, 0),
- ?ASSERT_EQ(VarD, 0),
- ?ASSERT_EQ(VarE, 0),
- ?ASSERT_EQ(VarF, 0),
- ?ASSERT_EQ(VarG, 0),
- ?ASSERT_EQ(VarH, 0),
- ?ASSERT_EQ(VarI, 0),
- ?ASSERT_EQ(HeaderLID, BodyLID),
- case EventID of
- 0 -> ?ASSERT_EQ(VarJ, 0), {npc_shop_request, ShopID};
- 2 ->
- ?ASSERT_EQ(ShopID, 0),
- ?ASSERT_EQ(VarJ, 0),
- lumilass_options_request;
- 3 ->
- ?ASSERT_EQ(ShopID, 0),
- ?ASSERT_EQ(VarJ, 0),
- ppcube_request;
- 4 -> ?ASSERT_EQ(ShopID, 0), ignore; %% @todo ppcube_recharge_all
- 5 -> ?ASSERT_EQ(ShopID, 0), ignore; %% @todo ppcube_recharge_one
- 6 ->
- ?ASSERT_EQ(ShopID, 0),
- ?ASSERT_EQ(VarJ, 0),
- ?ASSERT(), ignore; %% @todo put_on_outfit
- 7 ->
- ?ASSERT_EQ(ShopID, 0),
- ?ASSERT(), ignore; %% @todo remove_outfit
- 9 ->
- ?ASSERT_EQ(ShopID, 0),
- ?ASSERT_EQ(VarJ, 0),
- ?ASSERT_EQ(VarK, 0),
- player_type_availability_request;
- _ -> io:format("unknown 1a01 EventID ~p~n", [EventID])
- end;
-
-%% @doc Unknown command,
-parse(_Size, Command, Channel, _Data) ->
- {command, Command, Channel}.
-
-%% @todo Many unknown vars in the hit values.
-parse_hits(<< >>, Acc) ->
- lists:reverse(Acc);
-parse_hits(Hits, Acc) ->
- << A:224/bits, B:128/bits, _C:128/bits, _D:160/bits, Rest/bits >> = Hits,
- << _PosX1:32/little-float, _PosY1:32/little-float, _PosZ1:32/little-float, FromTargetID:32/little, ToTargetID:32/little, _AEnd1:32, _AEnd2:32 >> = A,
- %~ << Stuff2:32, PosX2:32/little-float, PosY2:32/little-float, PosZ2:32/little-float >> = B, %% player
- %~ << Stuff3:32, PosX3:32/little-float, PosY3:32/little-float, PosZ3:32/little-float >> = C, %% target
- %~ << D1:32, D2:32, D3:32, D4:32, D5:32 >> = D,
- parse_hits(Rest, [{hit, FromTargetID, ToTargetID, A, B}|Acc]).
-
%% @doc Send a shop listing.
%% @todo This packet (and its build_010a_list function) hasn't been reviewed at all yet.
-send_010a(ItemsList, Client=#client{gid=DestGID}) ->
+send_010a(ItemsList, Client=#egs_net{gid=DestGID}) ->
NbItems = length(ItemsList),
ItemsBin = build_010a_list(ItemsList, []),
packet_send(Client, << 16#010a0300:32, 0:64, DestGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64,
@@ -1213,7 +59,7 @@ build_010a_list([ItemID|Tail], Acc) ->
%% @doc Send character appearance and other information.
%% @todo Probably don't pattern match the data like this...
-send_010d(CharUser, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_010d(CharUser, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
CharGID = CharUser#users.gid,
CharLID = CharUser#users.lid,
<< _:640, CharBin/bits >> = psu_characters:character_user_to_binary(CharUser),
@@ -1224,33 +70,33 @@ send_010d(CharUser, Client=#client{gid=DestGID, lid=DestLID}) ->
%% @doc Trigger a character-related event.
send_0111(CharUser, EventID, Client) ->
send_0111(CharUser, EventID, 0, Client).
-send_0111(#users{gid=CharGID, lid=CharLID}, EventID, Param, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0111(#users{gid=CharGID, lid=CharLID}, EventID, Param, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
packet_send(Client, << 16#01110300:32, DestLID:16/little, 0:48, CharGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64,
CharGID:32/little, CharLID:32/little, EventID:32/little, Param:32/little >>).
%% @todo Types capability list.
%% @todo This packet hasn't been reviewed at all yet.
-send_0113(Client=#client{gid=DestGID}) ->
+send_0113(Client=#egs_net{gid=DestGID}) ->
{ok, File} = file:read_file("p/typesinfo.bin"),
packet_send(Client, << 16#01130300:32, 0:64, DestGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64, DestGID:32/little, File/binary >>).
%% @doc Update the character level, blastbar, luck and money information.
send_0115(User, Client) ->
send_0115(User, 16#ffffffff, Client).
-send_0115(User=#users{gid=CharGID, lid=CharLID}, EnemyTargetID, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0115(User=#users{gid=CharGID, lid=CharLID}, EnemyTargetID, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
packet_send(Client, << 16#01150300:32, DestLID:16/little, 0:48, CharGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64,
CharGID:32/little, CharLID:32/little, EnemyTargetID:32/little, (build_char_level(User))/binary >>).
%% @doc Revive player with optional SEs.
%% @todo SEs.
-send_0117(#users{gid=CharGID, lid=CharLID, currenthp=HP}, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0117(#users{gid=CharGID, lid=CharLID, currenthp=HP}, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
SE = << 0:64 >>,
packet_send(Client, << 16#01170300:32, DestLID:16/little, 0:48, CharGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64,
CharGID:32/little, CharLID:32/little, SE/binary, HP:32/little, 0:32 >>).
%% @doc Send the zone initialization command.
%% @todo Handle NbPlayers properly. There's more than 1 player!
-send_0200(ZoneID, ZoneType, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0200(ZoneID, ZoneType, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
Var = case ZoneType of
mission -> << 16#06000500:32, 16#01000000:32, 0:64, 16#00040000:32, 16#00010000:32, 16#00140000:32 >>;
myroom -> << 16#06000000:32, 16#02000000:32, 0:64, 16#40000000:32, 16#00010000:32, 16#00010000:32 >>;
@@ -1260,7 +106,7 @@ send_0200(ZoneID, ZoneType, Client=#client{gid=DestGID, lid=DestLID}) ->
DestLID:16/little, ZoneID:16/little, 1:32/little, 16#ffffffff:32, Var/binary, 16#ffffffff:32, 16#ffffffff:32 >>).
%% @doc Send character location, appearance and other information.
-send_0201(CharUser, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0201(CharUser, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
[CharTypeID, GameVersion] = case CharUser#users.type of
npc -> [16#00001d00, 255];
_ -> [16#00001200, 0]
@@ -1272,19 +118,14 @@ send_0201(CharUser, Client=#client{gid=DestGID, lid=DestLID}) ->
packet_send(Client, << 16#02010300:32, DestLID:16/little, 0:16, CharTypeID:32, CharGID:32/little,
0:64, 16#00011300:32, DestGID:32/little, 0:64, CharBin/binary, IsGM:8, 0:8, OnlineStatus:8, GameVersion:8, 0:608 >>).
-%% @doc Hello command. Sent when a client connects to the game or login server.
-%% @todo Can contain an error message if 0:1024 is setup similar to this: 0:32, 3:32/little, 0:48, Len:16/little, Error/binary, 0:Padding.
-send_0202(Client=#client{gid=DestGID, lid=DestLID}) ->
- packet_send(Client, << 16#020203bf:32, DestLID:16/little, 0:272, DestGID:32/little, 0:1024 >>).
-
%% @doc Spawn a player with the given GID and LID.
-send_0203(#users{gid=CharGID, lid=CharLID}, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0203(#users{gid=CharGID, lid=CharLID}, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
packet_send(Client, << 16#02030300:32, DestLID:16/little, 0:144, 16#00011300:32,
DestGID:32/little, 0:64, CharGID:32/little, CharLID:32/little >>).
%% @doc Unspawn the given character.
%% @todo The last 4 bytes are probably the number of players remaining in the zone.
-send_0204(User, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0204(User, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
CharTypeID = case User#users.type of
npc -> 16#00001d00;
_ -> 16#00001200
@@ -1294,13 +135,13 @@ send_0204(User, Client=#client{gid=DestGID, lid=DestLID}) ->
16#00011300:32, DestGID:32/little, 0:64, CharGID:32/little, CharLID:32/little, 100:32/little >>).
%% @doc Make the client load a new map.
-send_0205(CharUser, IsSeasonal, Client=#client{gid=DestGID, lid=DestLID, areanb=AreaNb}) ->
+send_0205(CharUser, IsSeasonal, Client=#egs_net{gid=DestGID, lid=DestLID, areanb=AreaNb}) ->
#users{lid=CharLID, area={_QuestID, ZoneID, MapID}, entryid=EntryID} = CharUser,
packet_send(Client, << 16#02050300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64,
16#ffffffff:32, ZoneID:32/little, MapID:32/little, EntryID:32/little, AreaNb:32/little, CharLID:16/little, 0:8, IsSeasonal:8 >>).
%% @doc Indicate to the client that loading should finish.
-send_0208(Client=#client{gid=DestGID, lid=DestLID, areanb=AreaNb}) ->
+send_0208(Client=#egs_net{gid=DestGID, lid=DestLID, areanb=AreaNb}) ->
packet_send(Client, << 16#02080300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64, AreaNb:32/little >>).
%% @todo No idea what this one does. For unknown reasons it uses channel 2.
@@ -1320,21 +161,17 @@ send_020f(ZoneData, SetID, SeasonID, Client) ->
packet_send(Client, << 16#020f0300:32, 16#ffff:16, 0:272, SetID, SeasonID, 0:16, Size:32/little, ZoneData/binary >>).
%% @doc Send the current UNIX time.
-send_0210(Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0210(Client=#egs_net{gid=DestGID, lid=DestLID}) ->
{M, S, _} = erlang:now(),
UnixTime = M * 1000000 + S,
packet_send(Client, << 16#02100300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:96, UnixTime:32/little >>).
%% @todo No idea what this is doing.
-send_0215(UnknownValue, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0215(UnknownValue, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
packet_send(Client, << 16#02150300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64, UnknownValue:32/little >>).
-%% @doc Send the game server's IP and port that the client requested.
-send_0216(IP, Port, Client=#client{gid=DestGID, lid=DestLID}) ->
- packet_send(Client, << 16#02160300:32, DestLID:16/little, 0:144, 16#00000f00:32, DestGID:32/little, 0:64, IP/binary, Port:16/little, 0:16 >>).
-
%% @doc End of character loading.
-send_021b(Client=#client{gid=DestGID, lid=DestLID}) ->
+send_021b(Client=#egs_net{gid=DestGID, lid=DestLID}) ->
packet_send(Client, << 16#021b0300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64 >>).
%% @doc Send the list of available universes.
@@ -1357,35 +194,20 @@ build_021e_uni([{UniID, {universe, Name, NbPlayers, _MaxPlayers}}|Tail], Acc) ->
build_021e_uni(Tail, [Bin|Acc]).
%% @doc Send the current universe info along with the current level cap.
-send_0222(UniID, Client=#client{gid=DestGID}) ->
+send_0222(UniID, Client=#egs_net{gid=DestGID}) ->
{_Type, Name, NbPlayers, MaxPlayers} = egs_universes:read(UniID),
Padding = 8 * (44 - byte_size(Name)),
LevelCap = egs_conf:read(level_cap),
packet_send(Client, << 16#02220300:32, 16#ffff:16, 0:16, 16#00001200:32, DestGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64,
UniID:32/little, NbPlayers:16/little, MaxPlayers:16/little, Name/binary, 0:Padding, LevelCap:32/little >>).
-%% @doc Send the auth key, or, in case of failure, a related error message.
-send_0223(AuthGID, AuthKey, Client=#client{gid=DestGID}) ->
- packet_send(Client, << 16#02230300:32, 0:160, 16#00000f00:32, DestGID:32/little, 0:64, AuthGID:32/little, AuthKey:32/bits >>).
-send_0223(ErrorMsg, Client=#client{gid=DestGID}) ->
- Length = byte_size(ErrorMsg) div 2 + 2,
- packet_send(Client, << 16#02230300:32, 0:160, 16#00000f00:32, DestGID:32/little, 0:128, 3:32/little, 0:48, Length:16/little, ErrorMsg/binary, 0:16 >>).
-
-%% @doc Send a MOTD page.
-send_0225(MOTD, CurrentPage, Client=#client{lid=DestLID}) ->
- Tokens = re:split(MOTD, "\n."),
- Msg = << << Line/binary, "\n", 0 >> || Line <- lists:sublist(Tokens, 1 + CurrentPage * 15, 15) >>,
- NbPages = 1 + length(Tokens) div 15,
- Length = byte_size(Msg) div 2 + 2,
- packet_send(Client, << 16#02250300:32, DestLID:16/little, 0:272, NbPages:8, CurrentPage:8, Length:16/little, Msg/binary, 0:16 >>).
-
%% @doc Display a notice on the player's screen.
%% There are four types of notices: dialog, top, scroll and timeout.
%% * dialog: A dialog in the center of the screen, which can be OK'd by players.
%% * top: Horizontal scroll on top of the screen, traditionally used for server-wide messages.
%% * scroll: Vertical scroll on the right of the screen, traditionally used for rare missions obtention messages.
%% * timeout: A dialog in the center of the screen that disappears after Duration seconds.
-send_0228(Type, Duration, Message, Client=#client{gid=DestGID}) ->
+send_0228(Type, Duration, Message, Client=#egs_net{gid=DestGID}) ->
TypeInt = case Type of dialog -> 0; top -> 1; scroll -> 2; timeout -> 3 end,
UCS2Message = << << X:8, 0:8 >> || X <- Message >>,
packet_send(Client, << 16#02280300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64,
@@ -1393,23 +215,15 @@ send_0228(Type, Duration, Message, Client=#client{gid=DestGID}) ->
%% @todo No idea!
%% @todo This packet hasn't been reviewed at all yet.
-send_022c(A, B, Client=#client{gid=DestGID}) ->
+send_022c(A, B, Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#022c0300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64, A:16/little, B:16/little >>).
%% @todo Not sure. Sent when going to or from room. Possibly when changing universes too?
-send_0230(Client=#client{gid=DestGID}) ->
+send_0230(Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#02300300:32, 16#ffff:16, 0:16, 16#00011300:32, DestGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64 >>).
-%% @doc Forward the player to a website. The website will open when the player closes the game. Used for login issues mostly.
-send_0231(URL, Client=#client{gid=DestGID, lid=DestLID}) ->
- URLBin = list_to_binary(URL),
- Length = byte_size(URLBin) + 1,
- Padding = 8 * (512 - Length - 1),
- packet_send(Client, << 16#02310300:32, DestLID:16/little, 0:16, 16#00000f00:32, DestGID:32/little, 0:64,
- 16#00000f00:32, DestGID:32/little, 0:64, Length:32/little, URLBin/binary, 0:Padding >>).
-
%% @doc Send the list of players already spawned in the zone when entering it.
-send_0233(Users, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0233(Users, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
NbUsers = length(Users),
Bin = build_0233_users(Users, []),
packet_send(Client, << 16#02330300:32, DestLID:16/little, 0:16, 16#00001200:32, DestGID:32/little, 0:64,
@@ -1422,11 +236,11 @@ build_0233_users([User|Tail], Acc) ->
build_0233_users(Tail, [<< Bin/binary, 0:32 >>|Acc]).
%% @doc Start the zone handling: load the zone file and the objects sent separately.
-send_0236(Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0236(Client=#egs_net{gid=DestGID, lid=DestLID}) ->
packet_send(Client, << 16#02360300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64 >>).
%% @doc Chat message.
-send_0304(FromGID, ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0304(FromGID, ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
{chat_modifiers, ChatType, ChatCutIn, ChatCutInAngle, ChatMsgLength, ChatChannel, ChatCharacterType} = ChatModifiers,
packet_send(Client, << 16#03040300:32, DestLID:16/little, 0:16, 16#00011300:32, FromGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64,
ChatTypeID:32, ChatGID:32/little, 0:64, ChatType:8, ChatCutIn:8, ChatCutInAngle:8, ChatMsgLength:8,
@@ -1435,7 +249,7 @@ send_0304(FromGID, ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage, Cl
%% @todo Force send a new player location. Used for warps.
%% @todo The value before IntDir seems to be the player's current animation. 01 stand up, 08 ?, 17 normal sit
%% @todo This packet hasn't been reviewed at all yet.
-send_0503({PrevX, PrevY, PrevZ, _AnyDir}, Client=#client{gid=DestGID}) ->
+send_0503({PrevX, PrevY, PrevZ, _AnyDir}, Client=#egs_net{gid=DestGID}) ->
{ok, User} = egs_users:read(DestGID),
#users{pos={X, Y, Z, Dir}, area={QuestID, ZoneID, MapID}, entryid=EntryID} = User,
IntDir = trunc(Dir * 182.0416),
@@ -1445,16 +259,16 @@ send_0503({PrevX, PrevY, PrevZ, _AnyDir}, Client=#client{gid=DestGID}) ->
%% @todo NPC inventory. Guessing it's only for NPC characters...
%% @todo This packet hasn't been reviewed at all yet.
-send_0a04(NPCGID, Client=#client{gid=DestGID}) ->
+send_0a04(NPCGID, Client=#egs_net{gid=DestGID}) ->
{ok, Bin} = file:read_file("p/packet0a04.bin"),
packet_send(Client, << 16#0a040300:32, 0:32, 16#00001d00:32, NPCGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64, Bin/binary >>).
%% @todo Inventory related. Doesn't seem to do anything.
-send_0a05(Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0a05(Client=#egs_net{gid=DestGID, lid=DestLID}) ->
packet_send(Client, << 16#0a050300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64 >>).
%% @doc Send the list of ItemUUID for the items in the inventory.
-send_0a06(CharUser, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0a06(CharUser, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
Len = length(CharUser#users.inventory),
UUIDs = lists:seq(1, Len),
Bin = iolist_to_binary([ << N:32/little >> || N <- UUIDs]),
@@ -1464,7 +278,7 @@ send_0a06(CharUser, Client=#client{gid=DestGID, lid=DestLID}) ->
%% @todo Handle more than just goggles.
%% @todo This packet hasn't been reviewed at all yet.
-send_0a0a(Inventory, Client=#client{gid=DestGID}) ->
+send_0a0a(Inventory, Client=#egs_net{gid=DestGID}) ->
{ok, << _:68608/bits, Rest/bits >>} = file:read_file("p/packet0a0a.bin"),
NbItems = length(Inventory),
ItemVariables = build_0a0a_item_variables(Inventory, 1, []),
@@ -1494,14 +308,14 @@ build_0a0a_item_constants([{ItemID, _Variables}|Tail], Acc) ->
build_0a0a_item_constants(Tail, [Bin|Acc]).
%% @doc Send an item's description.
-send_0a11(ItemID, ItemDesc, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0a11(ItemID, ItemDesc, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
Length = 1 + byte_size(ItemDesc) div 2,
packet_send(Client, << 16#0a110300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64,
ItemID:32, Length:32/little, ItemDesc/binary, 0:16 >>).
%% @doc Quest init.
%% @todo When first entering a zone it seems LID should be set to ffff apparently.
-send_0c00(CharUser, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0c00(CharUser, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
#users{area={QuestID, _ZoneID, _MapID}} = CharUser,
packet_send(Client, << 16#0c000300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64, QuestID:32/little,
16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32,
@@ -1511,7 +325,7 @@ send_0c00(CharUser, Client=#client{gid=DestGID, lid=DestLID}) ->
%% @todo Figure out last 4 bytes!
%% @todo This packet hasn't been reviewed at all yet.
-send_0c02(Client=#client{gid=DestGID}) ->
+send_0c02(Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#0c020300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64, 0:32 >>).
%% @doc Send the huge pack of quest files available in the counter.
@@ -1519,58 +333,25 @@ send_0c06(Pack, Client) ->
packet_send(Client, << 16#0c060300:32, 0:288, 1:32/little, Pack/binary >>).
%% @doc Reply that the player is allowed to use the lobby transport. Always allow.
-send_0c08(Client=#client{gid=DestGID}) ->
+send_0c08(Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#0c080300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:96 >>).
%% @doc Send the trial start notification.
%% @todo This packet hasn't been reviewed at all yet.
-send_0c09(Client=#client{gid=DestGID}) ->
+send_0c09(Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#0c090300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64, 0:64 >>).
%% @doc Send the counter's mission options (0 = invisible, 2 = disabled, 3 = available).
-send_0c10(Options, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_0c10(Options, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
Size = byte_size(Options),
packet_send(Client, << 16#0c100300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64, 1, 0, Size:16/little, Options/binary >>).
-%% @doc Send the general data and flags for the selected character.
-%% @todo Handle bitflags and value flags properly.
-send_0d01(User, Client=#client{gid=DestGID}) ->
- CharBin = psu_characters:character_tuple_to_binary(User),
- OptionsBin = psu_characters:options_tuple_to_binary(User#users.options),
- packet_send(Client, << 16#0d010300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, CharBin/binary,
- 16#ffbbef1c:32, 16#f8ff0700:32, 16#fc810916:32, 16#7802134c:32, 16#b0c0040f:32, 16#7cf0e583:32,
- 16#b7bce0c6:32, 16#7ff8f963:32, 16#3fd7ffff:32, 16#fff7ffff:32, 16#f3ff63e0:32, 16#1fe00000:32,
- 0:7744, OptionsBin/binary >>).
-
-%% @doc Send the character list for selection.
-%% @todo There's a few odd values blanked, also the last known location apparently.
-%% @todo This packet hasn't been reviewed at all yet.
-send_0d03(Data0, Data1, Data2, Data3, Client=#client{gid=DestGID}) ->
- [{status, Status0}, {char, Char0}|_] = Data0,
- [{status, Status1}, {char, Char1}|_] = Data1,
- [{status, Status2}, {char, Char2}|_] = Data2,
- [{status, Status3}, {char, Char3}|_] = Data3,
- packet_send(Client, << 16#0d030300:32, 0:32, 16#00011300:32, DestGID:32/little, 0:64,
- 16#00011300:32, DestGID:32/little, 0:104,
- Status0:8, 0:48, Char0/binary, 0:520,
- Status1:8, 0:48, Char1/binary, 0:520,
- Status2:8, 0:48, Char2/binary, 0:520,
- Status3:8, 0:48, Char3/binary, 0:512 >>).
-
-%% @doc Send the flags list. This is the whole list of available values, not the character's.
-%% Sent without fragmentation on official for unknown reasons. Do the same here.
-send_0d05(#client{socket=Socket, transport=Transport, gid=DestGID}) ->
- {ok, Flags} = file:read_file("p/flags.bin"),
- Packet = << 16#0d050300:32, 0:32, 16#00011300:32, DestGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64, Flags/binary >>,
- Size = 4 + byte_size(Packet),
- Transport:send(Socket, << Size:32/little, Packet/binary >>).
-
%% @todo Add a character (NPC or real) to the party members on the right of the screen.
%% @todo NPCid is 65535 for normal characters.
%% @todo Apparently the 4 location ids are set to 0 when inviting an NPC in the lobby - NPCs have their location set to 0 when in lobby; also odd value before PartyPos related to missions
%% @todo Not sure about LID. But seems like it.
%% @todo This packet hasn't been reviewed at all yet.
-send_1004(Type, User, PartyPos, Client=#client{gid=DestGID}) ->
+send_1004(Type, User, PartyPos, Client=#egs_net{gid=DestGID}) ->
[TypeID, LID, SomeFlag] = case Type of
npc_mission -> [16#00001d00, PartyPos, 2];
npc_invite -> [0, 16#ffffffff, 3];
@@ -1598,7 +379,7 @@ send_1004(Type, User, PartyPos, Client=#client{gid=DestGID}) ->
%% @doc Send the client's own player's party information, on the bottom left of the screen.
%% @todo Location and the 20 bytes following sometimes have values, not sure why; when joining a party maybe?
-send_1005(User, Client=#client{gid=DestGID}) ->
+send_1005(User, Client=#egs_net{gid=DestGID}) ->
#users{name=Name, level=Level, currenthp=CurrentHP, maxhp=MaxHP} = User,
Location = << 0:512 >>,
packet_send(Client, << 16#10050300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64,
@@ -1616,7 +397,7 @@ send_1005(User, Client=#client{gid=DestGID}) ->
%% @doc Party-related events.
send_1006(EventID, Client) ->
send_1006(EventID, 0, Client).
-send_1006(EventID, PartyPos, Client=#client{gid=DestGID}) ->
+send_1006(EventID, PartyPos, Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#10060300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, EventID:8, PartyPos:8, 0:16 >>).
%% @doc Send the player's current location.
@@ -1624,14 +405,14 @@ send_1006(EventID, PartyPos, Client=#client{gid=DestGID}) ->
%% @todo Receive the AreaName as UCS2 directly to allow for color codes and the like.
%% @todo Handle TargetLID probably (right after the padding).
%% @todo Do counters even have a name?
-send_100e(CounterID, AreaName, Client=#client{gid=DestGID}) ->
+send_100e(CounterID, AreaName, Client=#egs_net{gid=DestGID}) ->
PartyPos = 0,
UCS2Name = << << X:8, 0:8 >> || X <- AreaName >>,
Padding = 8 * (64 - byte_size(UCS2Name)),
CounterType = if CounterID =:= 16#ffffffff -> 2; true -> 1 end,
packet_send(Client, << 16#100e0300:32, 16#ffffffbf:32, 0:128, 16#00011300:32, DestGID:32, 0:64,
1, PartyPos, 0:48, 16#ffffff7f:32, UCS2Name/binary, 0:Padding, 0:32, CounterID:32/little, CounterType:32/little >>).
-send_100e({QuestID, ZoneID, MapID}, EntryID, AreaName, Client=#client{gid=DestGID}) ->
+send_100e({QuestID, ZoneID, MapID}, EntryID, AreaName, Client=#egs_net{gid=DestGID}) ->
PartyPos = 0,
UCS2Name = << << X:8, 0:8 >> || X <- AreaName >>,
Padding = 8 * (64 - byte_size(UCS2Name)),
@@ -1641,70 +422,70 @@ send_100e({QuestID, ZoneID, MapID}, EntryID, AreaName, Client=#client{gid=DestGI
%% @todo No idea. Also the 2 PartyPos in the built packet more often than not match, but sometimes don't? That's probably because one is PartyPos and the other is LID or something.
%% @todo This packet hasn't been reviewed at all yet.
-send_100f(NPCid, PartyPos, Client=#client{gid=DestGID}) ->
+send_100f(NPCid, PartyPos, Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#100f0300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64, NPCid:16/little, 1, PartyPos:8, PartyPos:32/little >>).
%% @doc Send the mission's quest file when starting a new mission.
%% @todo Handle correctly. 0:32 is actually a missing value. Value before that is unknown too.
%% @todo This packet hasn't been reviewed at all yet.
-send_1015(QuestID, Client=#client{gid=DestGID}) ->
+send_1015(QuestID, Client=#egs_net{gid=DestGID}) ->
QuestData = egs_quests_db:quest_nbl(QuestID),
Size = byte_size(QuestData),
packet_send(Client, << 16#10150300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64, QuestID:32/little, 16#01010000:32, 0:32, Size:32/little, QuestData/binary >>).
%% @todo No idea.
%% @todo This packet hasn't been reviewed at all yet.
-send_1016(PartyPos, Client=#client{gid=DestGID}) ->
+send_1016(PartyPos, Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#10160300:32, 16#ffff0000:32, 0:128, 16#00011300:32, DestGID:32/little, 0:64, PartyPos:32/little >>).
%% @todo No idea.
%% @todo This packet hasn't been reviewed at all yet.
-send_101a(NPCid, PartyPos, Client=#client{gid=DestGID}) ->
+send_101a(NPCid, PartyPos, Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#101a0300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64, NPCid:16/little, PartyPos:16/little, 16#ffffffff:32 >>).
%% @doc Mission start related.
-send_1020(Client=#client{gid=DestGID}) ->
+send_1020(Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#10200300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64 >>).
%% @doc Update HP in the party members information on the left.
%% @todo Handle PartyPos. Probably only pass HP later.
-send_1022(#users{currenthp=HP}, Client=#client{gid=DestGID}) ->
+send_1022(#users{currenthp=HP}, Client=#egs_net{gid=DestGID}) ->
PartyPos = 0,
packet_send(Client, << 16#10220300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, HP:32/little, PartyPos:32/little >>).
%% @todo Boss related command.
%% @todo This packet hasn't been reviewed at all yet.
-send_110e(Data, Client=#client{gid=DestGID}) ->
+send_110e(Data, Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#110e0300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, Data/binary, 0:32, 5:16/little, 12:16/little, 0:32, 260:32/little >>).
%% @todo Boss related command.
%% @todo This packet hasn't been reviewed at all yet.
-send_1113(Data, Client=#client{gid=DestGID}) ->
+send_1113(Data, Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#11130300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, Data/binary >>).
%% @todo Figure out what this packet does. Sane values for counter and missions for now.
%% @todo This packet hasn't been reviewed at all yet.
-send_1202(Client=#client{gid=DestGID}) ->
+send_1202(Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#12020300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, 0:32, 16#10000000:32, 0:64, 16#14000000:32, 0:32 >>).
%% @todo Always the same value, no idea what it's for.
-send_1204(Client=#client{gid=DestGID, lid=DestLID}) ->
+send_1204(Client=#egs_net{gid=DestGID, lid=DestLID}) ->
packet_send(Client, << 16#12040300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:96, 16#20000000:32, 0:256 >>).
%% @doc Object events response?
%% @todo Not sure what Value does exactly. It's either 0 or 1.
%% @todo This packet hasn't been reviewed at all yet.
-send_1205(EventID, BlockID, Value, Client=#client{gid=DestGID}) ->
+send_1205(EventID, BlockID, Value, Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#12050300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, EventID, BlockID, 0:16, Value, 0:24 >>).
%% @todo Figure out what this packet does. Sane values for counter and missions for now.
%% @todo This packet hasn't been reviewed at all yet.
-send_1206(Client=#client{gid=DestGID}) ->
+send_1206(Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#12060300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, 0:32, 16#80020000:32, 0:5120 >>).
%% @todo Figure out what this packet does. Sane values for counter and missions for now.
%% @todo This packet hasn't been reviewed at all yet.
-send_1207(Client=#client{gid=DestGID}) ->
+send_1207(Client=#egs_net{gid=DestGID}) ->
Chunk = << 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 0:224, 16#0000ffff:32, 16#ff000000:32, 16#64000a00:32 >>,
packet_send(Client, << 16#12070300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64,
Chunk/binary, Chunk/binary, Chunk/binary, Chunk/binary, Chunk/binary, Chunk/binary >>).
@@ -1712,65 +493,49 @@ send_1207(Client=#client{gid=DestGID}) ->
%% @todo Object interaction? Figure out. C probably the interaction type.
%% @todo Apparently A would be TargetID/ffffffff, B would be the player LID, C would be the object type? D still completely unknown.
%% @todo This packet hasn't been reviewed at all yet.
-send_1211(A, B, C, D, Client=#client{gid=DestGID}) ->
+send_1211(A, B, C, D, Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#12110300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, A:32/little, B:32/little, C:32/little, D:32/little >>).
%% @doc Make the client load the quest previously sent.
%% @todo This packet hasn't been reviewed at all yet.
-send_1212(Client=#client{gid=DestGID}) ->
+send_1212(Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#12120300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, 0:19200 >>).
%% @todo Not sure. Related to keys.
%% @todo This packet hasn't been reviewed at all yet.
-send_1213(A, B, Client=#client{gid=DestGID}) ->
+send_1213(A, B, Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#12130300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, A:32/little, B:32/little >>).
%% @todo Related to boss gates.
%% @todo This packet hasn't been reviewed at all yet.
-send_1215(A, B, Client=#client{gid=DestGID}) ->
+send_1215(A, B, Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#12150300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, A:32/little, 0:16, B:16/little >>).
%% @todo Not sure yet. Value is probably a TargetID. Used in Airboard Rally. Replying with the same value starts the race.
%% @todo This packet hasn't been reviewed at all yet.
-send_1216(Value, Client=#client{gid=DestGID}) ->
+send_1216(Value, Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#12160300:32, 0:32, 16#00011300:32, DestGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64, Value:32/little >>).
-%% @doc Send the player's partner card.
-%% @todo Handle the LID and comment properly.
-send_1500(User, Client=#client{gid=DestGID}) ->
- #users{slot=Slot, name=Name, race=Race, gender=Gender, class=Class, appearance=Appearance} = User,
- case Appearance of
- #flesh_appearance{voicetype=VoiceType, voicepitch=VoicePitch} -> ok;
- #metal_appearance{voicetype=VoiceType, voicepitch=VoicePitch} -> ok
- end,
- RaceBin = psu_characters:race_atom_to_binary(Race),
- GenderBin = psu_characters:gender_atom_to_binary(Gender),
- ClassBin = psu_characters:class_atom_to_binary(Class),
- Comment = << 0:2816 >>,
- packet_send(Client, << 16#15000300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64,
- Name/binary, RaceBin:8, GenderBin:8, ClassBin:8, VoiceType:8, VoicePitch:8, 0:24,
- DestGID:32/little, 0:224, Comment/binary, 1, 4, 1, Slot, 0:64 >>).
-
%% @todo Send an empty partner card list.
%% @todo This packet hasn't been reviewed at all yet.
-send_1501(Client=#client{gid=DestGID}) ->
+send_1501(Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#15010300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:96 >>).
%% @todo Send an empty blacklist.
%% @todo This packet hasn't been reviewed at all yet.
-send_1512(Client=#client{gid=DestGID}) ->
+send_1512(Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#15120300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:46144 >>).
%% @todo NPC related packet, sent when there's an NPC in the area.
%% @todo This packet hasn't been reviewed at all yet.
-send_1601(PartyPos, Client=#client{gid=DestGID}) ->
+send_1601(PartyPos, Client=#egs_net{gid=DestGID}) ->
{ok, << _:32, Bin/bits >>} = file:read_file("p/packet1601.bin"),
packet_send(Client, << 16#16010300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, PartyPos:32/little, Bin/binary >>).
%% @doc Send the player's NPC and PM information.
%% @todo The value 4 is the card priority. Find what 3 is. When sending, the first 0 is an unknown value.
%% @todo This packet hasn't been reviewed at all yet.
-send_1602(Client=#client{gid=DestGID}) ->
+send_1602(Client=#egs_net{gid=DestGID}) ->
NPCList = egs_npc_db:all(),
NbNPC = length(NPCList),
Bin = iolist_to_binary([<< NPCid:8, 0, 4, 0, 3, 0:24 >> || {NPCid, _Data} <- NPCList]),
@@ -1784,13 +549,13 @@ send_1602(Client=#client{gid=DestGID}) ->
%% @doc Send the list of parties to join.
%% @todo Handle lists of parties.
%% @todo Probably has to handle a LID here, although it should always be 0.
-send_1701(Client=#client{gid=DestGID}) ->
+send_1701(Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#17010300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:96 >>).
%% @doc Party information.
%% @todo Handle existing parties.
%% @todo This packet hasn't been reviewed at all yet.
-send_1706(CharName, Client=#client{gid=DestGID}) ->
+send_1706(CharName, Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#17060300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64,
16#00000300:32, 16#d5c0faff:32, 0:64, CharName/binary,
16#78000000:32, 16#01010000:32, 0:1536, 16#0100c800:32, 16#0601010a:32, 16#ffffffff:32, 0:32 >>).
@@ -1798,26 +563,26 @@ send_1706(CharName, Client=#client{gid=DestGID}) ->
%% @doc Party settings. Item distribution is random for now.
%% @todo Handle correctly.
%% @todo This packet hasn't been reviewed at all yet.
-send_170a(Client=#client{gid=DestGID}) ->
+send_170a(Client=#egs_net{gid=DestGID}) ->
packet_send(Client, << 16#170a0300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, 16#01010c08:32 >>).
%% @todo Find what the heck this packet is.
%% @todo This packet hasn't been reviewed at all yet.
-send_170c(Client=#client{gid=DestGID}) ->
+send_170c(Client=#egs_net{gid=DestGID}) ->
{ok, File} = file:read_file("p/packet170c.bin"),
packet_send(Client, << 16#170c0300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, File/binary >>).
%% @doc Send the background to use for the counter.
-send_1711(Bg, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_1711(Bg, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
packet_send(Client, << 16#17110300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64, Bg:8, 0:24 >>).
%% @doc NPC shop request reply.
-send_1a02(A, B, C, D, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_1a02(A, B, C, D, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
packet_send(Client, << 16#1a020300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:96,
A:16/little, B:16/little, C:16/little, D:16/little >>).
%% @doc Lumilass available hairstyles/headtypes handler.
-send_1a03(User, Client=#client{gid=DestGID, lid=DestLID}) ->
+send_1a03(User, Client=#egs_net{gid=DestGID, lid=DestLID}) ->
{ok, Conf} = file:consult("priv/lumilass.conf"),
NbHeadtypes = proplists:get_value({headtypes, User#users.gender, User#users.race}, Conf, 0),
HairstylesList = proplists:get_value({hairstyles, User#users.gender}, Conf),
@@ -1829,12 +594,12 @@ send_1a03(User, Client=#client{gid=DestGID, lid=DestLID}) ->
%% @doc PP cube handler.
%% @todo The 4 bytes before the file may vary. Everything past that is the same. Figure things out.
%% @todo This packet hasn't been reviewed at all yet.
-send_1a04(Client=#client{gid=DestGID}) ->
+send_1a04(Client=#egs_net{gid=DestGID}) ->
{ok, File} = file:read_file("p/ppcube.bin"),
packet_send(Client, << 16#1a040300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, 0:32, File/binary >>).
%% @doc Available types handler. Enable all 16 types.
-send_1a07(Client=#client{gid=DestGID, lid=DestLID}) ->
+send_1a07(Client=#egs_net{gid=DestGID, lid=DestLID}) ->
packet_send(Client, << 16#1a070300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:160,
16#01010101:32, 16#01010101:32, 16#01010101:32, 16#01010101:32 >>).
@@ -1924,20 +689,6 @@ build_item_variables(ItemID, ItemUUID, #psu_trap_item_variables{quantity=Quantit
%% Utility functions.
-%% @doc Return the language as an atom from its integer value.
-%% @todo Identify which of the english languages is american and which is uk.
-language_integer_to_atom(0) -> japanese;
-language_integer_to_atom(1) -> american_english;
-language_integer_to_atom(2) -> british_english;
-language_integer_to_atom(3) -> french;
-language_integer_to_atom(4) -> german;
-language_integer_to_atom(5) -> spanish;
-language_integer_to_atom(6) -> italian;
-language_integer_to_atom(7) -> korean;
-language_integer_to_atom(8) -> simplified_chinese;
-language_integer_to_atom(9) -> traditional_chinese;
-language_integer_to_atom(Language) -> io:format("unknown 080e Language ~p~n", [Language]).
-
%% @doc Prepare a packet. Return the real size and padding at the end.
packet_prepare(Packet) ->
Size = 4 + byte_size(Packet),
@@ -1953,7 +704,7 @@ packet_send(Client, Packet) ->
packet_send(Client, << Size:32/little, Packet/binary, Padding/binary >>, Size).
%% Send a normal command.
-packet_send(#client{socket=Socket, transport=Transport}, Packet, Size)
+packet_send(#egs_net{socket=Socket, transport=Transport}, Packet, Size)
when Size =< 16#4000 ->
Transport:send(Socket, Packet);
%% Send a fragmented command when size is too big.
@@ -1961,20 +712,15 @@ packet_send(Client, Packet, Size) ->
packet_fragment_send(Client, Packet, Size, 0).
%% Send the last chunk of a fragmented command.
-packet_fragment_send(#client{socket=Socket, transport=Transport}, Packet,
+packet_fragment_send(#egs_net{socket=Socket, transport=Transport}, Packet,
Size, Current) when Size - Current =< 16#4000 ->
FragmentSize = 16#10 + byte_size(Packet),
Fragment = << FragmentSize:32/little, 16#0b030000:32, Size:32/little, Current:32/little, Packet/binary >>,
Transport:send(Socket, Fragment);
%% Send another chunk of a fragmented command.
-packet_fragment_send(Client=#client{socket=Socket, transport=Transport}, Packet,
+packet_fragment_send(Client=#egs_net{socket=Socket, transport=Transport}, Packet,
Size, Current) ->
<< Chunk:131072/bits, Rest/bits >> = Packet,
Fragment = << 16#10400000:32, 16#0b030000:32, Size:32/little, Current:32/little, Chunk/binary >>,
Transport:send(Socket, Fragment),
packet_fragment_send(Client, Rest, Size, Current + 16#4000).
-
-%% @doc Keepalive. Just send an empty packet, the game doesn't really care.
-%% @todo If there's an actual keepalive command, use it instead.
-send_keepalive(Client) ->
- packet_send(Client, << 0:32 >>).
diff --git a/apps/egs/src/egs_script_lexer.erl b/apps/egs/src/egs_script_lexer.erl
index 348488f..9a662d0 100644
--- a/apps/egs/src/egs_script_lexer.erl
+++ b/apps/egs/src/egs_script_lexer.erl
@@ -1,4 +1,4 @@
--file("/usr/lib/erlang/lib/parsetools-2.0.6/include/leexinc.hrl", 0).
+-file("/usr/lib/erlang/lib/parsetools-2.0.7/include/leexinc.hrl", 0).
%% The source of this file is part of leex distribution, as such it
%% has the same Copyright as the other files in the leex
%% distribution. The Copyright is defined in the accompanying file
@@ -477,7 +477,7 @@ escape_char($s) -> $\s; %% \s = SPC
escape_char($d) -> $\d; %% \d = DEL
escape_char(C) -> C.
--file("/usr/lib/erlang/lib/parsetools-2.0.6/include/leexinc.hrl", 14).
+-file("/usr/lib/erlang/lib/parsetools-2.0.7/include/leexinc.hrl", 14).
format_error({illegal,S}) -> ["illegal characters ",io_lib:write_string(S)];
format_error({user,S}) -> S.
@@ -1073,4 +1073,4 @@ yyaction_6(TokenLine) ->
yyaction_7() ->
skip_token .
--file("/usr/lib/erlang/lib/parsetools-2.0.6/include/leexinc.hrl", 282).
+-file("/usr/lib/erlang/lib/parsetools-2.0.7/include/leexinc.hrl", 282).
diff --git a/apps/egs/src/egs_script_parser.erl b/apps/egs/src/egs_script_parser.erl
index 4b33e82..d19528c 100644
--- a/apps/egs/src/egs_script_parser.erl
+++ b/apps/egs/src/egs_script_parser.erl
@@ -4,11 +4,11 @@
unwrap({_,_,V}) -> V.
--file("/usr/lib/erlang/lib/parsetools-2.0.6/include/yeccpre.hrl", 0).
+-file("/usr/lib/erlang/lib/parsetools-2.0.7/include/yeccpre.hrl", 0).
%%
%% %CopyrightBegin%
%%
-%% Copyright Ericsson AB 1996-2010. All Rights Reserved.
+%% Copyright Ericsson AB 1996-2011. All Rights Reserved.
%%
%% The contents of this file are subject to the Erlang Public License,
%% Version 1.1, (the "License"); you may not use this file except in
@@ -35,10 +35,11 @@ parse(Tokens) ->
-spec parse_and_scan({function() | {atom(), atom()}, [_]}
| {atom(), atom(), [_]}) -> yecc_ret().
-parse_and_scan({F, A}) -> % Fun or {M, F}
+parse_and_scan({F, A}) ->
yeccpars0([], {{F, A}, no_line}, 0, [], []);
parse_and_scan({M, F, A}) ->
- yeccpars0([], {{{M, F}, A}, no_line}, 0, [], []).
+ Arity = length(A),
+ yeccpars0([], {{fun M:F/Arity, A}, no_line}, 0, [], []).
-spec format_error(any()) -> [char() | list()].
format_error(Message) ->
@@ -74,7 +75,7 @@ yeccpars0(Tokens, Tzr, State, States, Vstack) ->
Error
end.
-yecc_error_type(function_clause, [{?MODULE,F,ArityOrArgs} | _]) ->
+yecc_error_type(function_clause, [{?MODULE,F,ArityOrArgs,_} | _]) ->
case atom_to_list(F) of
"yeccgoto_" ++ SymbolL ->
{ok,[{atom,_,Symbol}],_} = erl_scan:string(SymbolL),
@@ -187,7 +188,7 @@ yecctoken2string(Other) ->
--file("src/egs_script_parser.erl", 190).
+-file("src/egs_script_parser.erl", 191).
yeccpars2(0=S, Cat, Ss, Stack, T, Ts, Tzr) ->
yeccpars2_0(S, Cat, Ss, Stack, T, Ts, Tzr);
diff --git a/apps/egs/src/psu/psu_appearance.erl b/apps/egs/src/psu/psu_appearance.erl
deleted file mode 100644
index d84fbb2..0000000
--- a/apps/egs/src/psu/psu_appearance.erl
+++ /dev/null
@@ -1,254 +0,0 @@
-%% @author Loïc Hoguin
-%% @copyright 2010-2011 Loïc Hoguin.
-%% @doc Character appearance functions.
-%%
-%% This file is part of EGS.
-%%
-%% EGS is free software: you can redistribute it and/or modify
-%% it under the terms of the GNU Affero 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 Affero General Public License for more details.
-%%
-%% You should have received a copy of the GNU Affero General Public License
-%% along with EGS. If not, see .
-
--module(psu_appearance).
--export([binary_to_tuple/2, tuple_to_binary/2, validate_char_create/3]).
-
--include("include/records.hrl").
-
-%% @doc Convert the binary character creation appearance data into a tuple.
-%% The lineshield color is ignored and set to 0 (neutral) by default instead.
-%% The badge is always set to 0 (none). Only beasts can later change it.
-%% The lips color and intensity is ignored and set to the default values {32767, 32767, 0} (flesh races only).
-binary_to_tuple(cast, Binary) ->
- << VoiceType:8, VoicePitch:8, _:24, Torso:32/unsigned-integer, Legs:32/unsigned-integer, Arms:32/unsigned-integer,
- Ears:32/unsigned-integer, Face:32/unsigned-integer, HeadType:32/unsigned-integer, MainColor:8, _:24,
- _:8, Eyebrows:8, Eyelashes:8, EyesGroup:8, Eyes:8, _:24, EyesColorY:32/little, EyesColorX:32/little,
- _:96, BodyColor:32/little, SubColor:32/little, HairstyleColorY:32/little,
- HairstyleColorX:32/little, Proportion:32/little, ProportionBoxX:32/little,
- ProportionBoxY:32/little, FaceBoxX:32/little, FaceBoxY:32/little >> = Binary,
- {metal_appearance, VoiceType, VoicePitch, Torso, Legs, Arms, Ears, Face, HeadType, MainColor, 0,
- Eyebrows, Eyelashes, EyesGroup, Eyes, EyesColorY, EyesColorX, BodyColor, SubColor, HairstyleColorY, HairstyleColorX,
- Proportion, ProportionBoxX, ProportionBoxY, FaceBoxX, FaceBoxY};
-
-binary_to_tuple(_, Binary) ->
- << VoiceType:8, VoicePitch:8, _:24, Jacket:32/unsigned-integer, Pants:32/unsigned-integer, Shoes:32/unsigned-integer,
- Ears:32/unsigned-integer, Face:32/unsigned-integer, Hairstyle:32/unsigned-integer, JacketColor:8, PantsColor:8, ShoesColor:8, _:8,
- _:8, Eyebrows:8, Eyelashes:8, EyesGroup:8, Eyes:8, BodySuit:8, _:16, EyesColorY:32/little, EyesColorX:32/little,
- _:96, SkinColor:32/little, _:32, HairstyleColorY:32/little,
- HairstyleColorX:32/little, Proportion:32/little, ProportionBoxX:32/little,
- ProportionBoxY:32/little, FaceBoxX:32/little, FaceBoxY:32/little >> = Binary,
- {flesh_appearance, VoiceType, VoicePitch, Jacket, Pants, Shoes, Ears, Face, Hairstyle, JacketColor, PantsColor, ShoesColor,
- 0, 0, Eyebrows, Eyelashes, EyesGroup, Eyes, BodySuit, EyesColorY, EyesColorX, 32767, 32767, 0, SkinColor, HairstyleColorY,
- HairstyleColorX, Proportion, ProportionBoxX, ProportionBoxY, FaceBoxX, FaceBoxY}.
-
-%% @doc Convert a tuple of appearance data into a binary to be sent to clients.
-tuple_to_binary(cast, Tuple) ->
- {metal_appearance, VoiceType, VoicePitch, Torso, Legs, Arms, Ears, Face, HeadType, MainColor, LineshieldColor,
- Eyebrows, Eyelashes, EyesGroup, Eyes, EyesColorY, EyesColorX, BodyColor, SubColor, HairstyleColorY, HairstyleColorX,
- Proportion, ProportionBoxX, ProportionBoxY, FaceBoxX, FaceBoxY} = Tuple,
- << VoiceType:8, VoicePitch:8, 0:24, Torso:32/unsigned-integer, Legs:32/unsigned-integer, Arms:32/unsigned-integer,
- Ears:32/unsigned-integer, Face:32/unsigned-integer, HeadType:32/unsigned-integer, MainColor:8, 0:16, LineshieldColor:8,
- 0:8, Eyebrows:8, Eyelashes:8, EyesGroup:8, Eyes:8, 0:24, EyesColorY:32/little, EyesColorX:32/little,
- 16#ff7f0000:32, 16#ff7f0000:32, 0:32, BodyColor:32/little, SubColor:32/little, HairstyleColorY:32/little,
- HairstyleColorX:32/little, Proportion:32/little, ProportionBoxX:32/little,
- ProportionBoxY:32/little, FaceBoxX:32/little, FaceBoxY:32/little >>;
-
-tuple_to_binary(_, Tuple) ->
- {flesh_appearance, VoiceType, VoicePitch, Jacket, Pants, Shoes, Ears, Face, Hairstyle, JacketColor, PantsColor, ShoesColor,
- LineshieldColor, Badge, Eyebrows, Eyelashes, EyesGroup, Eyes, BodySuit, EyesColorY, EyesColorX, LipsIntensity, LipsColorY, LipsColorX,
- SkinColor, HairstyleColorY, HairstyleColorX, Proportion, ProportionBoxX, ProportionBoxY, FaceBoxX, FaceBoxY} = Tuple,
- << VoiceType:8, VoicePitch:8, 0:24, Jacket:32/unsigned-integer, Pants:32/unsigned-integer, Shoes:32/unsigned-integer,
- Ears:32/unsigned-integer, Face:32/unsigned-integer, Hairstyle:32/unsigned-integer, JacketColor:8, PantsColor:8, ShoesColor:8, LineshieldColor:8,
- Badge:8, Eyebrows:8, Eyelashes:8, EyesGroup:8, Eyes:8, BodySuit:8, 0:16, EyesColorY:32/little, EyesColorX:32/little,
- LipsIntensity:32/little, LipsColorY:32/little, LipsColorX:32/little,
- SkinColor:32/little, 16#ffff0200:32, HairstyleColorY:32/little,
- HairstyleColorX:32/little, Proportion:32/little, ProportionBoxX:32/little,
- ProportionBoxY:32/little, FaceBoxX:32/little, FaceBoxY:32/little >>.
-
-%% @doc Validate the character creation appearance data.
-%% Trigger an exception rather than handling errors.
-validate_char_create(cast, male, Tuple) ->
- #metal_appearance{voicetype=VoiceType, torso=Torso, legs=Legs, arms=Arms, ears=Ears, face=Face, headtype=HeadType, eyelashes=Eyelashes, eyesgroup=EyesGroup, eyescolorx=EyesColorX} = Tuple,
- validate_char_create_common_metal(Tuple),
- true = (VoiceType >= 27 andalso VoiceType =< 38) orelse (VoiceType >= 89 andalso VoiceType =< 96),
- true = Torso =:= 16#00F70100 orelse Torso =:= 16#00F90100 orelse Torso =:= 16#00FC0100,
- true = Legs =:= 16#00F70101 orelse Legs =:= 16#00F90101 orelse Legs =:= 16#00FC0101,
- true = Arms =:= 16#00F70102 orelse Arms =:= 16#00F90102 orelse Arms =:= 16#00FC0102,
- if Face =:= 16#00040004 orelse Face =:= 16#0A040004 orelse Face =:= 16#14040004 orelse Face =:= 16#1E040004 orelse Face =:= 16#28040004 orelse Face =:= 16#000E0004 ->
- true = Ears =:= 16#001E0003 orelse Ears =:= 16#001F0003 orelse Ears =:= 16#00200003 orelse Ears =:= 16#00210003 orelse Ears =:= 16#00220003,
- true = EyesColorX =< 327679,
- validate_char_create_male_hairstyle(HeadType);
- Face =:= 16#00F40104 orelse Face =:= 16#00F50104 orelse Face =:= 16#00F60104 orelse Face =:= 16#00F70104 orelse Face =:= 16#00F80104 orelse Face =:= 16#00F90104 orelse
- Face =:= 16#00FA0104 orelse Face =:= 16#00FD0104 orelse Face =:= 16#00020204 orelse Face =:= 16#00030204 orelse Face =:= 16#00040204 orelse Face =:= 16#00060204 orelse Face =:= 16#00070204 ->
- Ears = 16#FFFFFFFF,
- true = EyesColorX =< 458751,
- true = HeadType =:= 16#00F40105 orelse HeadType =:= 16#00F50105 orelse HeadType =:= 16#00F60105 orelse HeadType =:= 16#00F70105 orelse HeadType =:= 16#00F80105 orelse
- HeadType =:= 16#00F90105 orelse HeadType =:= 16#00FA0105 orelse HeadType =:= 16#00FB0105 orelse HeadType =:= 16#00FD0105 orelse HeadType =:= 16#00020205 orelse
- HeadType =:= 16#00030205 orelse HeadType =:= 16#00040205 orelse HeadType =:= 16#00060205 orelse HeadType =:= 16#00070205
- end,
- true = Eyelashes =< 2,
- EyesGroup = 4;
-
-validate_char_create(cast, female, Tuple) ->
- #metal_appearance{voicetype=VoiceType, torso=Torso, legs=Legs, arms=Arms, ears=Ears, face=Face, headtype=HeadType, eyelashes=Eyelashes, eyesgroup=EyesGroup, eyescolorx=EyesColorX} = Tuple,
- validate_char_create_common_metal(Tuple),
- true = (VoiceType >= 39 andalso VoiceType =< 50) orelse (VoiceType >= 97 andalso VoiceType =< 101),
- true = Torso =:= 16#00F51100 orelse Torso =:= 16#00F91100 orelse Torso =:= 16#00FA1100,
- true = Legs =:= 16#00F51101 orelse Legs =:= 16#00F91101 orelse Legs =:= 16#00FA1101,
- true = Arms =:= 16#00F51102 orelse Arms =:= 16#00F91102 orelse Arms =:= 16#00F61102,
- if Face =:= 16#00041004 orelse Face =:= 16#0A041004 orelse Face =:= 16#14041004 orelse Face =:= 16#1E041004 orelse Face =:= 16#3C041004 ->
- true = Ears =:= 16#001E1003 orelse Ears =:= 16#001F1003 orelse Ears =:= 16#00201003 orelse Ears =:= 16#00211003 orelse Ears =:= 16#00221003,
- true = EyesColorX =< 327679,
- validate_char_create_female_hairstyle(HeadType);
- Face =:= 16#00F41104 orelse Face =:= 16#00F51104 orelse Face =:= 16#00F61104 orelse Face =:= 16#00F71104 orelse Face =:= 16#00F81104 orelse Face =:= 16#00F91104 orelse
- Face =:= 16#00FA1104 orelse Face =:= 16#00FD1104 orelse Face =:= 16#00031204 orelse Face =:= 16#00041204 orelse Face =:= 16#00051204 orelse Face =:= 16#00061204 orelse Face =:= 16#00081204 ->
- Ears = 16#FFFFFFFF,
- true = EyesColorX =< 458751,
- true = HeadType =:= 16#00F41105 orelse HeadType =:= 16#00F51105 orelse HeadType =:= 16#00F61105 orelse HeadType =:= 16#00F71105 orelse HeadType =:= 16#00F81105 orelse
- HeadType =:= 16#00F91105 orelse HeadType =:= 16#00FA1105 orelse HeadType =:= 16#00FB1105 orelse HeadType =:= 16#00FD1105 orelse HeadType =:= 16#00031205 orelse
- HeadType =:= 16#00041205 orelse HeadType =:= 16#00051205 orelse HeadType =:= 16#00061205 orelse HeadType =:= 16#00081205
- end,
- true = Eyelashes =< 12,
- EyesGroup = 5;
-
-validate_char_create(human, male, Tuple) ->
- #flesh_appearance{ears=Ears, face=Face, eyesgroup=EyesGroup, eyes=Eyes} = Tuple,
- validate_char_create_common_flesh(Tuple),
- validate_char_create_common_male_flesh(Tuple),
- true = Ears =:= 16#00000003 orelse Ears =:= 16#00010003,
- true = Face =:= 16#00010004 orelse Face =:= 16#01010004 orelse Face =:= 16#14010004 orelse Face =:= 16#15010004 orelse Face =:= 16#16010004 orelse Face =:= 16#17010004 orelse
- Face =:= 16#18010004 orelse Face =:= 16#19010004 orelse Face =:= 16#1A010004 orelse Face =:= 16#1E010004 orelse Face =:= 16#1F010004 orelse Face =:= 16#20010004 orelse
- Face =:= 16#21010004 orelse Face =:= 16#22010004 orelse Face =:= 16#23010004 orelse Face =:= 16#24010004 orelse Face =:= 16#28010004 orelse Face =:= 16#29010004 orelse
- Face =:= 16#2A010004 orelse Face =:= 16#2B010004 orelse Face =:= 16#2C010004 orelse Face =:= 16#2D010004 orelse Face =:= 16#2E010004 orelse Face =:= 16#000B0004,
- EyesGroup = 0,
- true = Eyes =< 5;
-
-validate_char_create(newman, male, Tuple) ->
- #flesh_appearance{ears=Ears, face=Face, eyesgroup=EyesGroup, eyes=Eyes} = Tuple,
- validate_char_create_common_flesh(Tuple),
- validate_char_create_common_male_flesh(Tuple),
- true = Ears =:= 16#00030003 orelse Ears =:= 16#00650003 orelse Ears =:= 16#00660003,
- true = Face =:= 16#00020004 orelse Face =:= 16#01020004 orelse Face =:= 16#14020004 orelse Face =:= 16#15020004 orelse Face =:= 16#16020004 orelse Face =:= 16#17020004 orelse
- Face =:= 16#18020004 orelse Face =:= 16#19020004 orelse Face =:= 16#1A020004 orelse Face =:= 16#1E020004 orelse Face =:= 16#1F020004 orelse Face =:= 16#20020004 orelse
- Face =:= 16#21020004 orelse Face =:= 16#22020004 orelse Face =:= 16#23020004 orelse Face =:= 16#24020004 orelse Face =:= 16#28020004 orelse Face =:= 16#29020004 orelse
- Face =:= 16#2A020004 orelse Face =:= 16#2B020004 orelse Face =:= 16#2C020004 orelse Face =:= 16#2D020004 orelse Face =:= 16#2E020004 orelse Face =:= 16#000C0004,
- EyesGroup = 2,
- true = Eyes =< 5;
-
-validate_char_create(beast, male, Tuple) ->
- #flesh_appearance{ears=Ears, face=Face, eyesgroup=EyesGroup, eyes=Eyes} = Tuple,
- validate_char_create_common_flesh(Tuple),
- validate_char_create_common_male_flesh(Tuple),
- true = Ears =:= 16#00020003 orelse Ears =:= 16#00CD0003 orelse Ears =:= 16#00CE0003,
- true = Face =:= 16#00030004 orelse Face =:= 16#0A030004 orelse Face =:= 16#14030004 orelse Face =:= 16#15030004 orelse Face =:= 16#16030004 orelse Face =:= 16#17030004 orelse
- Face =:= 16#18030004 orelse Face =:= 16#19030004 orelse Face =:= 16#1A030004 orelse Face =:= 16#1E030004 orelse Face =:= 16#1F030004 orelse Face =:= 16#20030004 orelse
- Face =:= 16#21030004 orelse Face =:= 16#22030004 orelse Face =:= 16#23030004 orelse Face =:= 16#24030004 orelse Face =:= 16#28030004 orelse Face =:= 16#29030004 orelse
- Face =:= 16#2A030004 orelse Face =:= 16#2B030004 orelse Face =:= 16#2C030004 orelse Face =:= 16#2D030004 orelse Face =:= 16#2E030004 orelse Face =:= 16#000D0004,
- EyesGroup = 6,
- true = Eyes =< 6;
-
-validate_char_create(human, female, Tuple) ->
- #flesh_appearance{ears=Ears, face=Face, eyesgroup=EyesGroup, eyes=Eyes} = Tuple,
- validate_char_create_common_flesh(Tuple),
- validate_char_create_common_female_flesh(Tuple),
- true = Ears =:= 16#00001003 orelse Ears =:= 16#00011003,
- true = Face =:= 16#00011004 orelse Face =:= 16#0A011004 orelse Face =:= 16#14011004 orelse Face =:= 16#1E011004 orelse Face =:= 16#3C011004,
- EyesGroup = 1,
- true = Eyes =< 5;
-
-validate_char_create(newman, female, Tuple) ->
- #flesh_appearance{ears=Ears, face=Face, eyesgroup=EyesGroup, eyes=Eyes} = Tuple,
- validate_char_create_common_flesh(Tuple),
- validate_char_create_common_female_flesh(Tuple),
- true = Ears =:= 16#00031003 orelse Ears =:= 16#00651003 orelse Ears =:= 16#00661003,
- true = Face =:= 16#00021004 orelse Face =:= 16#0A021004 orelse Face =:= 16#14021004 orelse Face =:= 16#1E021004 orelse Face =:= 16#3C021004,
- EyesGroup = 3,
- true = Eyes =< 5;
-
-validate_char_create(beast, female, Tuple) ->
- #flesh_appearance{ears=Ears, face=Face, eyesgroup=EyesGroup, eyes=Eyes} = Tuple,
- validate_char_create_common_flesh(Tuple),
- validate_char_create_common_female_flesh(Tuple),
- true = Ears =:= 16#00021003 orelse Ears =:= 16#00CD1003 orelse Ears =:= 16#00CE1003 orelse Ears =:= 16#00CF1003,
- true = Face =:= 16#00031004 orelse Face =:= 16#0A031004 orelse Face =:= 16#14031004 orelse Face =:= 16#1E031004 orelse Face =:= 16#3C031004,
- EyesGroup = 7,
- true = Eyes =< 6.
-
-%% @doc Validate the common settings for all metal characters.
-validate_char_create_common_metal(Tuple) ->
- #metal_appearance{maincolor=MainColor, eyebrows=Eyebrows, eyes=Eyes, eyescolory=EyesColorY, bodycolor=BodyColor, subcolor=SubColor,
- hairstylecolory=HairstyleColorY, hairstylecolorx=HairstyleColorX, proportion=Proportion, proportionboxx=ProportionBoxX, proportionboxy=ProportionBoxY,
- faceboxx=FaceBoxX, faceboxy=FaceBoxY} = Tuple,
- true = MainColor =< 7,
- true = Eyebrows =< 18,
- true = Eyes =< 2,
- true = EyesColorY =< 65535,
- true = BodyColor =< 131071,
- true = SubColor =< 393215,
- true = HairstyleColorY =< 65535,
- true = HairstyleColorX =< 327679,
- true = Proportion =< 131071,
- true = ProportionBoxX =< 131071,
- true = ProportionBoxY =< 131071,
- true = FaceBoxX =< 131071,
- true = FaceBoxY =< 131071.
-
-%% @doc Validate the common settings for all flesh characters.
-validate_char_create_common_flesh(Tuple) ->
- #flesh_appearance{jacketcolor=JacketColor, pantscolor=PantsColor, shoescolor=ShoesColor, eyebrows=Eyebrows, bodysuit=BodySuit,
- eyescolory=EyesColorY, eyescolorx=EyesColorX, skincolor=SkinColor, hairstylecolory=HairstyleColorY, hairstylecolorx=HairstyleColorX,
- proportion=Proportion, proportionboxx=ProportionBoxX, proportionboxy=ProportionBoxY, faceboxx=FaceBoxX, faceboxy=FaceBoxY} = Tuple,
- true = JacketColor =< 4,
- true = PantsColor =< 4,
- true = ShoesColor =< 4,
- true = Eyebrows =< 18,
- true = BodySuit =< 4,
- true = EyesColorY =< 65535,
- true = EyesColorX =< 327679,
- true = SkinColor =< 131071,
- true = HairstyleColorY =< 65535,
- true = HairstyleColorX =< 327679,
- true = Proportion =< 131071,
- true = ProportionBoxX =< 131071,
- true = ProportionBoxY =< 131071,
- true = FaceBoxX =< 131071,
- true = FaceBoxY =< 131071.
-
-%% @doc Validate the common settings for all male flesh characters.
-validate_char_create_common_male_flesh(Tuple) ->
- #flesh_appearance{voicetype=VoiceType, jacket=Jacket, pants=Pants, shoes=Shoes, hairstyle=Hairstyle, eyelashes=Eyelashes} = Tuple,
- true = (VoiceType >= 1 andalso VoiceType =< 14) orelse (VoiceType >= 76 andalso VoiceType =< 83),
- true = Jacket =:= 16#00060000 orelse Jacket =:= 16#00020000 orelse Jacket =:= 16#00030000,
- true = Pants =:= 16#00060001 orelse Pants =:= 16#000B0001 orelse Pants =:= 16#00030001,
- true = Shoes =:= 16#00060002 orelse Shoes =:= 16#00020002 orelse Shoes =:= 16#00040002,
- validate_char_create_male_hairstyle(Hairstyle),
- true = Eyelashes =< 2.
-
-%% @doc Validate the common settings for all female flesh characters.
-validate_char_create_common_female_flesh(Tuple) ->
- #flesh_appearance{voicetype=VoiceType, jacket=Jacket, pants=Pants, shoes=Shoes, hairstyle=Hairstyle, eyelashes=Eyelashes} = Tuple,
- true = (VoiceType >= 15 andalso VoiceType =< 26) orelse (VoiceType >= 84 andalso VoiceType =< 88),
- true = Jacket =:= 16#00011000 orelse Jacket =:= 16#00021000 orelse Jacket =:= 16#00031000,
- true = Pants =:= 16#00011001 orelse Pants =:= 16#00021001 orelse Pants =:= 16#00031001,
- true = Shoes =:= 16#00091002 orelse Shoes =:= 16#00071002 orelse Shoes =:= 16#00031002,
- validate_char_create_female_hairstyle(Hairstyle),
- true = Eyelashes =< 12.
-
-%% @doc Validate the hairstyle for all male characters.
-validate_char_create_male_hairstyle(Hairstyle) ->
- true = Hairstyle =:= 16#00000005 orelse Hairstyle =:= 16#000A0005 orelse Hairstyle =:= 16#00140005 orelse Hairstyle =:= 16#001E0005 orelse Hairstyle =:= 16#00280005 orelse Hairstyle =:= 16#00320005 orelse
- Hairstyle =:= 16#003C0005 orelse Hairstyle =:= 16#00460005 orelse Hairstyle =:= 16#00500005 orelse Hairstyle =:= 16#005A0005 orelse Hairstyle =:= 16#00640005 orelse Hairstyle =:= 16#006E0005 orelse
- Hairstyle =:= 16#00780005 orelse Hairstyle =:= 16#00820005 orelse Hairstyle =:= 16#008C0005 orelse Hairstyle =:= 16#00960005 orelse Hairstyle =:= 16#00A00005 orelse Hairstyle =:= 16#00AA0005.
-
-%% @doc Validate the hairstyle for all female characters.
-validate_char_create_female_hairstyle(Hairstyle) ->
- true = Hairstyle =:= 16#00001005 orelse Hairstyle =:= 16#000A1005 orelse Hairstyle =:= 16#00141005 orelse Hairstyle =:= 16#001E1005 orelse Hairstyle =:= 16#00281005 orelse Hairstyle =:= 16#00321005 orelse
- Hairstyle =:= 16#003C1005 orelse Hairstyle =:= 16#00461005 orelse Hairstyle =:= 16#00501005 orelse Hairstyle =:= 16#005A1005 orelse Hairstyle =:= 16#00641005 orelse Hairstyle =:= 16#006E1005 orelse
- Hairstyle =:= 16#00781005 orelse Hairstyle =:= 16#00821005 orelse Hairstyle =:= 16#008C1005 orelse Hairstyle =:= 16#00961005 orelse Hairstyle =:= 16#00A01005.
diff --git a/apps/egs/src/psu/psu_characters.erl b/apps/egs/src/psu/psu_characters.erl
index 2087b0d..1058b6a 100644
--- a/apps/egs/src/psu/psu_characters.erl
+++ b/apps/egs/src/psu/psu_characters.erl
@@ -19,9 +19,7 @@
-module(psu_characters).
-export([
- character_tuple_to_binary/1, character_user_to_binary/1, class_atom_to_binary/1, class_binary_to_atom/1,
- gender_atom_to_binary/1, gender_binary_to_atom/1, options_binary_to_tuple/1, options_tuple_to_binary/1,
- race_atom_to_binary/1, race_binary_to_atom/1, stats_tuple_to_binary/1
+ character_user_to_binary/1
]).
-include("include/records.hrl").
@@ -34,7 +32,7 @@ character_tuple_to_binary(Tuple) ->
RaceBin = race_atom_to_binary(Race),
GenderBin = gender_atom_to_binary(Gender),
ClassBin = class_atom_to_binary(Class),
- AppearanceBin = psu_appearance:tuple_to_binary(Race, Appearance),
+ AppearanceBin = egs_net:character_appearance_to_binary(Race, Appearance),
LevelsBin = egs_proto:build_char_level(Tuple),
<< Name/binary, RaceBin:8, GenderBin:8, ClassBin:8, AppearanceBin/binary, LevelsBin/binary >>.
@@ -47,8 +45,8 @@ character_tuple_to_binary(Tuple) ->
character_user_to_binary(User) ->
#users{gid=CharGID, lid=CharLID, npcid=NPCid, type=Type, level=Level, stats=Stats, currenthp=CurrentHP, maxhp=MaxHP,
pos={X, Y, Z, Dir}, area={QuestID, ZoneID, MapID}, entryid=EntryID, prev_area={PrevQuestID, PrevZoneID, PrevMapID}, prev_entryid=PrevEntryID} = User,
- CharBin = psu_characters:character_tuple_to_binary(User),
- StatsBin = psu_characters:stats_tuple_to_binary(Stats),
+ CharBin = character_tuple_to_binary(User),
+ StatsBin = stats_tuple_to_binary(Stats),
EXPNextLevel = 100,
EXPCurrentLevel = 0,
IntDir = trunc(Dir * 182.0416),
@@ -66,45 +64,10 @@ character_user_to_binary(User) ->
class_atom_to_binary(Class) ->
case Class of
- hunter -> 0;
- ranger -> 1;
- force -> 2;
- fighgunner -> 3;
- guntecher -> 4;
- wartecher -> 5;
- fortefighter -> 6;
- fortegunner -> 7;
- fortetecher -> 8;
- protranser -> 9;
- acrofighter -> 10;
- acrotecher -> 11;
- fighmaster -> 12;
- gunmaster -> 13;
- masterforce -> 14;
- acromaster -> 15
- end.
-
-%% @doc Convert the binary class to an atom.
-%% @todo Probably can make a list and use that list for both functions.
-
-class_binary_to_atom(ClassBin) ->
- case ClassBin of
- 0 -> hunter;
- 1 -> ranger;
- 2 -> force;
- 3 -> fighgunner;
- 4 -> guntecher;
- 5 -> wartecher;
- 6 -> fortefighter;
- 7 -> fortegunner;
- 8 -> fortetecher;
- 9 -> protranser;
- 10 -> acrofighter;
- 11 -> acrotecher;
- 12 -> fighmaster;
- 13 -> gunmaster;
- 14 -> masterforce;
- 15 -> acromaster
+ hunter -> 12;
+ ranger -> 13;
+ force -> 14;
+ acro -> 15
end.
%% @doc Convert a gender atom into a binary to be sent to clients.
@@ -115,35 +78,6 @@ gender_atom_to_binary(Gender) ->
female -> 1
end.
-%% @doc Convert the binary gender into an atom.
-
-gender_binary_to_atom(GenderBin) ->
- case GenderBin of
- 0 -> male;
- 1 -> female
- end.
-
-%% @doc Convert the binary options data into a tuple.
-%% The few unknown values are probably PS2 or 360 only.
-
-options_binary_to_tuple(Binary) ->
- << TextDisplaySpeed:8, Sound:8, MusicVolume:8, SoundEffectVolume:8, Vibration:8, RadarMapDisplay:8,
- CutInDisplay:8, MainMenuCursorPosition:8, _:8, Camera3rdY:8, Camera3rdX:8, Camera1stY:8, Camera1stX:8,
- Controller:8, WeaponSwap:8, LockOn:8, Brightness:8, FunctionKeySetting:8, _:8, ButtonDetailDisplay:8, _:32 >> = Binary,
- {options, TextDisplaySpeed, Sound, MusicVolume, SoundEffectVolume, Vibration, RadarMapDisplay,
- CutInDisplay, MainMenuCursorPosition, Camera3rdY, Camera3rdX, Camera1stY, Camera1stX,
- Controller, WeaponSwap, LockOn, Brightness, FunctionKeySetting, ButtonDetailDisplay}.
-
-%% @doc Convert a tuple of options data into a binary to be sent to clients.
-
-options_tuple_to_binary(Tuple) ->
- {options, TextDisplaySpeed, Sound, MusicVolume, SoundEffectVolume, Vibration, RadarMapDisplay,
- CutInDisplay, MainMenuCursorPosition, Camera3rdY, Camera3rdX, Camera1stY, Camera1stX,
- Controller, WeaponSwap, LockOn, Brightness, FunctionKeySetting, ButtonDetailDisplay} = Tuple,
- << TextDisplaySpeed, Sound, MusicVolume, SoundEffectVolume, Vibration, RadarMapDisplay,
- CutInDisplay, MainMenuCursorPosition, 0, Camera3rdY, Camera3rdX, Camera1stY, Camera1stX,
- Controller, WeaponSwap, LockOn, Brightness, FunctionKeySetting, 0, ButtonDetailDisplay, 0:32 >>.
-
%% @doc Convert a race atom into a binary to be sent to clients.
race_atom_to_binary(Race) ->
@@ -154,16 +88,6 @@ race_atom_to_binary(Race) ->
beast -> 3
end.
-%% @doc Convert the binary race into an atom.
-
-race_binary_to_atom(RaceBin) ->
- case RaceBin of
- 0 -> human;
- 1 -> newman;
- 2 -> cast;
- 3 -> beast
- end.
-
%% @doc Convert the tuple of stats data into a binary to be sent to clients.
stats_tuple_to_binary(Tuple) ->
diff --git a/apps/egs_net/rebar.config b/apps/egs_net/rebar.config
new file mode 100644
index 0000000..1750439
--- /dev/null
+++ b/apps/egs_net/rebar.config
@@ -0,0 +1,4 @@
+{deps, [
+ {erlson, ".*", {git, "https://github.com/alavrik/erlson.git", "HEAD"}}
+]}.
+{plugins, [erlson_rebar_plugin]}.
diff --git a/apps/egs_net/src/egs_net.app.src b/apps/egs_net/src/egs_net.app.src
new file mode 100644
index 0000000..bd45ea1
--- /dev/null
+++ b/apps/egs_net/src/egs_net.app.src
@@ -0,0 +1,12 @@
+%%-*- mode: erlang -*-
+{application, egs_net, [
+ {description, "EGS network layer."},
+ {vsn, "0.1.0"},
+ {modules, []},
+ {registered, []},
+ {applications, [
+ kernel,
+ stdlib,
+ cowboy
+ ]}
+]}.
diff --git a/apps/egs_net/src/egs_net.erl b/apps/egs_net/src/egs_net.erl
new file mode 100644
index 0000000..07bc495
--- /dev/null
+++ b/apps/egs_net/src/egs_net.erl
@@ -0,0 +1,1576 @@
+%% @author Loïc Hoguin
+%% @copyright 2010-2012 Loïc Hoguin.
+%% @doc Login and game servers network handling.
+%%
+%% This file is part of EGS.
+%%
+%% EGS is free software: you can redistribute it and/or modify
+%% it under the terms of the GNU Affero 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 Affero General Public License for more details.
+%%
+%% You should have received a copy of the GNU Affero General Public License
+%% along with EGS. If not, see .
+
+-module(egs_net).
+
+%% Client state manipulation API.
+-export([init/4]).
+-export([terminate/1]).
+-export([get_gid/1]).
+-export([set_gid/2]).
+-export([set_handler/2]).
+-export([set_keepalive/1]).
+
+%% Receive loop.
+-export([loop/1]).
+
+%% Response API.
+-export([account_character/2]).
+-export([account_characters_response/2]).
+-export([account_flags/4]).
+-export([comm_own_card/2]).
+-export([system_auth_error/2]).
+-export([system_game_server_response/3]).
+-export([system_hello/1]).
+-export([system_key_auth_info/3]).
+-export([system_motd_response/3]).
+-export([system_open_url/2]).
+
+%% @todo Temporary, remove.
+-export([character_appearance_to_binary/2]).
+
+-include_lib("erlson/include/erlson.hrl").
+
+%% Network state.
+
+-record(egs_net, {
+ socket :: ssl:sslsocket(),
+ transport :: module(),
+ handler :: module(),
+ buffer = <<>> :: binary(),
+ keepalive = false :: boolean(),
+
+ gid = 0 :: egs:gid(),
+ targetid = 16#ffff :: egs:targetid(),
+ slot = 0 :: 0..3, %% @todo Remove.
+ areanb = 0 :: non_neg_integer()
+}).
+
+%% Little-endian macros.
+
+-define(l16, :16/little).
+-define(l32, :32/little).
+-define(l32f, :32/little-float).
+
+%% Client state manipulation API.
+
+init(Socket, Transport, Handler, GID) ->
+ #egs_net{socket=Socket, transport=Transport,
+ handler=Handler, gid=GID}.
+
+terminate(#egs_net{socket=Socket, transport=Transport}) ->
+ Transport:close(Socket).
+
+get_gid(#egs_net{gid=GID}) ->
+ GID.
+
+set_gid(GID, State) ->
+ State#egs_net{gid=GID}.
+
+set_handler(Handler, State) ->
+ State#egs_net{handler=Handler}.
+
+set_keepalive(State) ->
+ State#egs_net{keepalive=true}.
+
+%% Receive loop.
+
+loop(State=#egs_net{socket=Socket, transport=Transport}) ->
+ Transport:setopts(Socket, [{active, once}]),
+ {OK, Closed, Error} = Transport:messages(),
+ receive
+ {OK, _, Data} -> handle(State, Data);
+ {Closed, _} -> closed;
+ {Error, _, _} -> closed;
+ {egs, keepalive} -> keepalive(State);
+ Info when element(1, Info) =:= egs -> info(State, Info)
+ end.
+
+handle(State=#egs_net{buffer=Buffer}, Data) ->
+ {Commands, Rest} = split(<< Buffer/binary, Data/binary >>),
+ dispatch(State#egs_net{buffer=Rest}, Commands).
+
+dispatch(State, []) ->
+ ?MODULE:loop(State);
+dispatch(State, [Data|Tail]) ->
+ case parse(Data) of
+ ignore ->
+ dispatch(State, Tail);
+ {Type, Event} ->
+ case call(State, Event, Type) of
+ closed -> closed;
+ ok -> dispatch(State, Tail);
+ {ok, State2} -> dispatch(State2, Tail)
+ end
+ end.
+
+%% If keepalive is enabled we just send an empty packet since the
+%% real keepalive packet is managed by Gameguard which is better disabled.
+keepalive(State=#egs_net{keepalive=false}) ->
+ ?MODULE:loop(State);
+keepalive(State=#egs_net{keepalive=true}) ->
+ send_packet(<< 8?l32, 0:32 >>, State),
+ ?MODULE:loop(State).
+
+info(State, Info) ->
+ case call(State, Info, info) of
+ closed -> closed;
+ ok -> ?MODULE:loop(State);
+ {ok, State2} -> ?MODULE:loop(State2)
+ end.
+
+%% @todo The try..catch should be only enabled for debug.
+call(State=#egs_net{handler=Handler, gid=GID}, Data, Name) ->
+ try
+ %% @todo Make this io:format optional.
+ io:format("~b -> ~p ~p~n", [GID, Name, Data]),
+ Handler:Name(Data, State)
+ catch Class:Reason ->
+ error_logger:error_msg(
+ "** Handler error in ~p for ~p:~n"
+ " ~p~n"
+ " for the reason ~p:~p~n"
+ "** Stacktrace: ~p~n~n",
+ [Handler, Name, Data, Class, Reason, erlang:get_stacktrace()])
+ end.
+
+%% Packet parsing code.
+
+split(Data) ->
+ split(Data, []).
+split(Data, Acc) when byte_size(Data) < 4 ->
+ {lists:reverse(Acc), Data};
+split(Data = << Size:32/little, _/bits >>, Acc) when Size > byte_size(Data) ->
+ {lists:reverse(Acc), Data};
+split(Data = << Size:32/little, _/bits >>, Acc) ->
+ << Packet:Size/binary, Rest/bits >> = Data,
+ split(Rest, [Packet|Acc]).
+
+%% Completely ignore the fragmented packet replies.
+parse(<< 8:32/little, 16#0b05:16, _:16 >>) ->
+ ignore;
+%% Catch parse errors and prints a dump of the packet with useful info.
+parse(<< Size:32/little, Category:8, Sub:8, Channel:8, _:8, Data/bits >>) ->
+ try begin
+ Event = parse(Size, Category * 256 + Sub, Channel, Data),
+ case {Event, Channel} of
+ {ignore, _} -> ignore; %% @todo Always return something.
+ {Event, 1} -> {cast, Event};
+ {Event, _} -> {event, Event}
+ end
+ end catch Class:Reason ->
+ error_logger:error_msg(lists:flatten([
+ "** Parse error in with reason ~p:~p~n"
+ " for command #~2.16.0b~2.16.0b of size ~b on channel ~b~n~n",
+ binary_to_dump(Data), "~n~n"]),
+ [Class, Reason, Category, Sub, Size, Channel]),
+ ignore
+ end.
+
+%% @todo Documentation.
+%% @todo Probably shouldn't be ignored.
+parse(92, 16#0102, 2, Data) ->
+ << DestTargetID?l16, _:16,
+ CharActType:32, CharGID?l32, 0:32, 0:32,
+ DestActType:32, DestGID?l32, 0:32, 0:32,
+ CharGID?l32, CharTargetID?l16, _:16,
+ AnimType:8, AnimState:8, Dir?l16, 0:32,
+ X?l32f, Y?l32f, Z?l32f, QuestID?l32,
+ ZoneID?l16, _:16, MapID?l16, _:16, EntryID?l16, _:16, 0:32
+ >> = Data,
+ ignore;
+
+%% @todo Documentation.
+%% @todo A B
+parse(Size, 16#0105, 2, Data) ->
+ << DestTargetID?l16, _:16,
+ 0:32, CharGID?l32, 0:32, 0:32,
+ DestActType:32, DestGID?l32, 0:32, 0:32,
+ CharGID?l32, CharTargetID?l16, _:16,
+ ItemIndex:8, EventID:8, PAID:8, A:8,
+ B?l32, Rest/binary
+ >> = Data,
+ Event = item_eventid_to_atom(EventID),
+ case Event of
+ item_drop ->
+ Size = 76,
+ << Quantity?l32, X?l32f, Y?l32f, Z?l32f >> = Rest,
+ {Event, ItemIndex, CharGID, CharTargetID, A, B,
+ Quantity, X, Y, Z};
+ Event ->
+ Size = 60,
+ <<>> = Rest,
+ {Event, ItemIndex, CharGID, CharTargetID, A, B}
+ end;
+
+%% @todo Documentation.
+%% @todo A _B
+parse(60, 16#010a, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ DestGID?l32, DestTargetID?l16, _:16,
+ EventID?l16, QuantityOrColor:8, A:8, Param:32/bits
+ >> = Data,
+ Event = npc_shop_eventid_to_atom(EventID),
+ case Event of
+ Event when Event =:= npc_shop_enter; Event =:= npc_shop_leave ->
+ << ShopID?l16, 0:16 >> = Param,
+ QuantityOrColor = 0,
+ A = 0,
+ {Event, ShopID};
+ npc_shop_buy ->
+ << ItemIndex?l16, 0:16 >> = Param,
+ QuantityOrColor = A,
+ {Event, ItemIndex, QuantityOrColor};
+ npc_shop_sell ->
+ << ItemIndex:8, _B:8, 0:16 >> = Param,
+ A = 0,
+ {Event, ItemIndex, QuantityOrColor}
+ end;
+
+%% @todo Documentation.
+%% @todo Probably shouldn't be ignored.
+%% @todo We should send the spawn to everyone in this command,
+%% rather than in area_change.
+parse(92, 16#010b, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:32, CharGID?l32, 0:64, 0:128,
+ CharGID?l32, CharTargetID?l16, _:16, 0:16, Dir?l16,
+ X?l32f, Y?l32f, Z?l32f, 0:64,
+ QuestID?l32, ZoneID?l16, _:16, MapID?l16, _:16, EntryID?l16, _:16
+ >> = Data,
+ DestTargetID = CharTargetID,
+ ignore; %% @todo character_enter_area
+
+%% @todo Documentation.
+parse(60, 16#0110, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:32, CharGID?l32, 0:64, 0:128,
+ CharGID?l32, CharTargetID?l16, _:16, EventID?l32, Param?l32
+ >> = Data,
+ Event = character_eventid_to_atom(EventID),
+ case Event of
+ character_type_change ->
+ {character_type_change, Param};
+ character_status_change ->
+ {character_status_change, Param};
+ Event when Event =/= unknown ->
+ Param = 0,
+ Event
+ end;
+
+parse(52, 16#020b, 2, Data) ->
+ << 16#ffff:16, _:16, 0:128, 0:128,
+ Slot:32/little, 0:8, BackToField:8, 0:16
+ >> = Data,
+ BackToFieldAtom = case BackToField of 0 -> false; 1 -> true end,
+ {system_character_select, Slot, BackToFieldAtom};
+
+parse(60, 16#020d, 2, Data) ->
+ << 16#ffff:16, _:16, 0:128, 0:128,
+ AuthGID:32/little, AuthKey:32/bits, 0:64
+ >> = Data,
+ {system_key_auth, AuthGID, AuthKey};
+
+parse(44, 16#0217, 2, Data) ->
+ << 16#ffff:16, _:16, 0:128, 0:128
+ >> = Data,
+ system_game_server_request;
+
+%% @todo _A _B
+parse(100, 16#0219, 2, Data) ->
+ << 16#ffff:16, _:16, 0:128, 0:128,
+ Username:192/bits,
+ Password:192/bits,
+ _A?l32, _B?l32
+ >> = Data,
+ Username2 = iolist_to_binary(
+ re:split(Username, "\\0", [{return, binary}])),
+ Password2 = iolist_to_binary(
+ re:split(Password, "\\0", [{return, binary}])),
+ {system_login_auth, Username2, Password2};
+
+parse(44, 16#021c, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128
+ >> = Data,
+ system_character_load_complete;
+
+parse(48, 16#021d, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ EntryID?l32
+ >> = Data,
+ unicube_request;
+
+parse(52, 16#021f, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ UniID?l32, EntryID?l32
+ >> = Data,
+ case UniID of
+ 0 -> ignore;
+ UniID -> {unicube_select, UniID, EntryID}
+ end;
+
+%% Same as 023f, except for the odd channel, and that only JP clients use it.
+parse(48, 16#0226, 3, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ Page:8, Language:8, 0:16
+ >> = Data,
+ {system_motd_request, Page, language_to_atom(Language)};
+
+%% Whether the MOTD is accepted.
+parse(48, 16#0227, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ AcceptMOTD?l32
+ >> = Data,
+ ignore;
+
+%% Same as 0226, except for the odd channel, and that only US clients use it.
+parse(48, 16#023f, 2, Data) ->
+ << 16#ffff:16, _:16, 0:128, 0:128,
+ Page:8, Language:8, 0:16
+ >> = Data,
+ {system_motd_request, Page, language_to_atom(Language)};
+
+%% @todo Check Size properly.
+%% @todo _A
+parse(Size, 16#0304, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ CharAccType?l32, CharGID?l32, 0:64,
+ ChatType:8, ChatCutIn:8, ChatCutInAngle:8, ChatLength:8,
+ ChatChannel:8, ChatCharType:8, 0:8, _A:8,
+ CharName:512/bits, ChatMsg/binary
+ >> = Data,
+ ChatTypeA = chat_type_to_atom(ChatType),
+ ChatCutInA = chat_cutin_to_atom(ChatCutIn),
+ ChatChannelA = chat_channel_to_atom(ChatChannel),
+ ChatCharTypeA = chat_character_type_to_atom(ChatCharType),
+ Modifiers = {chat_modifiers, ChatTypeA,
+ ChatCutInA, ChatCutInAngle, ChatChannelA, ChatCharTypeA},
+ {chat, CharAccType, CharGID, CharName, Modifiers, ChatLength, ChatMsg};
+
+%% @todo AreaNb should be the same that was sent with 0205 apparently.
+parse(48, 16#0806, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ AreaNb?l32
+ >> = Data,
+ ignore;
+
+%% @todo Check if NbAreaChanges is related to AreaNb or something.
+parse(60, 16#0807, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ QuestID?l32, ZoneID?l16, MapID?l16, EntryID?l16,
+ NbAreaChanges?l16, PartyPos?l32
+ >> = Data,
+ {area_change, QuestID, ZoneID, MapID, EntryID, PartyPos};
+
+%% @todo AreaNb should be the same that was sent with 0208 apparently.
+parse(48, 16#0808, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128, AreaNb?l32
+ >> = Data,
+ ignore;
+
+parse(648, 16#080c, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ 16#ffffffff:32, APCid?l16, _A?l16,
+ 16#ffffffff:32, 16#ffffffff:32, 0:16, _B?l16,
+ 0:4736
+ >> = Data,
+ {apc_force_invite, APCid};
+
+%% @todo Probably indicates a successful area change.
+parse(44, 16#080d, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128
+ >> = Data,
+ ignore;
+
+parse(68, 16#080e, 2, Data) ->
+ << 16#ffff:16, _:16, 0:128, 0:128,
+ 0:8, Language:8, 1:8, Entrance:8, Platform:8, 0:24,
+ Revision:8, Minor:4, _A:12, Major:4, _B:4,
+ 0:96
+ >> = Data,
+ LanguageA = language_to_atom(Language),
+ PlatformA = platform_to_atom(Platform),
+ Version = Major * 1000000 + Minor * 1000 + Revision,
+ {client_version, LanguageA, Entrance, PlatformA, Version};
+
+%% @todo No idea what this packet is about.
+parse(48, 16#080f, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ PartyPos?l32
+ >> = Data,
+ ignore;
+
+parse(60, 16#0811, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ CounterType:8, 41:8, FromZoneID?l16, FromMapID?l16, FromEntryID?l16,
+ CounterID?l32, 16#ffffffff:32
+ >> = Data,
+ {counter_enter, CounterID, FromZoneID, FromMapID, FromEntryID};
+
+parse(44, 16#0812, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128
+ >> = Data,
+ counter_leave;
+
+parse(52, 16#0813, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ 16#ffffffff:32, APCid?l32
+ >> = Data,
+ {npc_invite, APCid};
+
+%% @todo Probably indicates a successful mission block change.
+parse(44, 16#0814, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128
+ >> = Data,
+ ignore;
+
+%% @todo Probably indicates a successful area change.
+parse(44, 16#0815, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128
+ >> = Data,
+ ignore;
+
+parse(160, 16#0818, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ GPU:512/bits,
+ CPU:384/bits,
+ _A?l32
+ >> = Data,
+ GPU2 = iolist_to_binary(re:split(GPU, "\\0", [{return, binary}])),
+ CPU2 = iolist_to_binary(re:split(CPU, "\\0", [{return, binary}])),
+ {client_hardware, GPU2, CPU2};
+
+parse(48, 16#0a10, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ ItemID?l32
+ >> = Data,
+ {item_description_request, ItemID};
+
+parse(48, 16#0c01, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ QuestID?l32
+ >> = Data,
+ {mission_start, QuestID};
+
+parse(48, 16#0c05, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ CounterID?l32
+ >> = Data,
+ {counter_quest_files_request, CounterID};
+
+%% On official, Price = Rate * 200.
+parse(52, 16#0c07, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ QuestID?l32, Rate?l32
+ >> = Data,
+ lobby_transport_request;
+
+%% Probably indicates a successful mission block change.
+parse(44, 16#0c0d, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128
+ >> = Data,
+ ignore;
+
+parse(44, 16#0c0e, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128
+ >> = Data,
+ mission_abort;
+
+parse(48, 16#0c0f, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ CounterID?l32
+ >> = Data,
+ {counter_quest_options_request, CounterID};
+
+%% We completely skip the class given and set one depending on the
+%% race we received.
+%%
+%% We completely skip the class levels part as we can't trust what the
+%% client tells us, and they should be set to 1 everywhere anyway.
+parse(324, 16#0d02, 2, Data) ->
+ << 0:32, 0:128, 0:128,
+ Slot?l32,
+ NameBin:512/bits,
+ Race:8, Gender:8, _:8,
+ Infos/bits
+ >> = Data,
+ Name = iolist_to_binary(
+ re:split(NameBin, "\\0\\0", [{return, binary}])),
+ RaceAtom = race_to_atom(Race),
+ GenderAtom = gender_to_atom(Gender),
+ ClassAtom = case RaceAtom of
+ human -> acro;
+ newman -> force;
+ cast -> ranger;
+ beast -> hunter
+ end,
+ Appearance = case RaceAtom of
+ cast ->
+ << VoiceType:8, VoicePitch:8, _:24,
+ Torso:32, Legs:32, Arms:32,
+ Ears:32, Face:32, HeadType:32,
+ MainColor:8, _:32,
+ Eyebrows:8, Eyelashes:8, EyesGroup:8, Eyes:8,
+ _:24, EyesColorY?l32, EyesColorX?l32,
+ _:96, BodyColor?l32, SubColor?l32,
+ HairstyleColorY?l32, HairstyleColorX?l32,
+ Proportion?l32, ProportionBoxX?l32, ProportionBoxY?l32,
+ FaceBoxX?l32, FaceBoxY?l32,
+ _/binary
+ >> = Infos,
+ #{
+ arms=Arms,
+ body_color=BodyColor,
+ ears=Ears,
+ eyebrows=Eyebrows,
+ eyelashes=Eyelashes,
+ eyes_group=EyesGroup,
+ eyes=Eyes,
+ eyes_color_x=EyesColorX,
+ eyes_color_y=EyesColorY,
+ face=Face,
+ face_box_x=FaceBoxX,
+ face_box_y=FaceBoxY,
+ hairstyle_color_x=HairstyleColorX,
+ hairstyle_color_y=HairstyleColorY,
+ head_type=HeadType,
+ legs=Legs,
+ main_color=MainColor,
+ proportion=Proportion,
+ proportion_box_x=ProportionBoxX,
+ proportion_box_y=ProportionBoxY,
+ shield_color=0,
+ sub_color=SubColor,
+ torso=Torso,
+ voice_pitch=VoicePitch,
+ voice_type=VoiceType
+ };
+ RaceAtom ->
+ << VoiceType:8, VoicePitch:8, _:24,
+ Jacket:32, Pants:32, Shoes:32,
+ Ears:32, Face:32, Hairstyle:32,
+ JacketColor:8, PantsColor:8, ShoesColor:8,
+ _:16, Eyebrows:8, Eyelashes:8, EyesGroup:8, Eyes:8,
+ BodySuit:8, _:16, EyesColorY?l32, EyesColorX?l32,
+ _:96, SkinColor?l32, _:32,
+ HairstyleColorY?l32, HairstyleColorX?l32,
+ Proportion?l32, ProportionBoxX?l32, ProportionBoxY?l32,
+ FaceBoxX?l32, FaceBoxY?l32,
+ _/binary
+ >> = Infos,
+ #{
+ blast_badge=0,
+ body_suit=BodySuit,
+ ears=Ears,
+ eyebrows=Eyebrows,
+ eyelashes=Eyelashes,
+ eyes_group=EyesGroup,
+ eyes=Eyes,
+ eyes_color_x=EyesColorX,
+ eyes_color_y=EyesColorY,
+ face=Face,
+ face_box_x=FaceBoxX,
+ face_box_y=FaceBoxY,
+ hairstyle=Hairstyle,
+ hairstyle_color_x=HairstyleColorX,
+ hairstyle_color_y=HairstyleColorY,
+ jacket=Jacket,
+ jacket_color=JacketColor,
+ lips_color_x=0,
+ lips_color_y=32767,
+ lips_intensity=32767,
+ pants=Pants,
+ pants_color=PantsColor,
+ proportion=Proportion,
+ proportion_box_x=ProportionBoxX,
+ proportion_box_y=ProportionBoxY,
+ shield_color=0,
+ shoes=Shoes,
+ shoes_color=ShoesColor,
+ skin_color=SkinColor,
+ voice_pitch=VoicePitch,
+ voice_type=VoiceType
+ }
+ end,
+ validate_new_character(RaceAtom, GenderAtom, Appearance),
+ {account_create_character, Slot, Name,
+ RaceAtom, GenderAtom, ClassAtom, Appearance};
+
+parse(44, 16#0d06, 2, Data) ->
+ << 0:32, 0:128, 0:128
+ >> = Data,
+ account_characters_request;
+
+parse(68, 16#0d07, 2, Data) ->
+ << 0:32, 0:128, 0:128,
+ TextDisplaySpeed:8, Sound:8, MusicVolume:8, SoundEffectVolume:8,
+ Vibration:8, RadarMapDisplay:8, CutInDisplay:8, MainMenuCursorPos:8,
+ 0:8, Camera3rdY:8, Camera3rdX:8, Camera1stY:8,
+ Camera1stX:8, Controller:8, WeaponSwap:8, LockOn:8,
+ Brightness:8, FunctionKeySetting:8, 0:8, ButtonDetailDisplay:8,
+ 0:32
+ >> = Data,
+ %% Make sure the options are valid.
+ true = TextDisplaySpeed =< 1,
+ true = Sound =< 1,
+ true = MusicVolume =< 9,
+ true = SoundEffectVolume =< 9,
+ true = Vibration =< 1,
+ true = RadarMapDisplay =< 1,
+ true = CutInDisplay =< 1,
+ true = MainMenuCursorPos =< 1,
+ true = Camera3rdY =< 1,
+ true = Camera3rdX =< 1,
+ true = Camera1stY =< 1,
+ true = Camera1stX =< 1,
+ true = Controller =< 1,
+ true = WeaponSwap =< 1,
+ true = LockOn =< 1,
+ true = Brightness =< 4,
+ true = FunctionKeySetting =< 1,
+ true = ButtonDetailDisplay =< 2,
+ %% Options are considered safe past this point.
+ Options = {options,
+ TextDisplaySpeed, Sound, MusicVolume, SoundEffectVolume,
+ Vibration, RadarMapDisplay, CutInDisplay, MainMenuCursorPos,
+ Camera3rdY, Camera3rdX, Camera1stY, Camera1stX,
+ Controller, WeaponSwap, LockOn, Brightness,
+ FunctionKeySetting, ButtonDetailDisplay
+ },
+ {account_set_options, Options};
+
+%% @todo Maybe the first 32+128+128 bits contain useful info?
+%% @todo HitNb is an auto incremented hit number.
+parse(Size, 16#0e00, 2, Data) ->
+ << _:288,
+ NbHits?l32, PartyPos?l32, HitNb?l32,
+ HitsBin/binary
+ >> = Data,
+ Size = 56 + NbHits * 80,
+ Hits = [{hit, FromTargetID, ToTargetID}
+ || << X1?l32f, Y1?l32f, Z1?l32f, FromTargetID?l32, ToTargetID?l32,
+ _:64,
+ _:128, %% probably anim+dir followed by x,y,z
+ _:128, %% probably the same
+ _:128, _:32,
+ Rest/binary >>
+ <= HitsBin],
+ {hits, Hits};
+
+%% ObjectBaseTargetID is ObjectTargetID - 1024 or 16#ffff.
+%% All the ffffffff after PartyPos are other PartyPos values too.
+%% @todo Those aren't PartyPos but TargetID of the party.
+parse(112, 16#0f0a, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ BlockID?l16, GroupNb?l16, ObjectNb?l16, MapID?l16,
+ ObjectID?l16, A?l16, ObjectTargetID?l32,
+ ObjectType?l16, 0:16, ObjectBaseTargetID?l16, B?l16,
+ PartyPos?l32, C?l32, D?l32, 16#ffffffff:32,
+ 16#ffffffff:32, 16#ffffffff:32, 0:32, 0:32,
+ 0:32, ObjectType?l16, EventID:8, NbTargets:8,
+ E?l32
+ >> = Data,
+ case {ObjectType, EventID} of
+ { 5, 13} ->
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ {object_switch_on, ObjectID};
+ { 5, 14} ->
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ {object_switch_off, ObjectID};
+ { 9, 20} ->
+ ignore; %% @todo object_sensor_trigger
+ {14, 0} ->
+ ObjectID = 16#ffff,
+ ObjectTargetID = 16#ffffffff,
+ ObjectBaseTargetID = 16#ffff,
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ {object_warp_enter, BlockID, GroupNb, ObjectNb};
+ {22, 12} ->
+ A = 134,
+ ObjectTargetID = 16#ffffffff,
+ ObjectBaseTargetID = 16#ffff,
+ B = 116,
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ {object_key_console_enable, ObjectID};
+ {22, 23} ->
+ A = 134,
+ ObjectTargetID = 16#ffffffff,
+ ObjectBaseTargetID = 16#ffff,
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ {object_key_console_init, ObjectID};
+ {22, 24} ->
+ A = 134,
+ ObjectTargetID = 16#ffffffff,
+ ObjectBaseTargetID = 16#ffff,
+ B = 116,
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ {object_key_console_open_gate, ObjectID};
+ {31, 12} ->
+ A = 134,
+ ObjectTargetID = 16#ffffffff,
+ ObjectBaseTargetID = 16#ffff,
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ {object_key_enable, ObjectID};
+ {48, 4} ->
+ A = 134,
+ B = 116,
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ {object_boss_gate_enter, ObjectID};
+ {48, 5} ->
+ A = 134,
+ B = 116,
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ {object_boss_gate_leave, ObjectID};
+ {48, 6} ->
+ A = 134,
+ B = 116,
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ {object_boss_gate_activate, ObjectID};
+ {48, 7} ->
+ A = 134,
+ B = 116,
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ unknown; %% @todo object_boss_gate_??
+ {49, 3} ->
+ A = 134,
+ ObjectTargetID = 16#ffffffff,
+ ObjectBaseTargetID = 16#ffff,
+ B = 116,
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ {object_crystal_activate, ObjectID};
+ {50, 9} ->
+ %% @todo Handle more than one PartyPos.
+ A = 134,
+ ObjectTargetID = 16#ffffffff,
+ ObjectBaseTargetID = 16#ffff,
+ B = 116,
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ {object_healing_pad_tick, [PartyPos]};
+ {51, 1} ->
+ B = 116,
+ C = ObjectTargetID,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ {object_goggle_target_activate, ObjectID};
+ {56, 25} ->
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ %% @todo Do we only have the ObjectTargetID here?
+ {object_chair_sit, ObjectTargetID};
+ {56, 26} ->
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ %% @todo Do we only have the ObjectTargetID here?
+ {object_chair_stand, ObjectTargetID};
+ {57, 12} ->
+ A = 134,
+ ObjectTargetID = 16#ffffffff,
+ ObjectBaseTargetID = 16#ffff,
+ B = 116,
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ {object_vehicle_boost_enable, ObjectID};
+ {57, 28} ->
+ A = 134,
+ ObjectTargetID = 16#ffffffff,
+ ObjectBaseTargetID = 16#ffff,
+ B = 116,
+ C = 16#ffffffff,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ {object_vehicle_boost_respawn, ObjectID};
+ {71, 27} ->
+ A = 134,
+ C = ObjectTargetID,
+ D = 16#ffffffff,
+ NbTargets = 1,
+ E = 0,
+ unknown %% @todo object_trap3_??
+ end;
+
+parse(112, 16#1007, 2, Data) ->
+ << 0:32, 0:128, 0:128,
+ PartyPos?l32, CharName:512/bits
+ >> = Data,
+ {party_remove_member, PartyPos};
+
+parse(52, 16#1701, 2, Data) ->
+ << 0:32, 0:128, 0:128,
+ 0:32, 16#ffffffff
+ >> = Data,
+ counter_party_list_request;
+
+parse(44, 16#1705, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128
+ >> = Data,
+ counter_party_info_request;
+
+%% @todo Probably needs to be broadcasted to other players in the party.
+parse(48, 16#1707, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ QuestID?l32
+ >> = Data,
+ ignore; %% @todo {counter_quest_select, QuestID}
+
+parse(44, 16#1709, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128
+ >> = Data,
+ counter_party_options_request;
+
+parse(44, 16#170b, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128
+ >> = Data,
+ counter_background_locations_request;
+
+parse(48, 16#1710, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ CounterID?l32
+ >> = Data,
+ {counter_options_request, CounterID};
+
+parse(64, 16#1a01, 2, Data) ->
+ << DestTargetID?l16, _:16, 0:128, 0:128,
+ DestTargetID?l16, _:16, ShopID?l32, EventID?l32,
+ A?l32, B?l32
+ >> = Data,
+ Event = dialog_eventid_to_atom(EventID),
+ case Event of
+ npc_shop_request ->
+ A = 0,
+ {Event, ShopID};
+ lumilass_options_request ->
+ ShopID = 0,
+ A = 0,
+ Event;
+ ppcube_request ->
+ ShopID = 0,
+ A = 0,
+ Event;
+ ppcube_charge_all ->
+ ShopID = 0,
+ unknown; %% @todo
+ ppcube_charge_one ->
+ ShopID = 0,
+ unknown; %% @todo
+ put_on_outfit ->
+ ShopID = 0,
+ A = 0,
+ unknown; %% @todo
+ remove_outfit ->
+ ShopID = 0,
+ unknown; %% @todo
+ player_type_availability_request ->
+ ShopID = 0,
+ A = 0,
+ B = 0,
+ Event
+ end;
+
+%% @todo Proper cast parsing.
+parse(_, Command, 1, Data) ->
+ {Command, << 0:32, Command:16, 1:8, 0:8, Data/binary >>}.
+
+%% Data validation.
+
+validate_new_character(cast, male, Appearance) ->
+ validate_new_metal_character(Appearance),
+ VoiceType = Appearance.voice_type,
+ true = (VoiceType >= 27 andalso VoiceType =< 38)
+ orelse (VoiceType >= 89 andalso VoiceType =< 96),
+ true = Appearance.eyelashes =< 2,
+ 4 = Appearance.eyes_group,
+ true = lists:member(Appearance.torso,
+ [16#00F70100, 16#00F90100, 16#00FC0100]),
+ true = lists:member(Appearance.legs,
+ [16#00F70101, 16#00F90101, 16#00FC0101]),
+ true = lists:member(Appearance.arms,
+ [16#00F70102, 16#00F90102, 16#00FC0102]),
+ case lists:member(Appearance.face, [16#00040004, 16#0A040004,
+ 16#14040004, 16#1E040004, 16#28040004, 16#000E0004]) of
+ true ->
+ true = lists:member(Appearance.ears, [16#001E0003,
+ 16#001F0003, 16#00200003, 16#00210003, 16#00220003]),
+ true = Appearance.eyes_color_x =< 327679,
+ validate_new_male_hairstyle(Appearance.head_type);
+ false ->
+ true = lists:member(Appearance.face, [16#00F40104,
+ 16#00F50104, 16#00F60104, 16#00F70104, 16#00F80104,
+ 16#00F90104, 16#00FA0104, 16#00FD0104, 16#00020204,
+ 16#00030204, 16#00040204, 16#00060204, 16#00070204]),
+ 16#FFFFFFFF = Appearance.ears,
+ true = Appearance.eyes_color_x =< 458751,
+ true = lists:member(Appearance.head_type, [16#00F40105,
+ 16#00F50105, 16#00F60105, 16#00F70105, 16#00F80105,
+ 16#00F90105, 16#00FA0105, 16#00FB0105, 16#00FD0105,
+ 16#00020205, 16#00030205, 16#00040205, 16#00060205,
+ 16#00070205])
+ end;
+
+validate_new_character(cast, female, Appearance) ->
+ validate_new_metal_character(Appearance),
+ VoiceType = Appearance.voice_type,
+ true = (VoiceType >= 39 andalso VoiceType =< 50)
+ orelse (VoiceType >= 97 andalso VoiceType =< 101),
+ true = Appearance.eyelashes =< 12,
+ 5 = Appearance.eyes_group,
+ true = lists:member(Appearance.torso,
+ [16#00F51100, 16#00F91100, 16#00FA1100]),
+ true = lists:member(Appearance.legs,
+ [16#00F51101, 16#00F91101, 16#00FA1101]),
+ true = lists:member(Appearance.arms,
+ [16#00F51102, 16#00F91102, 16#00F61102]),
+ case lists:member(Appearance.face, [16#00041004, 16#0A041004,
+ 16#14041004, 16#1E041004, 16#3C041004]) of
+ true ->
+ true = lists:member(Appearance.ears, [16#001E1003,
+ 16#001F1003, 16#00201003, 16#00211003, 16#00221003]),
+ true = Appearance.eyes_color_x =< 327679,
+ validate_new_female_hairstyle(Appearance.head_type);
+ false ->
+ true = lists:member(Appearance.face, [16#00F41104,
+ 16#00F51104, 16#00F61104, 16#00F71104, 16#00F81104,
+ 16#00F91104, 16#00FA1104, 16#00FD1104, 16#00031204,
+ 16#00041204, 16#00051204, 16#00061204, 16#00081204]),
+ 16#FFFFFFFF = Appearance.ears,
+ true = Appearance.eyes_color_x =< 458751,
+ true = lists:member(Appearance.head_type, [16#00F41105,
+ 16#00F51105, 16#00F61105, 16#00F71105, 16#00F81105,
+ 16#00F91105, 16#00FA1105, 16#00FB1105, 16#00FD1105,
+ 16#00031205, 16#00041205, 16#00051205, 16#00061205,
+ 16#00081205])
+ end;
+
+validate_new_character(human, male, Appearance) ->
+ validate_new_fleshy_character(Appearance),
+ validate_new_fleshy_male_character(Appearance),
+ true = lists:member(Appearance.ears, [16#00000003, 16#00010003]),
+ true = lists:member(Appearance.face, [16#00010004, 16#01010004,
+ 16#14010004, 16#15010004, 16#16010004, 16#17010004, 16#18010004,
+ 16#19010004, 16#1A010004, 16#1E010004, 16#1F010004, 16#20010004,
+ 16#21010004, 16#22010004, 16#23010004, 16#24010004, 16#28010004,
+ 16#29010004, 16#2A010004, 16#2B010004, 16#2C010004, 16#2D010004,
+ 16#2E010004, 16#000B0004]),
+ 0 = Appearance.eyes_group,
+ true = Appearance.eyes =< 5;
+
+validate_new_character(newman, male, Appearance) ->
+ validate_new_fleshy_character(Appearance),
+ validate_new_fleshy_male_character(Appearance),
+ true = lists:member(Appearance.ears,
+ [16#00030003, 16#00650003, 16#00660003]),
+ true = lists:member(Appearance.face, [16#00020004, 16#01020004,
+ 16#14020004, 16#15020004, 16#16020004, 16#17020004, 16#18020004,
+ 16#19020004, 16#1A020004, 16#1E020004, 16#1F020004, 16#20020004,
+ 16#21020004, 16#22020004, 16#23020004, 16#24020004, 16#28020004,
+ 16#29020004, 16#2A020004, 16#2B020004, 16#2C020004, 16#2D020004,
+ 16#2E020004, 16#000C0004]),
+ 2 = Appearance.eyes_group,
+ true = Appearance.eyes =< 5;
+
+validate_new_character(beast, male, Appearance) ->
+ validate_new_fleshy_character(Appearance),
+ validate_new_fleshy_male_character(Appearance),
+ true = lists:member(Appearance.ears,
+ [16#00020003, 16#00CD0003, 16#00CE0003]),
+ true = lists:member(Appearance.face, [16#00030004, 16#0A030004,
+ 16#14030004, 16#15030004, 16#16030004, 16#17030004, 16#18030004,
+ 16#19030004, 16#1A030004, 16#1E030004, 16#1F030004, 16#20030004,
+ 16#21030004, 16#22030004, 16#23030004, 16#24030004, 16#28030004,
+ 16#29030004, 16#2A030004, 16#2B030004, 16#2C030004, 16#2D030004,
+ 16#2E030004, 16#000D0004]),
+ 6 = Appearance.eyes_group,
+ true = Appearance.eyes =< 6;
+
+validate_new_character(human, female, Appearance) ->
+ validate_new_fleshy_character(Appearance),
+ validate_new_fleshy_female_character(Appearance),
+ true = lists:member(Appearance.ears, [16#00001003, 16#00011003]),
+ true = lists:member(Appearance.face, [16#00011004, 16#0A011004,
+ 16#14011004, 16#1E011004, 16#3C011004]),
+ 1 = Appearance.eyes_group,
+ true = Appearance.eyes =< 5;
+
+validate_new_character(newman, female, Appearance) ->
+ validate_new_fleshy_character(Appearance),
+ validate_new_fleshy_female_character(Appearance),
+ true = lists:member(Appearance.ears,
+ [16#00031003, 16#00651003, 16#00661003]),
+ true = lists:member(Appearance.face, [16#00021004, 16#0A021004,
+ 16#14021004, 16#1E021004, 16#3C021004]),
+ 3 = Appearance.eyes_group,
+ true = Appearance.eyes =< 5;
+
+validate_new_character(beast, female, Appearance) ->
+ validate_new_fleshy_character(Appearance),
+ validate_new_fleshy_female_character(Appearance),
+ true = lists:member(Appearance.ears,
+ [16#00021003, 16#00CD1003, 16#00CE1003, 16#00CF1003]),
+ true = lists:member(Appearance.face, [16#00031004, 16#0A031004,
+ 16#14031004, 16#1E031004, 16#3C031004]),
+ 7 = Appearance.eyes_group,
+ true = Appearance.eyes =< 6.
+
+validate_new_metal_character(Appearance) ->
+ true = Appearance.body_color =< 131071,
+ true = Appearance.eyes =< 2,
+ true = Appearance.eyes_color_y =< 65535,
+ true = Appearance.main_color =< 7,
+ true = Appearance.sub_color =< 393215,
+ validate_new_common_character(Appearance).
+
+validate_new_fleshy_character(Appearance) ->
+ true = Appearance.body_suit =< 4,
+ true = Appearance.eyes_color_x =< 327679,
+ true = Appearance.jacket_color =< 4,
+ true = Appearance.pants_color =< 4,
+ true = Appearance.shoes_color =< 4,
+ true = Appearance.skin_color =< 131071,
+ validate_new_common_character(Appearance).
+
+validate_new_common_character(Appearance) ->
+ true = Appearance.eyebrows =< 18,
+ true = Appearance.eyes_color_y =< 65535,
+ true = Appearance.face_box_x =< 131071,
+ true = Appearance.face_box_y =< 131071,
+ true = Appearance.hairstyle_color_x =< 327679,
+ true = Appearance.hairstyle_color_y =< 65535,
+ true = Appearance.proportion =< 131071,
+ true = Appearance.proportion_box_x =< 131071,
+ true = Appearance.proportion_box_y =< 131071.
+
+validate_new_fleshy_male_character(Appearance) ->
+ VoiceType = Appearance.voice_type,
+ true = (VoiceType >= 1 andalso VoiceType =< 14)
+ orelse (VoiceType >= 76 andalso VoiceType =< 83),
+ true = lists:member(Appearance.jacket,
+ [16#00060000, 16#00020000, 16#00030000]),
+ true = lists:member(Appearance.pants,
+ [16#00060001, 16#000B0001, 16#00030001]),
+ true = lists:member(Appearance.shoes,
+ [16#00060002, 16#00020002, 16#00040002]),
+ validate_new_male_hairstyle(Appearance.hairstyle),
+ true = Appearance.eyelashes =< 2.
+
+validate_new_fleshy_female_character(Appearance) ->
+ VoiceType = Appearance.voice_type,
+ true = (VoiceType >= 15 andalso VoiceType =< 26)
+ orelse (VoiceType >= 84 andalso VoiceType =< 88),
+ true = lists:member(Appearance.jacket,
+ [16#00011000, 16#00021000, 16#00031000]),
+ true = lists:member(Appearance.pants,
+ [16#00011001, 16#00021001, 16#00031001]),
+ true = lists:member(Appearance.shoes,
+ [16#00091002, 16#00071002, 16#00031002]),
+ validate_new_female_hairstyle(Appearance.hairstyle),
+ true = Appearance.eyelashes =< 12.
+
+validate_new_male_hairstyle(Hairstyle) ->
+ true = lists:member(Hairstyle, [16#00000005, 16#000A0005, 16#00140005,
+ 16#001E0005, 16#00280005, 16#00320005, 16#003C0005, 16#00460005,
+ 16#00500005, 16#005A0005, 16#00640005, 16#006E0005, 16#00780005,
+ 16#00820005, 16#008C0005, 16#00960005, 16#00A00005, 16#00AA0005]).
+
+validate_new_female_hairstyle(Hairstyle) ->
+ true = lists:member(Hairstyle, [16#00001005, 16#000A1005, 16#00141005,
+ 16#001E1005, 16#00281005, 16#00321005, 16#003C1005, 16#00461005,
+ 16#00501005, 16#005A1005, 16#00641005, 16#006E1005, 16#00781005,
+ 16#00821005, 16#008C1005, 16#00961005, 16#00A01005]).
+
+%% Response API.
+%% @todo We need to be able to optionally output commands sent. How?
+
+%% @doc Send the general data and flags for the selected character.
+%% @todo Handle bitflags and value flags properly.
+%% @todo Check that DestTargetID is ffff here as it should be.
+account_character(Char, State=#egs_net{gid=DestGID, targetid=DestTargetID}) ->
+ CharBin = character_to_binary(Char),
+ OptionsBin = character_options_to_binary(Char.options),
+ send(16#0d01, <<
+ DestTargetID?l16, 0:144,
+ 16#00011300:32, DestGID?l32, 0:64,
+ CharBin/binary,
+ 0:8128, %% bit flags followed directly by value flags
+ OptionsBin/binary
+ >>, State).
+
+character_options_to_binary(Opts) ->
+ Brightness = Opts.brightness,
+ ButtonHelp = Opts.buttonhelp,
+ Cam1stX = Opts.cam1stx,
+ Cam1stY = Opts.cam1sty,
+ Cam3rdX = Opts.cam3rdx,
+ Cam3rdY = Opts.cam3rdy,
+ Controller = Opts.controller,
+ CursorPos = Opts.cursorpos,
+ CutIn = Opts.cutin,
+ FnKeys = Opts.fnkeys,
+ LockOn = Opts.lockon,
+ MusicVolume = Opts.musicvolume,
+ RadarMap = Opts.radarmap,
+ SfxVolume = Opts.sfxvolume,
+ Sound = Opts.sound,
+ TextSpeed = Opts.textspeed,
+ Vibration = Opts.vibration,
+ WeaponSwap = Opts.weaponswap,
+ << TextSpeed, Sound, MusicVolume, SfxVolume,
+ Vibration, RadarMap, CutIn, CursorPos,
+ 0, Cam3rdY, Cam3rdX, Cam1stY,
+ Cam1stX, Controller, WeaponSwap, LockOn,
+ Brightness, FnKeys, 0, ButtonHelp >>.
+
+%% @doc Send the character list for the selection screen.
+%% @todo Some values aren't handled yet, like the previous location.
+account_characters_response(Characters, State=#egs_net{gid=DestGID}) ->
+ [Char1, Char2, Char3, Char4] = Characters,
+ Char1Bin = account_character_to_binary(Char1),
+ Char2Bin = account_character_to_binary(Char2),
+ Char3Bin = account_character_to_binary(Char3),
+ Char4Bin = account_character_to_binary(Char4),
+ send(16#0d03, <<
+ 0:32,
+ 16#00011300:32, DestGID?l32, 0:64,
+ 16#00011300:32, DestGID?l32, 0:64,
+ 0:32,
+ Char1Bin/binary, Char2Bin/binary,
+ Char3Bin/binary, Char4Bin/binary
+ >>, State).
+
+%% @todo We shouldn't care about the version *here*, but rather in egs_store.
+account_character_to_binary(notfound) ->
+ << 0:2784 >>;
+account_character_to_binary({1, Char}) ->
+ CharBin = character_to_binary(Char),
+ << 0:8, 1:8, 0:48, CharBin/binary, 0:512 >>.
+
+character_to_binary(notfound) ->
+ << 0:2208 >>;
+character_to_binary(Char) ->
+ Name = Char.name,
+ NameBin = << Name/binary, 0:(512 - bit_size(Name)) >>,
+ Race = atom_to_race(Char.race),
+ Gender = atom_to_gender(Char.gender),
+ Class = atom_to_class(Char.class),
+ AppearanceBin = character_appearance_to_binary(Char.race, Char.appearance),
+ LevelsBin = character_levels_to_binary(Char),
+ << NameBin/binary, Race:8, Gender:8, Class:8,
+ AppearanceBin/binary, LevelsBin/binary >>.
+
+character_levels_to_binary(Char) ->
+ Level = Char.level,
+ BlastBar = Char.blast_bar,
+ Luck = Char.luck,
+ EXP = Char.exp,
+ Money = Char.money,
+ Playtime = Char.playtime,
+ ClassesBin = character_classes_to_binary(Char.type,
+ Char.hunter_level, Char.ranger_level,
+ Char.force_level, Char.acro_level),
+ << Level?l32, BlastBar?l16,
+ Luck:8, 0:40, EXP?l32, 0:32, Money?l32,
+ Playtime?l32, ClassesBin/binary >>.
+
+%% @todo Figure out what the extra values for NPC are.
+character_classes_to_binary(player, HunterLv, RangerLv, ForceLv, AcroLv) ->
+ << 0:160,
+ 1?l32, 1?l32, 1?l32, 1?l32,
+ 1?l32, 1?l32, 1?l32, 1?l32,
+ 1?l32, 1?l32, 1?l32, 1?l32,
+ HunterLv?l32, RangerLv?l32,
+ ForceLv?l32, AcroLv?l32 >>;
+character_classes_to_binary(npc, HunterLv, RangerLv, ForceLv, AcroLv) ->
+ << 1?l32, 1?l32, 1?l32, 1?l32,
+ 1?l32, 1?l32, 1?l32, 1?l32,
+ 1?l32, 1?l32, 1?l32, 1?l32,
+ HunterLv?l32, RangerLv?l32,
+ ForceLv?l32, AcroLv?l32,
+ 16#4e4f4630:32, 16#08000000:32, 0:32, 0:32, 16#4e454e44:32 >>.
+
+character_appearance_to_binary(cast, Appearance) ->
+ Arms = Appearance.arms,
+ BodyColor = Appearance.body_color,
+ Ears = Appearance.ears,
+ Eyebrows = Appearance.eyebrows,
+ Eyelashes = Appearance.eyelashes,
+ EyesColorX = Appearance.eyes_color_x,
+ EyesColorY = Appearance.eyes_color_y,
+ EyesGroup = Appearance.eyes_group,
+ Eyes = Appearance.eyes,
+ Face = Appearance.face,
+ FaceBoxX = Appearance.face_box_x,
+ FaceBoxY = Appearance.face_box_y,
+ HairstyleColorX = Appearance.hairstyle_color_x,
+ HairstyleColorY = Appearance.hairstyle_color_y,
+ HeadType = Appearance.head_type,
+ Legs = Appearance.legs,
+ MainColor = Appearance.main_color,
+ Proportion = Appearance.proportion,
+ ProportionBoxX = Appearance.proportion_box_x,
+ ProportionBoxY = Appearance.proportion_box_y,
+ ShieldColor = Appearance.shield_color,
+ SubColor = Appearance.sub_color,
+ Torso = Appearance.torso,
+ VoicePitch = Appearance.voice_pitch,
+ VoiceType = Appearance.voice_type,
+ << VoiceType:8, VoicePitch:8, 0:24,
+ Torso:32, Legs:32, Arms:32, Ears:32, Face:32, HeadType:32,
+ MainColor:8, 0:16, ShieldColor:8,
+ 0:8, Eyebrows:8, Eyelashes:8, EyesGroup:8, Eyes:8,
+ 0:24, EyesColorY?l32, EyesColorX?l32,
+ 16#ff7f0000:32, 16#ff7f0000:32, 0:32,
+ BodyColor?l32, SubColor?l32,
+ HairstyleColorY?l32, HairstyleColorX?l32,
+ Proportion?l32, ProportionBoxX?l32, ProportionBoxY?l32,
+ FaceBoxX?l32, FaceBoxY?l32 >>;
+character_appearance_to_binary(_, Appearance) ->
+ Badge = Appearance.blast_badge,
+ BodySuit = Appearance.body_suit,
+ Ears = Appearance.ears,
+ Eyebrows = Appearance.eyebrows,
+ Eyelashes = Appearance.eyelashes,
+ EyesColorX = Appearance.eyes_color_x,
+ EyesColorY = Appearance.eyes_color_y,
+ EyesGroup = Appearance.eyes_group,
+ Eyes = Appearance.eyes,
+ Face = Appearance.face,
+ FaceBoxX = Appearance.face_box_x,
+ FaceBoxY = Appearance.face_box_y,
+ Hairstyle = Appearance.hairstyle,
+ HairstyleColorX = Appearance.hairstyle_color_x,
+ HairstyleColorY = Appearance.hairstyle_color_y,
+ Jacket = Appearance.jacket,
+ JacketColor = Appearance.jacket_color,
+ LipsColorX = Appearance.lips_color_x,
+ LipsColorY = Appearance.lips_color_y,
+ LipsIntensity = Appearance.lips_intensity,
+ Pants = Appearance.pants,
+ PantsColor = Appearance.pants_color,
+ Proportion = Appearance.proportion,
+ ProportionBoxX = Appearance.proportion_box_x,
+ ProportionBoxY = Appearance.proportion_box_y,
+ ShieldColor = Appearance.shield_color,
+ Shoes = Appearance.shoes,
+ ShoesColor = Appearance.shoes_color,
+ SkinColor = Appearance.skin_color,
+ VoicePitch = Appearance.voice_pitch,
+ VoiceType = Appearance.voice_type,
+ << VoiceType:8, VoicePitch:8, 0:24,
+ Jacket:32, Pants:32, Shoes:32, Ears:32, Face:32, Hairstyle:32,
+ JacketColor:8, PantsColor:8, ShoesColor:8, ShieldColor:8,
+ Badge:8, Eyebrows:8, Eyelashes:8, EyesGroup:8, Eyes:8,
+ BodySuit:8, 0:16, EyesColorY?l32, EyesColorX?l32,
+ LipsIntensity?l32, LipsColorY?l32, LipsColorX?l32, SkinColor?l32,
+ 16#ffff0200:32, HairstyleColorY?l32, HairstyleColorX?l32,
+ Proportion?l32, ProportionBoxX?l32, ProportionBoxY?l32,
+ FaceBoxX?l32, FaceBoxY?l32 >>.
+
+%% @doc Send the defined account flags for the server.
+account_flags(ValueFlags, BoolFlags, TempFlags, State=#egs_net{gid=DestGID}) ->
+ NbValue = length(ValueFlags),
+ NbBool = length(BoolFlags),
+ NbTemp = length(TempFlags),
+ F = fun(Flag) ->
+ FlagBin = list_to_binary(Flag),
+ Padding = 8 * (16 - byte_size(FlagBin)),
+ << FlagBin/binary, 0:Padding >>
+ end,
+ ValueFlagsBin = iolist_to_binary(lists:map(F, ValueFlags)),
+ BoolFlagsBin = iolist_to_binary(lists:map(F, BoolFlags)),
+ TempFlagsBin = iolist_to_binary(lists:map(F, TempFlags)),
+ send(16#0d05, <<
+ 0:32,
+ 16#00011300:32, DestGID?l32, 0:64,
+ 16#00011300:32, DestGID?l32, 0:64,
+ ValueFlagsBin/binary, BoolFlagsBin/binary, TempFlagsBin/binary,
+ NbValue?l32, NbBool?l32, NbTemp?l32
+ >>, State).
+
+%% @doc Send the player's own partner card.
+comm_own_card(Char, State=#egs_net{gid=DestGID, targetid=DestTargetID}) ->
+ Slot = Char.slot,
+ Name = Char.name,
+ NameBin = << Name/binary, 0:(512 - bit_size(Name)) >>,
+ Race = atom_to_race(Char.race),
+ Gender = atom_to_gender(Char.gender),
+ Class = atom_to_class(Char.class),
+ VoiceType = Char.appearance.voice_type,
+ VoicePitch = Char.appearance.voice_pitch,
+ Comment = Char.card_comment,
+ CommentBin = << Comment/binary, 0:(2816 - bit_size(Comment)) >>,
+ send(16#1500, <<
+ DestTargetID?l32, 0:144,
+ 16#00011300:32, DestGID?l32, 0:64,
+ NameBin/binary,
+ Race:8, Gender:8, Class:8, VoiceType:8, VoicePitch:8, 0:24,
+ DestGID?l32, 0:224,
+ CommentBin/binary,
+ 1:8, 4:8, 1:8, Slot:8,
+ 0:64 >>, State).
+
+%% @doc Display an error to the client.
+system_auth_error(Error, State=#egs_net{gid=DestGID}) ->
+ Length = byte_size(Error) div 2 + 2,
+ send(16#0223, <<
+ 0:160,
+ 16#00000f00:32, DestGID?l32, 0:64,
+ 0:64,
+ 3?l32, 0:48, Length?l16,
+ Error/binary, 0:16
+ >>, State).
+
+%% @doc Send the game server's IP and port that the client will connect to.
+%% @todo Take IP as a list, not a binary.
+system_game_server_response(ServerIP, ServerPort,
+ State=#egs_net{gid=DestGID, targetid=DestTargetID}) ->
+ send(16#0216, <<
+ DestTargetID?l16, 0:16,
+ 0:128,
+ 16#00000f00:32, DestGID?l32, 0:64,
+ ServerIP/binary, ServerPort?l16, 0:16
+ >>, State).
+
+%% @doc Say hello to a newly connected client.
+system_hello(State=#egs_net{gid=DestGID, targetid=DestTargetID}) ->
+ send(16#0202, << DestTargetID?l16, 0:272, DestGID?l32, 0:1024 >>, State).
+
+%% @doc Send the authentication information for key-based authentication.
+system_key_auth_info(AuthGID, AuthKey, State=#egs_net{gid=DestGID}) ->
+ send(16#0223, <<
+ 0:160,
+ 16#00000f00:32, DestGID?l32, 0:64,
+ AuthGID?l32, AuthKey/binary
+ >>, State).
+
+%% @doc Send the given MOTD page to the client for display.
+%%
+%% The full MOTD is expected as this function will only take the
+%% page it needs, automatically.
+system_motd_response(MOTD, Page, State=#egs_net{targetid=DestTargetID}) ->
+ Lines = re:split(MOTD, "\n\\0"),
+ NbPages = 1 + length(Lines) div 15,
+ true = Page >= 0,
+ true = Page < NbPages,
+ Text = << << Line/binary, "\n", 0 >>
+ || Line <- lists:sublist(Lines, 1 + Page * 15, 15) >>,
+ Length = byte_size(Text) div 2 + 2,
+ send(16#0225, <<
+ DestTargetID?l16, 0:272,
+ NbPages:8, Page:8, Length?l16,
+ Text/binary, 0:16
+ >>, State).
+
+%% @doc Make the client open the given URL in a browser, after the game closes.
+%% @todo Take URL as a list, not a binary.
+system_open_url(URL, State=#egs_net{gid=DestGID, targetid=DestTargetID}) ->
+ Length = byte_size(URL) + 1,
+ Padding = 8 * (512 - Length - 1),
+ send(16#0231, <<
+ DestTargetID?l16, 0:16,
+ 16#00000f00:32, DestGID?l32, 0:64,
+ 16#00000f00:32, DestGID?l32, 0:64,
+ Length?l32, URL/binary, 0:Padding
+ >>, State).
+
+%% Response primitives.
+
+send(Command, Data, State) ->
+ send(Command, 3, Data, State).
+
+%% @todo We may also optionally output packets sent here.
+send(Command, Channel, Data, State) ->
+ Size = 8 + byte_size(Data),
+ {Size2, Padding} = case Size rem 4 of
+ 0 -> {Size, <<>>};
+ 2 -> {Size + 2, << 0:16 >>}
+ end,
+ send_packet(<< Size2?l32, Command:16, Channel:8, 0:8,
+ Data/binary, Padding/binary >>, State).
+
+send_packet(Packet, #egs_net{socket=Socket, transport=Transport})
+ when byte_size(Packet) =< 16#4000 ->
+ Transport:send(Socket, Packet);
+send_packet(Packet, State) ->
+ send_fragments(Packet, byte_size(Packet), 0, State).
+
+send_fragments(Packet, Size, Current, #egs_net{
+ socket=Socket, transport=Transport})
+ when Size - Current =< 16#4000 ->
+ FragmentSize = 16#10 + byte_size(Packet),
+ Transport:send(Socket, << FragmentSize?l32, 16#0b030000:32,
+ Size?l32, Current?l32, Packet/binary >>);
+send_fragments(Packet, Size, Current, State=#egs_net{
+ socket=Socket, transport=Transport}) ->
+ << Fragment:16#4000/binary, Rest/binary >> = Packet,
+ Transport:send(Socket, << 16#10400000:32, 16#0b030000:32,
+ Size?l32, Current?l32, Fragment/binary >>),
+ send_fragments(Rest, Size, Current + 16#4000, State).
+
+%% Data conversion.
+
+atom_to_class(hunter) -> 12;
+atom_to_class(ranger) -> 13;
+atom_to_class(force ) -> 14;
+atom_to_class(acro ) -> 15.
+
+atom_to_gender(male ) -> 0;
+atom_to_gender(female) -> 1.
+
+atom_to_race(human ) -> 0;
+atom_to_race(newman) -> 1;
+atom_to_race(cast ) -> 2;
+atom_to_race(beast ) -> 3.
+
+character_eventid_to_atom( 1) -> unknown; %% @todo
+character_eventid_to_atom( 2) -> character_type_capabilities_request;
+character_eventid_to_atom( 3) -> character_type_change;
+character_eventid_to_atom( 4) -> unknown; %% @todo
+character_eventid_to_atom( 6) -> unknown; %% @todo
+character_eventid_to_atom( 7) -> character_death;
+character_eventid_to_atom( 8) -> character_death_return_to_lobby;
+character_eventid_to_atom( 9) -> unknown; %% @todo
+character_eventid_to_atom(10) -> character_status_change.
+
+chat_type_to_atom(0) -> speak;
+chat_type_to_atom(1) -> shout;
+chat_type_to_atom(2) -> whisper.
+
+chat_cutin_to_atom( 0) -> none;
+chat_cutin_to_atom( 1) -> laugh;
+chat_cutin_to_atom( 2) -> smile;
+chat_cutin_to_atom( 3) -> wry_smile;
+chat_cutin_to_atom( 4) -> surprised;
+chat_cutin_to_atom( 5) -> confused;
+chat_cutin_to_atom( 6) -> disappointed;
+chat_cutin_to_atom( 7) -> deep_in_thought;
+chat_cutin_to_atom( 8) -> sneer;
+chat_cutin_to_atom( 9) -> dissatisfied;
+chat_cutin_to_atom(10) -> angry.
+
+chat_channel_to_atom(0) -> public;
+chat_channel_to_atom(1) -> private.
+
+chat_character_type_to_atom(0) -> player;
+chat_character_type_to_atom(2) -> apc. %% @todo Check that this is right.
+
+class_to_atom(12) -> hunter;
+class_to_atom(13) -> ranger;
+class_to_atom(14) -> force;
+class_to_atom(15) -> acro.
+
+dialog_eventid_to_atom(0) -> npc_shop_request;
+dialog_eventid_to_atom(2) -> lumilass_options_request;
+dialog_eventid_to_atom(3) -> ppcube_request;
+dialog_eventid_to_atom(4) -> ppcube_charge_all;
+dialog_eventid_to_atom(5) -> ppcube_charge_one;
+dialog_eventid_to_atom(6) -> put_on_outfit;
+dialog_eventid_to_atom(7) -> remove_outfit;
+dialog_eventid_to_atom(9) -> player_type_availability_request.
+
+gender_to_atom(0) -> male;
+gender_to_atom(1) -> female.
+
+item_eventid_to_atom( 1) -> item_equip;
+item_eventid_to_atom( 2) -> item_unequip;
+item_eventid_to_atom( 3) -> item_link_pa;
+item_eventid_to_atom( 4) -> item_unlink_pa;
+item_eventid_to_atom( 5) -> item_drop;
+item_eventid_to_atom( 7) -> item_learn_pa;
+item_eventid_to_atom( 8) -> item_use;
+item_eventid_to_atom( 9) -> item_set_trap;
+item_eventid_to_atom(18) -> item_unlearn_pa.
+
+language_to_atom(0) -> japanese;
+language_to_atom(1) -> american_english;
+language_to_atom(2) -> british_english;
+language_to_atom(3) -> french;
+language_to_atom(4) -> german;
+language_to_atom(5) -> spanish;
+language_to_atom(6) -> italian;
+language_to_atom(7) -> korean;
+language_to_atom(8) -> simplified_chinese;
+language_to_atom(9) -> traditional_chinese.
+
+npc_shop_eventid_to_atom(1) -> npc_shop_enter;
+npc_shop_eventid_to_atom(2) -> npc_shop_buy;
+npc_shop_eventid_to_atom(3) -> npc_shop_sell;
+npc_shop_eventid_to_atom(4) -> unknown; %% @todo npc_shop_gift_wrap
+npc_shop_eventid_to_atom(5) -> npc_shop_leave;
+npc_shop_eventid_to_atom(6) -> unknown. %% @todo
+
+platform_to_atom(0) -> ps2;
+platform_to_atom(1) -> pc.
+
+race_to_atom(0) -> human;
+race_to_atom(1) -> newman;
+race_to_atom(2) -> cast;
+race_to_atom(3) -> beast.
+
+%% Debug.
+
+binary_to_dump(Data) ->
+ binary_to_dump(Data, 4, []).
+binary_to_dump(<<>>, _, Acc) ->
+ lists:reverse(Acc);
+binary_to_dump(Data, 0, Acc) ->
+ binary_to_dump(Data, 4, ["~n"|Acc]);
+binary_to_dump(<< A, B, C, D, Rest/binary >>, N, Acc) ->
+ Str = io_lib:format("~2.16.0b ~2.16.0b ~2.16.0b ~2.16.0b ", [A, B, C, D]),
+ binary_to_dump(Rest, N - 1, [Str|Acc]).
diff --git a/apps/egs_store/priv/.gitignore b/apps/egs_store/priv/.gitignore
new file mode 100644
index 0000000..e69de29
diff --git a/apps/egs_store/src/egs_store.app.src b/apps/egs_store/src/egs_store.app.src
new file mode 100644
index 0000000..4b9e5b2
--- /dev/null
+++ b/apps/egs_store/src/egs_store.app.src
@@ -0,0 +1,13 @@
+%%-*- mode: erlang -*-
+{application, egs_store, [
+ {description, "EGS storage layer."},
+ {vsn, "0.1.0"},
+ {modules, []},
+ {registered, []},
+ {applications, [
+ kernel,
+ stdlib
+ ]},
+ {mod, {egs_store_app, []}},
+ {env, []}
+]}.
diff --git a/apps/egs_store/src/egs_store.erl b/apps/egs_store/src/egs_store.erl
new file mode 100644
index 0000000..24447f4
--- /dev/null
+++ b/apps/egs_store/src/egs_store.erl
@@ -0,0 +1,92 @@
+%% This file is part of EGS.
+%%
+%% EGS is free software: you can redistribute it and/or modify
+%% it under the terms of the GNU Affero 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 Affero General Public License for more details.
+%%
+%% You should have received a copy of the GNU Affero General Public License
+%% along with EGS. If not, see .
+
+-module(egs_store).
+-behaviour(gen_server).
+
+-export([start_link/0,
+ load_character/2, load_characters/2, save_character/3]). %% API.
+-export([init/1, handle_call/3, handle_cast/2,
+ handle_info/2, terminate/2, code_change/3]). %% gen_server.
+
+-define(SERVER, ?MODULE).
+-define(ACCOUNTS_TBL, accounts_tbl).
+-define(ACCOUNTS_VSN, 1).
+-define(CHARACTERS_TBL, characters_tbl).
+-define(CHARACTERS_VSN, 1).
+
+%% API.
+
+start_link() ->
+ gen_server:start_link({local, ?SERVER}, ?MODULE, [], []).
+
+load_character(GID, Slot) ->
+ gen_server:call(?SERVER, {load_character, GID, Slot}).
+
+load_characters(GID, Slots) ->
+ gen_server:call(?SERVER, {load_characters, GID, Slots}).
+
+save_character(GID, Slot, Data) ->
+ gen_server:call(?SERVER, {save_character, GID, Slot, Data}).
+
+%% gen_server.
+
+init([]) ->
+ {ok, App} = application:get_application(),
+ PrivDir = code:priv_dir(App),
+ AccountsFile = PrivDir ++ "/accounts.tbl",
+ CharactersFile = PrivDir ++ "/characters.tbl",
+ {ok, ?ACCOUNTS_TBL} = dets:open_file(?ACCOUNTS_TBL,
+ [{file, AccountsFile}]),
+ io:format("accounts tbl:~n~p~n~n", [dets:info(?ACCOUNTS_TBL)]),
+ {ok, ?CHARACTERS_TBL} = dets:open_file(?CHARACTERS_TBL,
+ [{file, CharactersFile}]),
+ io:format("characters tbl:~n~p~n~n", [dets:info(?CHARACTERS_TBL)]),
+ {ok, undefined}.
+
+handle_call({load_character, GID, Slot}, _From, State) ->
+ case dets:lookup(?CHARACTERS_TBL, {GID, Slot}) of
+ [{{GID, Slot}, Version, Data}] ->
+ {reply, {ok, Version, Data}, State};
+ [] ->
+ {reply, {error, notfound}, State}
+ end;
+handle_call({load_characters, GID, Slots}, _From, State) ->
+ Characters = lists:map(fun(Slot) ->
+ case dets:lookup(?CHARACTERS_TBL, {GID, Slot}) of
+ [{{GID, Slot}, Version, Data}] ->
+ {Version, Data};
+ [] ->
+ notfound
+ end
+ end, Slots),
+ {reply, {ok, Characters}, State};
+handle_call({save_character, GID, Slot, Data}, _From, State) ->
+ ok = dets:insert(?CHARACTERS_TBL, {{GID, Slot}, ?CHARACTERS_VSN, Data}),
+ {reply, ok, State};
+handle_call(_Request, _From, State) ->
+ {reply, ignored, State}.
+
+handle_cast(_Msg, State) ->
+ {noreply, State}.
+
+handle_info(_Info, State) ->
+ {noreply, State}.
+
+terminate(_Reason, _State) ->
+ ok.
+
+code_change(_OldVsn, State, _Extra) ->
+ {ok, State}.
diff --git a/apps/egs_store/src/egs_store_app.erl b/apps/egs_store/src/egs_store_app.erl
new file mode 100644
index 0000000..116eea9
--- /dev/null
+++ b/apps/egs_store/src/egs_store_app.erl
@@ -0,0 +1,31 @@
+%% This file is part of EGS.
+%%
+%% EGS is free software: you can redistribute it and/or modify
+%% it under the terms of the GNU Affero 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 Affero General Public License for more details.
+%%
+%% You should have received a copy of the GNU Affero General Public License
+%% along with EGS. If not, see .
+
+-module(egs_store_app).
+-behaviour(application).
+-export([start/2, stop/1]). %% API.
+
+-type application_start_type()
+ :: normal | {takeover, node()} | {failover, node()}.
+
+%% API.
+
+-spec start(application_start_type(), any()) -> {ok, pid()}.
+start(_Type, _StartArgs) ->
+ egs_store_sup:start_link().
+
+-spec stop(any()) -> ok.
+stop(_State) ->
+ ok.
diff --git a/apps/egs_store/src/egs_store_sup.erl b/apps/egs_store/src/egs_store_sup.erl
new file mode 100644
index 0000000..def623d
--- /dev/null
+++ b/apps/egs_store/src/egs_store_sup.erl
@@ -0,0 +1,31 @@
+%% This file is part of EGS.
+%%
+%% EGS is free software: you can redistribute it and/or modify
+%% it under the terms of the GNU Affero 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 Affero General Public License for more details.
+%%
+%% You should have received a copy of the GNU Affero General Public License
+%% along with EGS. If not, see .
+
+-module(egs_store_sup).
+-behaviour(supervisor).
+
+-export([start_link/0]). %% API.
+-export([init/1]). %% Supervisor.
+
+-spec start_link() -> {ok, pid()}.
+start_link() ->
+ supervisor:start_link({local, ?MODULE}, ?MODULE, []).
+
+-spec init([]) -> {ok, {{one_for_one, 10, 10}, [supervisor:child_spec(), ...]}}.
+init([]) ->
+ {ok, {{one_for_one, 10, 10}, [
+ {egs_store, {egs_store, start_link, []},
+ permanent, 5000, worker, [egs_store]}
+ ]}}.
diff --git a/p/flags.bin b/p/flags.bin
deleted file mode 100644
index 6024875..0000000
Binary files a/p/flags.bin and /dev/null differ
diff --git a/priv/egs.conf b/priv/egs.conf
index 3d0b642..575bf72 100644
--- a/priv/egs.conf
+++ b/priv/egs.conf
@@ -27,9 +27,26 @@
%% @doc Game server IP address and port.
%% They can be modified freely without problem.
%% Note that the port should be available and above 1024.
-{game_server, {<< 127, 0, 0, 1 >>, 12061}}.
+{game_server, {<< 91, 121, 75, 204 >>, 12061}}.
%% Caps and limitations.
%% @doc Maximum level players can reach.
{level_cap, 200}.
+
+%% Flags.
+
+%% @todo doc
+{value_flags, [
+ "EGS_VALUE_FLAG"
+]}.
+
+%% @todo doc
+{bool_flags, [
+ "EGS_BOOL_FLAG"
+]}.
+
+%% @todo doc
+{temp_flags, [
+ "EGS_TEMP_FLAG"
+]}.
diff --git a/rebar.config b/rebar.config
index 5c2777a..70e4cc0 100644
--- a/rebar.config
+++ b/rebar.config
@@ -1,6 +1,8 @@
{sub_dirs, [
"apps/egs",
+ "apps/egs_net",
"apps/egs_patch",
+ "apps/egs_store",
"apps/prs"
]}.
{dialyzer_opts, [src, {warnings, [