From ceea04c3b438cee9a8470f2abc6297429bf347c5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Lo=C3=AFc=20Hoguin?= Date: Wed, 16 May 2012 12:33:12 +0200 Subject: [PATCH] Work in progress on egs_net (will be amended) --- apps/egs/include/records.hrl | 10 +- apps/egs/rebar.config | 2 + apps/egs/src/egs.erl | 1 - apps/egs/src/egs_app.erl | 1 + apps/egs/src/egs_char.erl | 64 ++ apps/egs/src/egs_char_select.erl | 96 +- apps/egs/src/egs_game.erl | 132 ++- apps/egs/src/egs_game_protocol.erl | 14 +- apps/egs/src/egs_login.erl | 57 +- apps/egs/src/egs_login_protocol.erl | 8 +- apps/egs/src/egs_network.erl | 86 -- apps/egs/src/egs_proto.erl | 1398 ++--------------------- apps/egs/src/egs_script_lexer.erl | 6 +- apps/egs/src/egs_script_parser.erl | 13 +- apps/egs/src/psu/psu_appearance.erl | 254 ----- apps/egs/src/psu/psu_characters.erl | 92 +- apps/egs_net/rebar.config | 4 + apps/egs_net/src/egs_net.app.src | 12 + apps/egs_net/src/egs_net.erl | 1576 ++++++++++++++++++++++++++ apps/egs_store/priv/.gitignore | 0 apps/egs_store/src/egs_store.app.src | 13 + apps/egs_store/src/egs_store.erl | 92 ++ apps/egs_store/src/egs_store_app.erl | 31 + apps/egs_store/src/egs_store_sup.erl | 31 + p/flags.bin | Bin 20060 -> 0 bytes priv/egs.conf | 19 +- rebar.config | 2 + 27 files changed, 2079 insertions(+), 1935 deletions(-) create mode 100644 apps/egs/src/egs_char.erl delete mode 100644 apps/egs/src/egs_network.erl delete mode 100644 apps/egs/src/psu/psu_appearance.erl create mode 100644 apps/egs_net/rebar.config create mode 100644 apps/egs_net/src/egs_net.app.src create mode 100644 apps/egs_net/src/egs_net.erl create mode 100644 apps/egs_store/priv/.gitignore create mode 100644 apps/egs_store/src/egs_store.app.src create mode 100644 apps/egs_store/src/egs_store.erl create mode 100644 apps/egs_store/src/egs_store_app.erl create mode 100644 apps/egs_store/src/egs_store_sup.erl delete mode 100644 p/flags.bin 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 6024875823554ca307ccfb17544bfe8a54bb105c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 20060 zcmeI)OLF5j5XNy;NC3hb>ml>Uc9e9Bqzd~fP8><@AMz3#9>yX>oFzO{`EuKXCEoNMO^AKd!UY5nN5eso&@ z=+=);>qn>cqji0$$78IA9$=gMq3N_fEU&xf9_kZ)3Y|8dhWl;aUB_BVF@4;IrTc7k z{U^8m=(PRlwcgVt$6L7e#N*9a>unr%pPg~oeXPUoV;y!M>#+M+huz0I>^{5uX=(ef zM)vt>`QiN2`ImDjxY-}V=||1@Va6l4`QuYc_u1X%<-Msd*7X?bxh+5Sm!H;6$aiaO zwKuQT-n>>@&xiM0dz%|MS;^sFeo6&O1xf`<1^y=$n7(B8`nC-9G%bT%Kdd{sU|F{^ zW3Yag8IpDHeCgBj^E}GzciW!YtgklftHb&-ALsL&&KIr21>ZVcAFaa$-#T3It-}T1 zI$ZFr!v)_uT=1>Ke9^cHpD?XmzRKmRT)xWXt6aXy<*Qu2%H^v3NBZk)d}zlHgZ$_ zfA%u}G|Ml2zWurBdb|96Uf!&G=SwfkXT7s9E@Pa38teSiI1IOKuk%l{JPh}C{~~;b zls!D%Us0R;DQa^+MQ!eiv@N&+rL4e^2zkj9js}x|{kSg6V}DXUOg>CL zOg>CLOg>CLOg>D$SkIIX>+U6aG5Ik0F!?a~F!?a~F!?a~p4@mpJHI%`_NhOZ{$Top z=?|tqnEqh;gXs^ZKbZc+{3#zMA0{6rA0{6rA0{6rA0{6rU#w@!hslS@hslS@hslS@ zhslS@hspQqj(1ExOg>CLOg>CLOg>CLOg^otbl`duzi&vUUM#0xn0jIAg{c>& zUYL4e>V+B4=u{7`Kjg*a#pK20#pK20#pK20#pK20jZS&F9+4N57n2v07n2v07n2v0 z7n2v0H#+6zdP81JUQAw0UQAw0UQAw0UQAw0-sqH<>koM`c`lNXa0lNXa0lNXa0lNXa0lQ){Y z5AOPz9P=i}yvZ?da?G0?^Cri<$uVzo%$pqZ;+VG?INz=H-Mr@Ayyo4!=H0yJ-Mr@A zyyo4!=H0yJ-Mr>a*1H9}UHi3uI={;8kxo8?uDW zosU*~JD=UoH`=}3#Q8Fwe>VH&+I`dBozL}drObEJ_T2eg?^fF7?tHFyEA8@lel-0L ziDZ9!wf(VPmsi>ReH-t!gT~syp{)-2_BO8-kY8GUN(D*zA{`9nq`boC@Jj|qQKEPw8-{rho#ue&#SG0dP2O|Iju>GTCo+35Xy{3)CG>$lzR Hclqz%SEW=0 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, [