diff --git a/ebin/egs.app b/ebin/egs.app index a4d1845..82c7951 100644 --- a/ebin/egs.app +++ b/ebin/egs.app @@ -8,6 +8,10 @@ egs_sup, egs_exit_mon, egs_user_model, + egs_network, + egs_login, + egs_char_select, + egs_game, reloader, psu_game, psu_login, diff --git a/src/egs_char_select.erl b/src/egs_char_select.erl new file mode 100644 index 0000000..defa911 --- /dev/null +++ b/src/egs_char_select.erl @@ -0,0 +1,107 @@ +%% @author Loïc Hoguin +%% @copyright 2010 Loïc Hoguin. +%% @doc Game server's character selection callback module. +%% +%% 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_char_select). +-export([keepalive/1, info/2, cast/3, raw/3, event/2]). + +%% @todo These headers are only included because of egs_user_model and items. We don't want that here. +-include("include/records.hrl"). +-include("include/psu/items.hrl"). + +-record(state, {socket, gid}). + +%% @doc Send a keepalive. +keepalive(#state{socket=Socket}) -> + psu_proto:send_keepalive(Socket). + +%% @doc We don't expect any message here. +%% @todo Throw an error instead? +info(_Msg, _State) -> + ok. + +%% @doc Nothing to broadcast. +%% @todo Throw an error instead? +cast(_Command, _Data, _State) -> + ok. + +%% @doc Dismiss all raw commands with a log notice. +%% @todo Have a log event handler instead. +raw(Command, _Data, State) -> + io:format("~p (~p): dismissed command ~4.16.0b~n", [?MODULE, State#state.gid, Command]). + +%% Events. + +%% @doc Character screen selection request and delivery. +event(char_select_request, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + psu_game:send_0d03(data_load(User#egs_user_model.folder, 0), data_load(User#egs_user_model.folder, 1), data_load(User#egs_user_model.folder, 2), data_load(User#egs_user_model.folder, 3)); + +%% @doc The options default to 0 for everything except brightness to 4. +%% @todo Don't forget to check for the character's name. +event({char_select_create, Slot, CharBin}, #state{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_characters:validate_name(Name), + %~ psu_appearance:validate_char_create(Race, Gender, Appearance), + %% end of check, continue doing it wrong past that point for now + {ok, User} = egs_user_model:read(GID), + Dir = io_lib:format("save/~s", [User#egs_user_model.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 >>); + +%% @doc Load the selected character into the game's universe. +%% @todo The area_change should happen only after we received 021c back from the client. +event({char_select_enter, Slot, _BackToPreviousField}, #state{socket=Socket, gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + [{status, 1}, {char, CharBin}, {options, OptionsBin}] = data_load(User#egs_user_model.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), + Character = #characters{slot=Slot, name=Name, race=Race, gender=Gender, class=Class, appearance=Appearance, options=Options, % TODO: temporary set the slot here, won't be needed later + inventory= [{16#11010000, #psu_special_item_variables{}}, {16#11020000, #psu_special_item_variables{}}, {16#11020100, #psu_special_item_variables{}}, {16#11020200, #psu_special_item_variables{}}, + {16#01010900, #psu_striking_weapon_item_variables{is_active=0, slot=0, current_pp=99, max_pp=100, element=#psu_element{type=1, percent=50}, pa=#psu_pa{type=0, level=0}}}, + {16#01010a00, #psu_striking_weapon_item_variables{is_active=0, slot=0, current_pp=99, max_pp=100, element=#psu_element{type=2, percent=50}, pa=#psu_pa{type=0, level=0}}}, + {16#01010b00, #psu_striking_weapon_item_variables{is_active=0, slot=0, current_pp=99, max_pp=100, element=#psu_element{type=3, percent=50}, pa=#psu_pa{type=0, level=0}}}]}, + User2 = User#egs_user_model{state=online, character=Character, area=#psu_area{questid=1100000, zoneid=0, mapid=4}, entryid=5, + prev_area={psu_area, 0, 0, 0}, prev_entryid=0, pos=#pos{x=0.0, y=0.0, z=0.0, dir=0.0}, setid=0}, + egs_user_model:write(User2), + psu_game:char_load(User2), + {ok, egs_game, {state, Socket, GID}}. + +%% Internal. + +%% @doc Load the given character's data. +%% @todo This function is temporary until we get permanent mnesia 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. diff --git a/src/egs_game.erl b/src/egs_game.erl new file mode 100644 index 0000000..e7d77fb --- /dev/null +++ b/src/egs_game.erl @@ -0,0 +1,740 @@ +%% @author Loïc Hoguin +%% @copyright 2010 Loïc Hoguin. +%% @doc Game server's client authentication callback module. +%% +%% 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_game). +-export([keepalive/1, info/2, cast/3, raw/3, event/2]). + +%% @todo This header is probably only included because of egs_user_model. We don't want that here. +-include("include/records.hrl"). +-include("include/maps.hrl"). +-include("include/psu/items.hrl"). + +-record(state, {socket, gid}). + +%% @doc Send a keepalive. +keepalive(#state{socket=Socket}) -> + psu_proto:send_keepalive(Socket). + +%% @doc Forward the broadcasted command to the client. +info({egs, cast, Command}, #state{gid=GID}) -> + << A:64/bits, _:32, B:96/bits, _:64, C/bits >> = Command, + psu_game:send(<< A/binary, 16#00011300:32, B/binary, 16#00011300:32, GID:32/little-unsigned-integer, C/binary >>); + +%% @doc Forward the chat message to the client. +info({egs, chat, ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage}, _State) -> + psu_game:send_0304(ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage); + +%% @doc Inform the client that a player has spawn. +%% @todo Should be something along the lines of 010d 0205 203 201. +info({egs, player_spawn, _Player}, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + {ok, SpawnList} = egs_user_model:select({neighbors, User}), + psu_game:send_0233(SpawnList); + +%% @doc Inform the client that a player has unspawn. +info({egs, player_unspawn, Player}, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + psu_game:send_0204(User, Player, 5); + +%% @doc Warp the player to the given location. +info({egs, warp, QuestID, ZoneID, MapID, EntryID}, State) -> + event({area_change, QuestID, ZoneID, MapID, EntryID}, State). + +%% Broadcasts. + +%% @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, State=#state{gid=GID}) -> + << _:424, Dir:24/little-unsigned-integer, _PrevCoords:96, X:32/little-float, Y:32/little-float, Z:32/little-float, + QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer, _:32 >> = Data, + FloatDir = Dir / 46603.375, + {ok, User} = egs_user_model:read(GID), + NewUser = User#egs_user_model{pos=#pos{x=X, y=Y, z=Z, dir=FloatDir}, area=#psu_area{questid=QuestID, zoneid=ZoneID, mapid=MapID}, entryid=EntryID}, + egs_user_model:write(NewUser), + cast(valid, Data, State); + +%% @doc Stand still. Save the position and then dispatch it. +cast(16#0514, Data, State=#state{gid=GID}) -> + << _:424, Dir:24/little-unsigned-integer, X:32/little-float, Y:32/little-float, Z:32/little-float, + QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, + MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer, _/bits >> = Data, + FloatDir = Dir / 46603.375, + {ok, User} = egs_user_model:read(GID), + NewUser = User#egs_user_model{pos=#pos{x=X, y=Y, z=Z, dir=FloatDir}, area=#psu_area{questid=QuestID, zoneid=ZoneID, mapid=MapID}, entryid=EntryID}, + egs_user_model:write(NewUser), + cast(valid, Data, State); + +%% @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 State. +cast(Command, Data, #state{gid=GID}) + when Command =:= 16#0101; + Command =:= 16#0102; + Command =:= 16#0104; + Command =:= 16#0107; + Command =:= 16#010f; + Command =:= 16#050f; + Command =:= valid -> + << _:32, A:64/bits, _:64, B:192/bits, _:64, C/bits >> = Data, + case egs_user_model:read(GID) of + {error, _Reason} -> + ignore; + {ok, Self} -> + LID = Self#egs_user_model.lid, + Packet = << A/binary, 16#00011300:32, GID:32/little-unsigned-integer, B/binary, + GID:32/little-unsigned-integer, LID:32/little-unsigned-integer, C/binary >>, + {ok, SpawnList} = egs_user_model:select({neighbors, Self}), + lists:foreach(fun(User) -> User#egs_user_model.pid ! {egs, cast, Packet} end, SpawnList) + end. + +%% Raw commands. + +%% @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 >>, #state{gid=GID}) -> + << SpawnID:32/little-unsigned-integer, _:64, Type:32/little-unsigned-integer, _:64 >> = Data, + case Type of + 7 -> % spawn cleared @todo 1201 sent back with same values apparently, but not always + log("cleared spawn ~b", [SpawnID]), + {ok, User} = egs_user_model:read(GID), + {BlockID, EventID} = psu_instance:spawn_cleared_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, SpawnID), + if EventID =:= false -> ignore; + true -> psu_game:send_1205(EventID, BlockID, 0) + end; + _ -> + ignore + end; + +%% @todo Handle this packet. +%% @todo 3rd Unsafe Passage C, EventID 10 BlockID 2 = mission cleared? +raw(16#0404, << _:352, Data/bits >>, _State) -> + << EventID:8, BlockID:8, _:16, Value:8, _/bits >> = Data, + log("unknown command 0404: eventid ~b blockid ~b value ~b", [EventID, BlockID, Value]), + psu_game:send_1205(EventID, BlockID, Value); + +%% @todo Used in the tutorial. Not sure what it does. Give an item (the PA) maybe? +raw(16#0a09, _Data, #state{gid=GID}) -> + psu_game:send(<< 16#0a090300:32, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00003300:32, 0:32 >>); + +%% @todo Figure out this command. +raw(16#0c11, << _:352, A:32/little, B:32/little >>, #state{gid=GID}) -> + log("0c11 ~p ~p", [A, B]), + psu_game:send(<< 16#0c120300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 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 >>, #state{gid=GID}) -> + << Flag:128/bits, A:16/bits, _:8, B/bits >> = Data, + log("flag handler for ~s", [re:replace(Flag, "\\0+", "", [global, {return, binary}])]), + psu_game:send(<< 16#0d040300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, Flag/binary, A/binary, 1, B/binary >>); + +%% @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 >>, _State) -> + << A:32/little-unsigned-integer, 0:16, B:16/little-unsigned-integer, 0:16, C:16/little-unsigned-integer, 0, Whut:8, D:16/little-unsigned-integer, 0:16, + E:16/little-unsigned-integer, 0:16, F:16/little-unsigned-integer, G:16/little-unsigned-integer, H:16/little-unsigned-integer, I:32/little-unsigned-integer >> = Data, + log("init vehicle: ~b ~b ~b ~b ~b ~b ~b ~b ~b ~b", [A, B, C, Whut, D, E, F, G, H, I]), + psu_game:send(<< (psu_game:header(16#1208))/binary, A:32/little-unsigned-integer, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, + 0:16, B:16/little-unsigned-integer, 0:16, C:16/little-unsigned-integer, 0:16, D:16/little-unsigned-integer, 0:112, + E:16/little-unsigned-integer, 0:16, F:16/little-unsigned-integer, H:16/little-unsigned-integer, 1, 0, 100, 0, 10, 0, G:16/little-unsigned-integer, 0:16 >>); + +%% @doc Enter vehicle. +%% @todo Separate the reply. +raw(16#0f02, << _:352, Data/bits >>, _State) -> + << A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer >> = Data, + log("enter vehicle: ~b ~b ~b", [A, B, C]), + HP = 100, + psu_game:send(<< (psu_game:header(16#120a))/binary, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer, HP:32/little-unsigned-integer >>); + +%% @doc Sent right after entering the vehicle. Can't move without it. +%% @todo Separate the reply. +raw(16#0f07, << _:352, Data/bits >>, _State) -> + << A:32/little-unsigned-integer, B:32/little-unsigned-integer >> = Data, + log("after enter vehicle: ~b ~b", [A, B]), + psu_game:send(<< (psu_game:header(16#120f))/binary, A:32/little-unsigned-integer, B:32/little-unsigned-integer >>); + +%% @todo Not sure yet. +raw(16#1019, _Data, _State) -> + ignore; + %~ psu_game:send(<< (psu_game:header(16#1019))/binary, 0:192, 16#00200000:32, 0:32 >>); + +%% @todo Not sure about that one though. Probably related to 1112 still. +raw(16#1106, << _:352, Data/bits >>, _State) -> + psu_game:send_110e(Data); + +%% @doc Probably asking permission to start the video (used for syncing?). +raw(16#1112, << _:352, Data/bits >>, _State) -> + psu_game:send_1113(Data); + +%% @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 >>, _State) -> + << Value:32/little-unsigned-integer >> = Data, + log("command 1216 with value ~b", [Value]), + psu_game:send_1216(Value); + +%% @doc Dismiss all unknown raw commands with a log notice. +%% @todo Have a log event handler instead. +raw(Command, _Data, State) -> + io:format("~p (~p): dismissed command ~4.16.0b~n", [?MODULE, State#state.gid, Command]). + +%% Events. + +%% @todo When changing lobby to the room, 0230 must also be sent. Same when going from room to lobby. +%% @todo Probably move area_load inside the event and make other events call this one when needed. +event({area_change, QuestID, ZoneID, MapID, EntryID}, State) -> + event({area_change, QuestID, ZoneID, MapID, EntryID, 16#ffffffff}, State); +event({area_change, QuestID, ZoneID, MapID, EntryID, PartyPos}, _State) -> + case PartyPos of + 16#ffffffff -> + log("area change (~b,~b,~b,~b,~b)", [QuestID, ZoneID, MapID, EntryID, PartyPos]), + psu_game:area_load(QuestID, ZoneID, MapID, EntryID); + _Any -> %% @todo Handle area_change event for NPCs in story missions. + ignore + end; + +%% @doc After the character has been (re)loaded, change the area he's in. +%% @todo The area_load function should probably not change the user's values. +%% @todo Remove that ugly code when the above is done. +event(char_load_complete, State=#state{gid=GID}) -> + {ok, User=#egs_user_model{area=#psu_area{questid=QuestID, zoneid=ZoneID, mapid=MapID}, + entryid=EntryID}} = egs_user_model:read(GID), + egs_user_model:write(User#egs_user_model{area=#psu_area{questid=0, zoneid=0, mapid=0}, entryid=0}), + event({area_change, QuestID, ZoneID, MapID, EntryID}, State); + +%% @doc Chat broadcast handler. Dispatch the message to everyone (for now). +%% Disregard the name sent by the server. Use the name saved in memory instead, to prevent client-side editing. +%% @todo Only broadcast to people in the same map. +%% @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}, #state{gid=UserGID}) -> + [BcastTypeID, BcastGID, BcastName] = case FromGID of + 0 -> %% This probably shouldn't happen. Just make it crash on purpose. + log("chat FromGID=0"), + ignore; + UserGID -> %% player chat: disregard whatever was sent except modifiers and message. + {ok, User} = egs_user_model:read(UserGID), + [16#00001200, User#egs_user_model.id, (User#egs_user_model.character)#characters.name]; + NPCGID -> %% npc chat: @todo Check that the player is the party leader and this npc is in his party. + {ok, User} = egs_user_model:read(NPCGID), + [16#00001d00, FromGID, (User#egs_user_model.character)#characters.name] + end, + %% log the message as ascii to the console + [LogName|_] = re:split(BcastName, "\\0\\0", [{return, binary}]), + [TmpMessage|_] = re:split(ChatMsg, "\\0\\0", [{return, binary}]), + LogMessage = re:replace(TmpMessage, "\\n", " ", [global, {return, binary}]), + log("chat from ~s: ~s", [[re:replace(LogName, "\\0", "", [global, {return, binary}])], [re:replace(LogMessage, "\\0", "", [global, {return, binary}])]]), + %% broadcast + {ok, List} = egs_user_model:select(all), + lists:foreach(fun(X) -> X#egs_user_model.pid ! {psu_chat, BcastTypeID, BcastGID, BcastName, Modifiers, ChatMsg} end, List); + +%% @todo There's at least 9 different sets of locations. Handle all of them correctly. +event(counter_background_locations_request, _State) -> + psu_game:send_170c(); + +%% @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. +event({counter_enter, CounterID, FromZoneID, FromMapID, FromEntryID}, #state{gid=GID}) -> + log("counter load ~b", [CounterID]), + {ok, OldUser} = egs_user_model:read(GID), + OldArea = OldUser#egs_user_model.area, + FromArea = {psu_area, OldArea#psu_area.questid, FromZoneID, FromMapID}, + User = OldUser#egs_user_model{areatype=counter, area={psu_area, 16#7fffffff, 0, 0}, entryid=0, prev_area=FromArea, prev_entryid=FromEntryID}, + egs_user_model:write(User), + AreaName = "Counter", + QuestFile = "data/lobby/counter.quest.nbl", + ZoneFile = "data/lobby/counter.zone.nbl", + %% broadcast unspawn to other people + {ok, UnspawnList} = egs_user_model:select({neighbors, OldUser}), + lists:foreach(fun(Other) -> Other#egs_user_model.pid ! {psu_player_unspawn, User} end, UnspawnList), + %% load counter + psu_proto:send_0c00(User), + psu_proto:send_020e(User, QuestFile), + psu_proto:send_0a05(User), + psu_proto:send_010d(User, User#egs_user_model{lid=0}), + psu_game:send_0200(mission), + psu_game:send_020f(ZoneFile, 0, 16#ff), + psu_proto:send_0205(User, 0), + psu_game:send_100e(16#7fffffff, 0, 0, AreaName, CounterID), + psu_proto:send_0215(User, 0), + psu_proto:send_0215(User, 0), + psu_proto:send_020c(User), + psu_game:send_1202(), + psu_game:send_1204(), + psu_game:send_1206(), + psu_game:send_1207(), + psu_game:send_1212(), + psu_proto:send_0201(User, User#egs_user_model{lid=0}), + psu_game:send_0a06(), + case User#egs_user_model.partypid of + undefined -> ignore; + _ -> psu_game:send_022c(0, 16#12) + end, + psu_game:send_0208(), + psu_game:send_0236(); + +%% @doc Leave mission counter handler. +event(counter_leave, State=#state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + PrevArea = User#egs_user_model.prev_area, + event({area_change, PrevArea#psu_area.questid, PrevArea#psu_area.zoneid, PrevArea#psu_area.mapid, User#egs_user_model.prev_entryid}, State); + +%% @doc Send the code for the background image to use. But there's more that should be sent though. +%% @todo Apparently background values 1 2 3 are never used on official servers. Find out why. +event({counter_options_request, CounterID}, _State) -> + log("counter options request ~p", [CounterID]), + [{quests, _}, {bg, Background}|_Tail] = proplists:get_value(CounterID, ?COUNTERS), + psu_game:send_1711(Background); + +%% @todo Handle when the party already exists! And stop doing it wrong. +event(counter_party_info_request, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + psu_game:send_1706((User#egs_user_model.character)#characters.name); + +%% @todo Item distribution is always set to random for now. +event(counter_party_options_request, _State) -> + psu_game:send_170a(); + +%% @doc Request the counter's quest files. +event({counter_quest_files_request, CounterID}, _State) -> + log("counter quest files request ~p", [CounterID]), + [{quests, Filename}|_Tail] = proplists:get_value(CounterID, ?COUNTERS), + psu_game:send_0c06(Filename); + +%% @doc Counter available mission list request handler. +event({counter_quest_options_request, CounterID}, _State) -> + log("counter quest options request ~p", [CounterID]), + [{quests, _}, {bg, _}, {options, Options}] = proplists:get_value(CounterID, ?COUNTERS), + psu_game:send_0c10(Options); + +%% @todo A and B are mostly unknown. Like most of everything else from the command 0e00... +event({hit, FromTargetID, ToTargetID, A, B}, State=#state{gid=GID}) -> + {ok, User} = egs_user_model: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), + case Type of + box -> + %% @todo also has a hit sent, we should send it too + events(Events, State); + _ -> + PlayerHP = (NewUser#egs_user_model.character)#characters.currenthp, + case lists:member(death, TargetSE) of + true -> SE = 16#01000200; + false -> SE = 16#01000000 + end, + psu_game:send(<< 16#0e070300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, + 1:32/little-unsigned-integer, 16#01050000:32, Damage:32/little-unsigned-integer, + A/binary, 0:64, PlayerHP:32/little-unsigned-integer, 0:32, SE:32, + 0:32, TargetHP:32/little-unsigned-integer, 0:32, B/binary, + 16#04320000:32, 16#80000000:32, 16#26030000:32, 16#89068d00:32, 16#0c1c0105:32, 0:64 >>) + % after TargetHP is SE-related too? + end, + %% exp + if HasEXP =:= true -> + Character = NewUser#egs_user_model.character, + Level = Character#characters.mainlevel, + psu_game:send_0115(GID, ToTargetID, Level#level.number, Level#level.exp, Character#characters.money); + true -> ignore + end, + %% save + egs_user_model:write(NewUser); + +event({hits, Hits}, State) -> + events(Hits, State); + +%% @todo Send something other than just "dammy". +event({item_description_request, ItemID}, _State) -> + case proplists:get_value(ItemID, ?ITEMS) of + undefined -> psu_game:send_0a11(ItemID, "Always bet on Dammy."); + #psu_item{description=Desc} -> psu_game:send_0a11(ItemID, Desc) + end; + +%% @todo A and B are unknown. +%% Melee uses a format similar to: AAAA--BBCCCC----DDDDDDDDEE----FF with +%% AAAA the attack sound effect, BB the range, CCCC and DDDDDDDD unknown but related to angular range or similar, EE number of targets and FF the model. +%% Bullets and tech weapons formats are unknown but likely use a slightly different format. +%% @todo Others probably want to see that you changed your weapon. +%% @todo Apparently B is always ItemID+1. Not sure why. +%% @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}, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + Inventory = (User#egs_user_model.character)#characters.inventory, + case lists:nth(ItemIndex + 1, Inventory) of + {ItemID, Variables} when element(1, Variables) =:= psu_special_item_variables -> + << Category:8, _:24 >> = << ItemID:32 >>, + psu_game:send(<< 16#01050300:32, 0:64, TargetGID:32/little, 0:64, 16#00011300:32, GID:32/little, 0:64, + TargetGID:32/little, TargetLID:32/little, ItemIndex:8, 1:8, Category:8, A:8, B:32/little >>); + {ItemID, Variables} when element(1, Variables) =:= psu_striking_weapon_item_variables -> + ItemInfo = proplists:get_value(ItemID, ?ITEMS), + #psu_item{data=Constants} = ItemInfo, + #psu_striking_weapon_item{attack_sound=Sound, hitbox_a=HitboxA, hitbox_b=HitboxB, + hitbox_c=HitboxC, hitbox_d=HitboxD, nb_targets=NbTargets, effect=Effect, model=Model} = Constants, + << Category:8, _:24 >> = << ItemID:32 >>, + {SoundInt, SoundType} = case Sound of + {default, Val} -> {Val, 0}; + {custom, Val} -> {Val, 8} + end, + psu_game:send(<< 16#01050300:32, 0:64, TargetGID:32/little, 0:64, 16#00011300:32, GID:32/little, 0:64, + TargetGID:32/little, TargetLID:32/little, ItemIndex:8, 1:8, Category:8, A:8, B:32/little, + SoundInt:32/little, HitboxA:16, HitboxB:16, HitboxC:16, HitboxD:16, SoundType:4, NbTargets:4, 0:8, Effect:8, Model:8 >>); + undefined -> + %% @todo Shouldn't be needed later when NPCs are handled correctly. + ignore + end; + +%% @todo A and B are unknown. +%% @see item_equip +event({item_unequip, ItemIndex, TargetGID, TargetLID, A, B}, #state{gid=GID}) -> + Category = case ItemIndex of + % units would be 8, traps would be 12 + 19 -> 2; % armor + Y when Y =:= 5; Y =:= 6; Y =:= 7 -> 0; % clothes + _ -> 1 % weapons + end, + psu_game:send(<< 16#01050300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, + 0:64, TargetGID:32/little-unsigned-integer, TargetLID:32/little-unsigned-integer, ItemIndex, 2, Category, A, B:32/little-unsigned-integer >>); + +%% @todo Just ignore the meseta price for now and send the player where he wanna be! +event(lobby_transport_request, _State) -> + psu_game:send_0c08(true); + +event(lumilass_options_request, _State) -> + psu_game:send_1a03(); + +%% @todo Probably replenish the player HP when entering a non-mission area rather than when aborting the mission? +event(mission_abort, State=#state{gid=GID}) -> + psu_game:send_1006(11, 0), + {ok, User} = egs_user_model:read(GID), + %% delete the mission + if User#egs_user_model.instancepid =:= undefined -> ignore; + true -> psu_instance:stop(User#egs_user_model.instancepid) + end, + %% full hp + Character = User#egs_user_model.character, + MaxHP = Character#characters.maxhp, + NewCharacter = Character#characters{currenthp=MaxHP}, + NewUser = User#egs_user_model{character=NewCharacter, setid=0}, + egs_user_model:write(NewUser), + %% map change + if User#egs_user_model.areatype =:= mission -> + PrevArea = User#egs_user_model.prev_area, + event({area_change, PrevArea#psu_area.questid, PrevArea#psu_area.zoneid, PrevArea#psu_area.mapid, User#egs_user_model.prev_entryid}, State); + true -> ignore + end; + +%% @todo Forward the mission start to other players of the same party, whatever their location is. +event({mission_start, QuestID}, _State) -> + log("mission start ~b", [QuestID]), + psu_game:send_1020(), + psu_game:send_1015(QuestID), + psu_game:send_0c02(); + +%% @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}, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + %% Create NPC. + log("npc force invite ~p", [NPCid]), + TmpNPCUser = psu_npc:user_init(NPCid, ((User#egs_user_model.character)#characters.mainlevel)#level.number), + %% Create and join party. + case User#egs_user_model.partypid of + undefined -> + {ok, PartyPid} = psu_party:start_link(GID); + PartyPid -> + ignore + end, + {ok, PartyPos} = psu_party:join(PartyPid, npc, TmpNPCUser#egs_user_model.id), + #egs_user_model{instancepid=InstancePid, area=Area, entryid=EntryID, pos=Pos} = User, + NPCUser = TmpNPCUser#egs_user_model{lid=PartyPos, partypid=PartyPid, instancepid=InstancePid, areatype=mission, area=Area, entryid=EntryID, pos=Pos}, + egs_user_model:write(NPCUser), + egs_user_model:write(User#egs_user_model{partypid=PartyPid}), + %% Send stuff. + Character = NPCUser#egs_user_model.character, + SentNPCCharacter = Character#characters{gid=NPCid, npcid=NPCid}, + SentNPCUser = NPCUser#egs_user_model{character=SentNPCCharacter}, + psu_proto:send_010d(User, SentNPCUser), + psu_proto:send_0201(User, SentNPCUser), + psu_proto:send_0215(User, 0), + psu_game:send_0a04(SentNPCUser#egs_user_model.id), + psu_game:send_022c(0, 16#12), + psu_game:send_1004(npc_mission, SentNPCUser, PartyPos), + psu_game:send_100f((SentNPCUser#egs_user_model.character)#characters.npcid, PartyPos), + psu_game:send_1601(PartyPos); + +%% @todo Also at the end send a 101a (NPC:16, PartyPos:16, ffffffff). Not sure about PartyPos. +event({npc_invite, NPCid}, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + %% Create NPC. + log("invited npcid ~b", [NPCid]), + TmpNPCUser = psu_npc:user_init(NPCid, ((User#egs_user_model.character)#characters.mainlevel)#level.number), + %% Create and join party. + case User#egs_user_model.partypid of + undefined -> + {ok, PartyPid} = psu_party:start_link(GID), + psu_game:send_022c(0, 16#12); + PartyPid -> + ignore + end, + {ok, PartyPos} = psu_party:join(PartyPid, npc, TmpNPCUser#egs_user_model.id), + NPCUser = TmpNPCUser#egs_user_model{lid=PartyPos, partypid=PartyPid}, + egs_user_model:write(NPCUser), + egs_user_model:write(User#egs_user_model{partypid=PartyPid}), + %% Send stuff. + Character = NPCUser#egs_user_model.character, + SentNPCCharacter = Character#characters{gid=NPCid, npcid=NPCid}, + SentNPCUser = NPCUser#egs_user_model{character=SentNPCCharacter}, + psu_game:send_1004(npc_invite, SentNPCUser, PartyPos), + psu_game:send_101a(NPCid, PartyPos); + +%% @todo Should be 0115(money) 010a03(confirm sale). +%% @todo We probably need to save the ShopID somewhere since it isn't given back here. +event({npc_shop_buy, ShopItemIndex, Quantity}, _State) -> + log("npc shop buy itemindex ~p quantity ~p", [ShopItemIndex, Quantity]); + +%% @todo Currently send the normal items shop for all shops, differentiate. +event({npc_shop_enter, ShopID}, #state{gid=GID}) -> + log("npc shop enter ~p", [ShopID]), + case proplists:get_value(ShopID, ?SHOPS) of + undefined -> %% @todo Temporary; prevent players from getting stuck. + {ok, File} = file:read_file("p/itemshop.bin"), + psu_game:send(<< 16#010a0300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, + GID:32/little-unsigned-integer, 0:64, GID:32/little-unsigned-integer, 0:32, File/binary >>); + ItemsList -> + psu_game:send_010a(ItemsList) + end; + +event({npc_shop_leave, ShopID}, #state{gid=GID}) -> + log("npc shop leave ~p", [ShopID]), + psu_game:send(<< 16#010a0300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, + GID:32/little-unsigned-integer, 0:64, GID:32/little-unsigned-integer, 0:32 >>); + +%% @todo Should be 0115(money) 010a03(confirm sale). +event({npc_shop_sell, InventoryItemIndex, Quantity}, _State) -> + log("npc shop sell itemindex ~p quantity ~p", [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}, _State) -> + log("npc shop request ~p", [ShopID]), + case ShopID of + 80 -> psu_game:send_1a02(0, 17, 17, 3, 9); %% lumilass + 90 -> psu_game:send_1a02(0, 5, 1, 4, 5); %% parum weapon grinding + 91 -> psu_game:send_1a02(0, 5, 5, 4, 7); %% tenora weapon grinding + 92 -> psu_game:send_1a02(0, 5, 0, 4, 0); %% yohmei weapon grinding + 93 -> psu_game:send_1a02(0, 5, 18, 4, 0); %% kubara weapon grinding + _ -> psu_game:send_1a02(0, 0, 1, 0, 0) + end; + +%% @todo Not sure what are those hardcoded values. +event({object_boss_gate_activate, ObjectID}, _State) -> + psu_game:send_1213(ObjectID, 0), + psu_game:send_1215(2, 16#7008), + %% @todo Following sent after the warp? + psu_game:send_1213(37, 0), + %% @todo Why resend this? + psu_game:send_1213(ObjectID, 0); + +event({object_boss_gate_enter, ObjectID}, _State) -> + psu_game:send_1213(ObjectID, 1); + +%% @todo Do we need to send something back here? +event({object_boss_gate_leave, _ObjectID}, _State) -> + ignore; + +event({object_box_destroy, ObjectID}, _State) -> + psu_game:send_1213(ObjectID, 3); + +%% @todo Second send_1211 argument should be User#egs_user_model.lid. Fix when it's correctly handled. +event({object_chair_sit, ObjectTargetID}, _State) -> + %~ {ok, User} = egs_user_model:read(get(gid)), + psu_game:send_1211(ObjectTargetID, 0, 8, 0); + +%% @todo Second psu_game:send_1211 argument should be User#egs_user_model.lid. Fix when it's correctly handled. +event({object_chair_stand, ObjectTargetID}, _State) -> + %~ {ok, User} = egs_user_model:read(get(gid)), + psu_game:send_1211(ObjectTargetID, 0, 8, 2); + +event({object_crystal_activate, ObjectID}, _State) -> + psu_game:send_1213(ObjectID, 1); + +%% @doc Server-side event. +event({object_event_trigger, BlockID, EventID}, _State) -> + psu_game:send_1205(EventID, BlockID, 0); + +event({object_goggle_target_activate, ObjectID}, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + {BlockID, EventID} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), + psu_game:send_1205(EventID, BlockID, 0), + psu_game:send_1213(ObjectID, 8); + +event({object_key_console_enable, ObjectID}, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + {BlockID, [EventID|_]} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), + psu_game:send_1205(EventID, BlockID, 0), + psu_game:send_1213(ObjectID, 1); + +event({object_key_console_init, ObjectID}, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + {BlockID, [_, EventID, _]} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), + psu_game:send_1205(EventID, BlockID, 0); + +event({object_key_console_open_gate, ObjectID}, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + {BlockID, [_, _, EventID]} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), + psu_game:send_1205(EventID, BlockID, 0), + psu_game:send_1213(ObjectID, 1); + +%% @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}, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + {BlockID, [EventID|_]} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), + psu_game:send_1205(EventID, BlockID, 0), + psu_game:send_1213(ObjectID, 1); + +%% @todo Some switch objects apparently work differently, like the light switch in Mines in MAG'. +event({object_switch_off, ObjectID}, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + {BlockID, EventID} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), + psu_game:send_1205(EventID, BlockID, 1), + psu_game:send_1213(ObjectID, 0); + +event({object_switch_on, ObjectID}, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + {BlockID, EventID} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), + psu_game:send_1205(EventID, BlockID, 0), + psu_game:send_1213(ObjectID, 1); + +event({object_vehicle_boost_enable, ObjectID}, _State) -> + psu_game:send_1213(ObjectID, 1); + +event({object_vehicle_boost_respawn, ObjectID}, _State) -> + psu_game:send_1213(ObjectID, 0); + +%% @todo Second send_1211 argument should be User#egs_user_model.lid. Fix when it's correctly handled. +event({object_warp_take, BlockID, ListNb, ObjectNb}, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + Pos = psu_instance:warp_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, BlockID, ListNb, ObjectNb), + NewUser = User#egs_user_model{pos=Pos}, + egs_user_model:write(NewUser), + psu_game:send_0503(User#egs_user_model.pos), + psu_game:send_1211(16#ffffffff, 0, 14, 0); + +event({party_remove_member, PartyPos}, #state{gid=GID}) -> + log("party remove member ~b", [PartyPos]), + {ok, DestUser} = egs_user_model:read(GID), + {ok, RemovedGID} = psu_party:get_member(DestUser#egs_user_model.partypid, PartyPos), + psu_party:remove_member(DestUser#egs_user_model.partypid, PartyPos), + {ok, RemovedUser} = egs_user_model:read(RemovedGID), + case (RemovedUser#egs_user_model.character)#characters.type of + npc -> egs_user_model:delete(RemovedGID); + _ -> ignore + end, + psu_game:send_1006(8, PartyPos), + psu_game:send_0204(DestUser, RemovedUser, 1), + psu_proto:send_0215(DestUser, 0); + +event({player_options_change, Options}, #state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + file:write_file(io_lib:format("save/~s/~b-character.options", [User#egs_user_model.folder, (User#egs_user_model.character)#characters.slot]), Options); + +%% @todo If the player has a scape, use it! Otherwise red screen. +%% @todo Right now we force revive and don't update the player's HP. +event(player_death, _State) -> + % @todo send_0115(get(gid), 16#ffffffff, LV=1, EXP=idk, Money=1000), % apparently sent everytime you die... + %% use scape: + NewHP = 10, + psu_game:send_0117(NewHP), + psu_game:send_1022(NewHP); + %% red screen with return to lobby choice: + %~ psu_game:send_0111(3, 1); + +%% @todo Refill the player's HP to maximum, remove SEs etc. +event(player_death_return_to_lobby, State=#state{gid=GID}) -> + {ok, User} = egs_user_model:read(GID), + PrevArea = User#egs_user_model.prev_area, + event({area_change, PrevArea#psu_area.questid, PrevArea#psu_area.zoneid, PrevArea#psu_area.mapid, User#egs_user_model.prev_entryid}, State); + +event(player_type_availability_request, _State) -> + psu_game:send_1a07(); + +event(player_type_capabilities_request, _State) -> + psu_game:send_0113(); + +event(ppcube_request, _State) -> + psu_game:send_1a04(); + +%% @doc Uni cube handler. +event(unicube_request, _State) -> + psu_game:send_021e(); + +%% @doc Uni selection handler. +%% @todo When selecting 'Your room', load a default room. +%% @todo When selecting 'Reload', reload the character in the current lobby. +%% @todo Delete NPC characters and stop the party on entering myroom too. +event({unicube_select, Selection, EntryID}, State=#state{gid=GID}) -> + case Selection of + cancel -> ignore; + 16#ffffffff -> + log("uni selection (my room)"), + psu_game:send_0230(), + % 0220 + event({area_change, 1120000, 0, 100, 0}, State); + _UniID -> + log("uni selection (reload)"), + psu_game:send_0230(), + % 0220 + %% force reloading the character and data files (@todo hack, uses myroom questid to do it) + {ok, User} = egs_user_model:read(GID), + case User#egs_user_model.partypid of + undefined -> + ignore; + PartyPid -> + %% @todo Replace stop by leave when leaving stops the party correctly when nobody's there anymore. + %~ psu_party:leave(User#egs_user_model.partypid, User#egs_user_model.id) + {ok, NPCList} = psu_party:get_npc(PartyPid), + [egs_user_model:delete(NPCGID) || {_Spot, NPCGID} <- NPCList], + psu_party:stop(PartyPid) + end, + Area = User#egs_user_model.area, + NewRow = User#egs_user_model{partypid=undefined, area=Area#psu_area{questid=1120000, zoneid=undefined}, entryid=EntryID}, + egs_user_model:write(NewRow), + event({area_change, Area#psu_area.questid, Area#psu_area.zoneid, Area#psu_area.mapid, EntryID}, State) + end. + +%% Internal. + +%% @doc Trigger many events. +events(Events, State) -> + [event(Event, State) || Event <- Events], + ok. + +%% @doc Log message to the console. +log(Message) -> + io:format("~p: ~s~n", [get(gid), Message]). + +log(Message, Format) -> + FormattedMessage = io_lib:format(Message, Format), + log(FormattedMessage). diff --git a/src/egs_login.erl b/src/egs_login.erl new file mode 100644 index 0000000..6c4eb73 --- /dev/null +++ b/src/egs_login.erl @@ -0,0 +1,75 @@ +%% @author Loïc Hoguin +%% @copyright 2010 Loïc Hoguin. +%% @doc Game server's client authentication callback module. +%% +%% 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_login). +-export([init/1, keepalive/1, info/2, cast/3, raw/3, event/2]). + +%% @todo This header is only included because of egs_user_model. We don't want that here. +-include("include/records.hrl"). + +-record(state, {socket}). + +%% @doc Initialize the game state and start receiving messages. +%% @todo Link against egs_exit_mon. +%% @todo Handle keepalive messages globally. +%% @todo Probably have init where start_link is. +init(Socket) -> + psu_proto:send_0202(Socket, 0), + timer:send_interval(5000, {egs, keepalive}), + egs_network:recv(<< >>, ?MODULE, #state{socket=Socket}). + +%% @doc Don't keep alive here, authentication should go fast. +keepalive(_State) -> + ok. + +%% @doc We don't expect any message here. +%% @todo Throw an error instead? +info(_Msg, _State) -> + ok. + +%% @doc Nothing to broadcast. +%% @todo Throw an error instead? +cast(_Command, _Data, _State) -> + ok. + +%% @doc Dismiss all raw commands with a log notice. +%% @todo Have a log event handler instead. +raw(Command, _Data, _State) -> + io:format("~p: dismissed command ~4.16.0b~n", [?MODULE, Command]). + +%% Events. + +%% @todo Check the client version info here too. Not just on login. +event({system_client_version_info, _Language, _Platform, _Version}, _State) -> + ok; + +%% @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. +%% @todo Remove the put calls when all the send_xxxx are moved out of psu_game and into psu_proto. +event({system_key_auth_request, AuthGID, AuthKey}, #state{socket=Socket}) -> + {ok, User} = egs_user_model:read(AuthGID), + {wait_for_authentication, AuthKey} = User#egs_user_model.state, + put(socket, Socket), + put(gid, AuthGID), + LID = 1 + mnesia:dirty_update_counter(counters, lobby, 1) rem 1023, + Time = calendar:datetime_to_gregorian_seconds(calendar:universal_time()), + User2 = User#egs_user_model{id=AuthGID, pid=self(), socket=Socket, state=authenticated, time=Time, lid=LID}, + egs_user_model:write(User2), + psu_proto:send_0d05(User2), + {ok, egs_char_select, {state, Socket, AuthGID}}. diff --git a/src/egs_network.erl b/src/egs_network.erl new file mode 100644 index 0000000..c381cad --- /dev/null +++ b/src/egs_network.erl @@ -0,0 +1,103 @@ +%% @author Loïc Hoguin +%% @copyright 2010 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([listen/2, recv/3]). %% API. +-export([accept/2]). %% Internal. + +-define(OPTIONS, [binary, {active, true}, {reuseaddr, true}, {certfile, "priv/ssl/servercert.pem"}, {keyfile, "priv/ssl/serverkey.pem"}, {password, "alpha"}]). + +%% @doc Listen for connections. +listen(Port, CallbackMod) -> + error_logger:info_report(io_lib:format("started a listener for ~p on port ~b", [CallbackMod, Port])), + {ok, LSocket} = ssl:listen(Port, ?OPTIONS), + ?MODULE:accept(LSocket, CallbackMod). + +%% @doc Accept connections. +accept(LSocket, CallbackMod) -> + case ssl:transport_accept(LSocket, 5000) of + {ok, CSocket} -> + case ssl:ssl_accept(CSocket, 5000) of + ok -> + Pid = spawn(CallbackMod, init, [CSocket]), + ssl:controlling_process(CSocket, Pid); + {error, _Reason} -> + ignore + end; + {error, _Reason} -> + ignore + end, + ?MODULE:accept(LSocket, CallbackMod). + +%% @doc Main loop for the network stack. Receive and handle messages. +recv(SoFar, CallbackMod, State) -> + receive + {ssl, _Any, Data} -> + {Commands, Rest} = split(<< SoFar/bits, Data/bits >>, []), + {ok, NextCallbackMod, NewState} = dispatch(Commands, CallbackMod, CallbackMod, State), + ?MODULE:recv(Rest, NextCallbackMod, NewState); + {ssl_closed, _} -> + ssl_closed; %% exit + {ssl_error, _, _} -> + ssl_error; %% exit + {egs, keepalive} -> + CallbackMod:keepalive(State), + ?MODULE:recv(SoFar, CallbackMod, State); + Tuple when element(1, Tuple) =:= egs -> + case CallbackMod:info(Tuple, State) of + {ok, NewState} -> ?MODULE:recv(SoFar, CallbackMod, NewState); + _Any -> ?MODULE:recv(SoFar, CallbackMod, State) + end; + _ -> + ?MODULE:recv(SoFar, CallbackMod, State) + end. + +%% @doc Dispatch the commands received to the right handler. +dispatch([], _CallbackMod, NextMod, State) -> + {ok, NextMod, State}; +dispatch([Data|Tail], CallbackMod, NextMod, State) -> + Ret = case psu_proto:parse(Data) of + {command, Command, Channel} -> + case Channel of + 1 -> CallbackMod:cast(Command, Data, State); + _ -> CallbackMod:raw(Command, Data, State) + end; + ignore -> + ignore; + Event -> + CallbackMod:event(Event, State) + end, + case Ret of + {ok, NewMod, NewState} -> + dispatch(Tail, CallbackMod, NewMod, NewState); + {ok, NewState} -> + dispatch(Tail, CallbackMod, NextMod, NewState); + _Any -> + dispatch(Tail, CallbackMod, NextMod, State) + 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) -> + BitSize = Size * 8, + << Split:BitSize/bits, Rest/bits >> = Data, + split(Rest, [Split|Acc]). diff --git a/src/psu/psu_game.erl b/src/psu/psu_game.erl index 62ade89..79d2f46 100644 --- a/src/psu/psu_game.erl +++ b/src/psu/psu_game.erl @@ -19,7 +19,7 @@ -module(psu_game). -export([start_link/1, cleanup/1]). %% External. --export([listen/2, accept/2, process_init/2, process/0, char_select/0, loop/1]). %% Internal. +-compile(export_all). %% @todo Temporarily export all until send_xxxx functions are moved to psu_proto. -include("include/records.hrl"). -include("include/maps.hrl"). @@ -32,8 +32,8 @@ %% @spec start_link(Port) -> {ok,Pid::pid()} %% @doc Start the game server. start_link(Port) -> - {ok, MPid} = egs_exit_mon:start_link({?MODULE, cleanup}), - LPid = spawn(?MODULE, listen, [Port, MPid]), + %~ {ok, MPid} = egs_exit_mon:start_link({?MODULE, cleanup}), + LPid = spawn(egs_network, listen, [Port, egs_login]), {ok, LPid}. %% @spec cleanup(Pid) -> ok @@ -59,166 +59,9 @@ cleanup(Pid) -> ignore end. -%% @doc Listen for connections. -listen(Port, MPid) -> - error_logger:info_report(io_lib:format("psu_game listening on port ~b", [Port])), - {ok, LSocket} = ssl:listen(Port, ?OPTIONS), - ?MODULE:accept(LSocket, MPid). - -%% @doc Accept connections. -accept(LSocket, MPid) -> - case ssl:transport_accept(LSocket, 5000) of - {ok, CSocket} -> - case ssl:ssl_accept(CSocket, 5000) of - ok -> - Pid = spawn(?MODULE, process_init, [CSocket, MPid]), - ssl:controlling_process(CSocket, Pid); - {error, _Reason} -> - reload - end; - _ -> - reload - end, - ?MODULE:accept(LSocket, MPid). - -%% @doc Initialize the client process by saving the socket to the process dictionary. -process_init(CSocket, MPid) -> - link(MPid), - put(socket, CSocket), - psu_proto:send_0202(CSocket, 0), - timer:send_interval(5000, {psu_keepalive}), - process(). - -%% @doc Process the new connections. -%% Send an hello packet, authenticate the user and send him to character select. -process() -> - case psu_proto:packet_recv(get(socket), 5000) of - {ok, Orig} -> - case psu_proto:parse(Orig) of - ignore -> ?MODULE:process(); - Event -> process_event(Event) - end; - {error, timeout} -> - reload, - ?MODULE:process(); - {error, closed} -> - closed - end. - -%% @todo Check the client version info here too. Not just in psu_login. -process_event({system_client_version_info, _Language, _Platform, _Version}) -> - ?MODULE:process(); - -process_event({system_key_auth_request, AuthGID, AuthKey}) -> - CSocket = get(socket), - case egs_user_model:read(AuthGID) of - {error, badarg} -> - log("can't find user, closing"), - ssl:close(CSocket); - {ok, User} -> - case User#egs_user_model.state of - {wait_for_authentication, AuthKey} -> - put(gid, AuthGID), - log("auth success"), - LID = 1 + mnesia:dirty_update_counter(counters, lobby, 1) rem 1023, - Time = calendar:datetime_to_gregorian_seconds(calendar:universal_time()), - NewUser = #egs_user_model{id=AuthGID, pid=self(), socket=CSocket, state=authenticated, time=Time, folder=User#egs_user_model.folder, lid=LID}, - egs_user_model:write(NewUser), - psu_proto:send_0d05(NewUser), - ?MODULE:char_select(); - _ -> - log("quit, auth failed"), - ssl:close(CSocket) - end - end; - -process_event({command, Command, Channel, _Data}) -> - log("process_event: dismissed command ~4.16.0b channel ~b", [Command, Channel]), - ?MODULE:process(). - -%% @doc Character selection screen loop. -char_select() -> - case psu_proto:packet_recv(get(socket), 5000) of - {ok, Orig} -> - case psu_proto:parse(Orig) of - ignore -> ?MODULE:char_select(); - Event -> char_select_event(Event) - end; - {error, timeout} -> - psu_proto:send_keepalive(get(socket)), - reload, - ?MODULE:char_select(); - {error, closed} -> - closed %% exit - end. - -%% @todo Reenable appearance validation whenever things go live. -char_select_event({char_select_create, Slot, CharBin}) -> - log("character creation ~b", [Slot]), - %% 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 - {ok, User} = egs_user_model:read(get(gid)), - _ = file:make_dir(io_lib:format("save/~s", [User#egs_user_model.folder])), - file:write_file(io_lib:format("save/~s/~b-character", [User#egs_user_model.folder, Slot]), CharBin), - file:write_file(io_lib:format("save/~s/~b-character.options", [User#egs_user_model.folder, Slot]), << 0:128, 4, 0:56 >>), % default 0 to everything except brightness 4 - ?MODULE:char_select(); - -char_select_event({char_select_enter, Slot, _BackToPreviousField}) -> - log("selected character ~b", [Slot]), - char_select_load(Slot); - -char_select_event(char_select_request) -> - {ok, User} = egs_user_model:read(get(gid)), - send_0d03(data_load(User#egs_user_model.folder, 0), data_load(User#egs_user_model.folder, 1), data_load(User#egs_user_model.folder, 2), data_load(User#egs_user_model.folder, 3)), - ?MODULE:char_select(); - -char_select_event({command, Command, Channel, _Data}) -> - log("char_select_event: dismissed command ~4.16.0b channel ~b", [Command, Channel]), - ?MODULE:char_select(). - -%% @doc Load the selected character in the start lobby and start the main game's loop. -%% The default entry point currently is 4th floor, Linear Line counter. -char_select_load(Number) -> - {ok, OldUser} = egs_user_model:read(get(gid)), - [{status, 1}, {char, CharBin}, {options, OptionsBin}] = data_load(OldUser#egs_user_model.folder, Number), - << Name:512/bits, RaceBin:8, GenderBin:8, ClassBin:8, AppearanceBin:776/bits, _/bits >> = CharBin, - psu_characters:validate_name(Name), % TODO: don't validate name when loading character, do it at creation - 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), - Character = #characters{slot=Number, name=Name, race=Race, gender=Gender, class=Class, appearance=Appearance, options=Options, % TODO: temporary set the slot here, won't be needed later - inventory= [{16#11010000, #psu_special_item_variables{}}, {16#11020000, #psu_special_item_variables{}}, {16#11020100, #psu_special_item_variables{}}, {16#11020200, #psu_special_item_variables{}}, - {16#01010900, #psu_striking_weapon_item_variables{is_active=0, slot=0, current_pp=99, max_pp=100, element=#psu_element{type=1, percent=50}, pa=#psu_pa{type=0, level=0}}}, - {16#01010a00, #psu_striking_weapon_item_variables{is_active=0, slot=0, current_pp=99, max_pp=100, element=#psu_element{type=2, percent=50}, pa=#psu_pa{type=0, level=0}}}, - {16#01010b00, #psu_striking_weapon_item_variables{is_active=0, slot=0, current_pp=99, max_pp=100, element=#psu_element{type=3, percent=50}, pa=#psu_pa{type=0, level=0}}}]}, - User = OldUser#egs_user_model{state=online, character=Character, area=#psu_area{questid=undefined, zoneid=undefined, mapid=undefined}, - prev_area={psu_area, 0, 0, 0}, prev_entryid=0, pos=#pos{x=0.0, y=0.0, z=0.0, dir=0.0}, setid=0}, - egs_user_model:write(User), - char_load(User), - event({area_change, 1100000, 0, 4, 5}), - ssl:setopts(get(socket), [{active, true}]), - ?MODULE:loop(<< >>). - -%% @doc Load the given character's data. -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, _} -> - [{status, 0}, {char, << 0:2208 >>}] - end. - %% @doc Load and send the character information to the client. %% @todo Should wait for the 021c reply before doing area_change. +%% @todo Move this whole function directly to psu_proto, probably. char_load(User) -> send_0d01(User), % 0246 @@ -432,739 +275,6 @@ npc_load(Leader, [{PartyPos, NPCGID}|NPCList]) -> send_1016(PartyPos), npc_load(Leader, NPCList). -%% @doc Game's main loop. -%% @todo We probably don't want to send a keepalive packet unnecessarily. -loop(SoFar) -> - receive - {psu_broadcast, Orig} -> - << A:64/bits, _:32, B:96/bits, _:64, C/bits >> = Orig, - GID = get(gid), - send(<< A/binary, 16#00011300:32, B/binary, 16#00011300:32, GID:32/little-unsigned-integer, C/binary >>), - ?MODULE:loop(SoFar); - {psu_chat, ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage} -> - send_0304(ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage), - ?MODULE:loop(SoFar); - {psu_keepalive} -> - psu_proto:send_keepalive(get(socket)), - ?MODULE:loop(SoFar); - {psu_player_spawn, _Spawn} -> - % Should be something along the lines of 203 201 204 or something. - {ok, User} = egs_user_model:read(get(gid)), - {ok, SpawnList} = egs_user_model:select({neighbors, User}), - send_0233(SpawnList), - ?MODULE:loop(SoFar); - {psu_player_unspawn, Spawn} -> - {ok, User} = egs_user_model:read(get(gid)), - send_0204(User, Spawn, 5), - ?MODULE:loop(SoFar); - {psu_warp, QuestID, ZoneID, MapID, EntryID} -> - event({area_change, QuestID, ZoneID, MapID, EntryID}), - ?MODULE:loop(SoFar); - {ssl, _, Data} -> - {Packets, Rest} = psu_proto:packet_split(<< SoFar/bits, Data/bits >>), - [dispatch(Orig) || Orig <- Packets], - ?MODULE:loop(Rest); - {ssl_closed, _} -> - ssl_closed; %% exit - {ssl_error, _, _} -> - ssl_error; %% exit - _ -> - ?MODULE:loop(SoFar) - after 1000 -> - reload, - ?MODULE:loop(SoFar) - end. - -%% @doc Dispatch the command to the right handler. -dispatch(Orig) -> - case psu_proto:parse(Orig) of - {command, Command, Channel, Data} -> - case Channel of - 1 -> broadcast(Command, Orig); - _ -> handle(Command, Data) - end; - ignore -> - ignore; - Event -> - event(Event) - end. - -%% @doc Position change broadcast handler. Save the position and then dispatch it. -broadcast(16#0503, Orig) -> - << _:424, Dir:24/little-unsigned-integer, _PrevCoords:96, X:32/little-float, Y:32/little-float, Z:32/little-float, - QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer, _:32 >> = Orig, - FloatDir = Dir / 46603.375, - {ok, User} = egs_user_model:read(get(gid)), - NewUser = User#egs_user_model{pos=#pos{x=X, y=Y, z=Z, dir=FloatDir}, area=#psu_area{questid=QuestID, zoneid=ZoneID, mapid=MapID}, entryid=EntryID}, - egs_user_model:write(NewUser), - broadcast(default, Orig); - -%% @doc Stand still broadcast handler. Save the position and then dispatch it. -broadcast(16#0514, Orig) -> - << _:424, Dir:24/little-unsigned-integer, X:32/little-float, Y:32/little-float, Z:32/little-float, - QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, - MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer, _/bits >> = Orig, - FloatDir = Dir / 46603.375, - {ok, User} = egs_user_model:read(get(gid)), - NewUser = User#egs_user_model{pos=#pos{x=X, y=Y, z=Z, dir=FloatDir}, area=#psu_area{questid=QuestID, zoneid=ZoneID, mapid=MapID}, entryid=EntryID}, - egs_user_model:write(NewUser), - broadcast(default, Orig); - -%% @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 an User instead of a GID probably. -broadcast(Command, Orig) - when Command =:= 16#0101; - Command =:= 16#0102; - Command =:= 16#0104; - Command =:= 16#0107; - Command =:= 16#010f; - Command =:= 16#050f; - Command =:= default -> - << _:32, A:64/bits, _:64, B:192/bits, _:64, C/bits >> = Orig, - GID = get(gid), - case egs_user_model:read(GID) of - {error, _Reason} -> - ignore; - {ok, Self} -> - LID = Self#egs_user_model.lid, - Packet = << A/binary, 16#00011300:32, GID:32/little-unsigned-integer, B/binary, - GID:32/little-unsigned-integer, LID:32/little-unsigned-integer, C/binary >>, - {ok, SpawnList} = egs_user_model:select({neighbors, Self}), - lists:foreach(fun(User) -> User#egs_user_model.pid ! {psu_broadcast, Packet} end, SpawnList) - end. - -%% @doc Trigger many events. -events(Events) -> - [event(Event) || Event <- Events], - ok. - -%% @todo When changing lobby to the room, 0230 must also be sent. Same when going from room to lobby. -%% @todo Probably move area_load inside the event and make other events call this one when needed. -event({area_change, QuestID, ZoneID, MapID, EntryID}) -> - event({area_change, QuestID, ZoneID, MapID, EntryID, 16#ffffffff}); -event({area_change, QuestID, ZoneID, MapID, EntryID, PartyPos}) -> - case PartyPos of - 16#ffffffff -> - log("area change (~b,~b,~b,~b,~b)", [QuestID, ZoneID, MapID, EntryID, PartyPos]), - area_load(QuestID, ZoneID, MapID, EntryID); - _Any -> %% @todo Handle area_change event for NPCs in story missions. - ignore - end; - -%% @doc Chat broadcast handler. Dispatch the message to everyone (for now). -%% Disregard the name sent by the server. Use the name saved in memory instead, to prevent client-side editing. -%% @todo Only broadcast to people in the same map. -%% @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}) -> - UserGID = get(gid), - [BcastTypeID, BcastGID, BcastName] = case FromGID of - 0 -> %% This probably shouldn't happen. Just make it crash on purpose. - log("chat FromGID=0"), - ignore; - UserGID -> %% player chat: disregard whatever was sent except modifiers and message. - {ok, User} = egs_user_model:read(UserGID), - [16#00001200, User#egs_user_model.id, (User#egs_user_model.character)#characters.name]; - NPCGID -> %% npc chat: @todo Check that the player is the party leader and this npc is in his party. - {ok, User} = egs_user_model:read(NPCGID), - [16#00001d00, FromGID, (User#egs_user_model.character)#characters.name] - end, - %% log the message as ascii to the console - [LogName|_] = re:split(BcastName, "\\0\\0", [{return, binary}]), - [TmpMessage|_] = re:split(ChatMsg, "\\0\\0", [{return, binary}]), - LogMessage = re:replace(TmpMessage, "\\n", " ", [global, {return, binary}]), - log("chat from ~s: ~s", [[re:replace(LogName, "\\0", "", [global, {return, binary}])], [re:replace(LogMessage, "\\0", "", [global, {return, binary}])]]), - %% broadcast - {ok, List} = egs_user_model:select(all), - lists:foreach(fun(X) -> X#egs_user_model.pid ! {psu_chat, BcastTypeID, BcastGID, BcastName, Modifiers, ChatMsg} end, List); - -%% @todo There's at least 9 different sets of locations. Handle all of them correctly. -event(counter_background_locations_request) -> - send_170c(); - -%% @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. -event({counter_enter, CounterID, FromZoneID, FromMapID, FromEntryID}) -> - log("counter load ~b", [CounterID]), - {ok, OldUser} = egs_user_model:read(get(gid)), - OldArea = OldUser#egs_user_model.area, - FromArea = {psu_area, OldArea#psu_area.questid, FromZoneID, FromMapID}, - User = OldUser#egs_user_model{areatype=counter, area={psu_area, 16#7fffffff, 0, 0}, entryid=0, prev_area=FromArea, prev_entryid=FromEntryID}, - egs_user_model:write(User), - AreaName = "Counter", - QuestFile = "data/lobby/counter.quest.nbl", - ZoneFile = "data/lobby/counter.zone.nbl", - %% broadcast unspawn to other people - {ok, UnspawnList} = egs_user_model:select({neighbors, OldUser}), - lists:foreach(fun(Other) -> Other#egs_user_model.pid ! {psu_player_unspawn, User} end, UnspawnList), - %% load counter - psu_proto:send_0c00(User), - psu_proto:send_020e(User, QuestFile), - psu_proto:send_0a05(User), - psu_proto:send_010d(User, User#egs_user_model{lid=0}), - send_0200(mission), - send_020f(ZoneFile, 0, 16#ff), - psu_proto:send_0205(User, 0), - send_100e(16#7fffffff, 0, 0, AreaName, CounterID), - psu_proto:send_0215(User, 0), - psu_proto:send_0215(User, 0), - psu_proto:send_020c(User), - send_1202(), - send_1204(), - send_1206(), - send_1207(), - send_1212(), - psu_proto:send_0201(User, User#egs_user_model{lid=0}), - send_0a06(), - case User#egs_user_model.partypid of - undefined -> ignore; - _ -> send_022c(0, 16#12) - end, - send_0208(), - send_0236(); - -%% @doc Leave mission counter handler. -event(counter_leave) -> - {ok, User} = egs_user_model:read(get(gid)), - PrevArea = User#egs_user_model.prev_area, - event({area_change, PrevArea#psu_area.questid, PrevArea#psu_area.zoneid, PrevArea#psu_area.mapid, User#egs_user_model.prev_entryid}); - -%% @doc Send the code for the background image to use. But there's more that should be sent though. -%% @todo Apparently background values 1 2 3 are never used on official servers. Find out why. -event({counter_options_request, CounterID}) -> - log("counter options request ~p", [CounterID]), - [{quests, _}, {bg, Background}|_Tail] = proplists:get_value(CounterID, ?COUNTERS), - send_1711(Background); - -%% @todo Handle when the party already exists! And stop doing it wrong. -event(counter_party_info_request) -> - {ok, User} = egs_user_model:read(get(gid)), - send_1706((User#egs_user_model.character)#characters.name); - -%% @todo Item distribution is always set to random for now. -event(counter_party_options_request) -> - send_170a(); - -%% @doc Request the counter's quest files. -event({counter_quest_files_request, CounterID}) -> - log("counter quest files request ~p", [CounterID]), - [{quests, Filename}|_Tail] = proplists:get_value(CounterID, ?COUNTERS), - send_0c06(Filename); - -%% @doc Counter available mission list request handler. -event({counter_quest_options_request, CounterID}) -> - log("counter quest options request ~p", [CounterID]), - [{quests, _}, {bg, _}, {options, Options}] = proplists:get_value(CounterID, ?COUNTERS), - send_0c10(Options); - -%% @todo A and B are mostly unknown. Like most of everything else from the command 0e00... -event({hit, FromTargetID, ToTargetID, A, B}) -> - GID = get(gid), - {ok, User} = egs_user_model: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), - case Type of - box -> - %% @todo also has a hit sent, we should send it too - events(Events); - _ -> - PlayerHP = (NewUser#egs_user_model.character)#characters.currenthp, - case lists:member(death, TargetSE) of - true -> SE = 16#01000200; - false -> SE = 16#01000000 - end, - send(<< 16#0e070300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, - 1:32/little-unsigned-integer, 16#01050000:32, Damage:32/little-unsigned-integer, - A/binary, 0:64, PlayerHP:32/little-unsigned-integer, 0:32, SE:32, - 0:32, TargetHP:32/little-unsigned-integer, 0:32, B/binary, - 16#04320000:32, 16#80000000:32, 16#26030000:32, 16#89068d00:32, 16#0c1c0105:32, 0:64 >>) - % after TargetHP is SE-related too? - end, - %% exp - if HasEXP =:= true -> - Character = NewUser#egs_user_model.character, - Level = Character#characters.mainlevel, - send_0115(GID, ToTargetID, Level#level.number, Level#level.exp, Character#characters.money); - true -> ignore - end, - %% save - egs_user_model:write(NewUser); - -event({hits, Hits}) -> - [event(Hit) || Hit <- Hits], - ok; - -%% @todo Send something other than just "dammy". -event({item_description_request, ItemID}) -> - case proplists:get_value(ItemID, ?ITEMS) of - undefined -> send_0a11(ItemID, "Always bet on Dammy."); - #psu_item{description=Desc} -> send_0a11(ItemID, Desc) - end; - -%% @todo A and B are unknown. -%% Melee uses a format similar to: AAAA--BBCCCC----DDDDDDDDEE----FF with -%% AAAA the attack sound effect, BB the range, CCCC and DDDDDDDD unknown but related to angular range or similar, EE number of targets and FF the model. -%% Bullets and tech weapons formats are unknown but likely use a slightly different format. -%% @todo Others probably want to see that you changed your weapon. -%% @todo Apparently B is always ItemID+1. Not sure why. -%% @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}) -> - GID = get(gid), - {ok, User} = egs_user_model:read(GID), - Inventory = (User#egs_user_model.character)#characters.inventory, - case lists:nth(ItemIndex + 1, Inventory) of - {ItemID, Variables} when element(1, Variables) =:= psu_special_item_variables -> - << Category:8, _:24 >> = << ItemID:32 >>, - send(<< 16#01050300:32, 0:64, TargetGID:32/little, 0:64, 16#00011300:32, GID:32/little, 0:64, - TargetGID:32/little, TargetLID:32/little, ItemIndex:8, 1:8, Category:8, A:8, B:32/little >>); - {ItemID, Variables} when element(1, Variables) =:= psu_striking_weapon_item_variables -> - ItemInfo = proplists:get_value(ItemID, ?ITEMS), - #psu_item{data=Constants} = ItemInfo, - #psu_striking_weapon_item{attack_sound=Sound, hitbox_a=HitboxA, hitbox_b=HitboxB, - hitbox_c=HitboxC, hitbox_d=HitboxD, nb_targets=NbTargets, effect=Effect, model=Model} = Constants, - << Category:8, _:24 >> = << ItemID:32 >>, - {SoundInt, SoundType} = case Sound of - {default, Val} -> {Val, 0}; - {custom, Val} -> {Val, 8} - end, - send(<< 16#01050300:32, 0:64, TargetGID:32/little, 0:64, 16#00011300:32, GID:32/little, 0:64, - TargetGID:32/little, TargetLID:32/little, ItemIndex:8, 1:8, Category:8, A:8, B:32/little, - SoundInt:32/little, HitboxA:16, HitboxB:16, HitboxC:16, HitboxD:16, SoundType:4, NbTargets:4, 0:8, Effect:8, Model:8 >>); - undefined -> - %% @todo Shouldn't be needed later when NPCs are handled correctly. - ignore - end; - -%% @todo A and B are unknown. -%% @see item_equip -event({item_unequip, ItemIndex, TargetGID, TargetLID, A, B}) -> - GID = get(gid), - Category = case ItemIndex of - % units would be 8, traps would be 12 - 19 -> 2; % armor - Y when Y =:= 5; Y =:= 6; Y =:= 7 -> 0; % clothes - _ -> 1 % weapons - end, - send(<< 16#01050300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, - 0:64, TargetGID:32/little-unsigned-integer, TargetLID:32/little-unsigned-integer, ItemIndex, 2, Category, A, B:32/little-unsigned-integer >>); - -%% @todo Just ignore the meseta price for now and send the player where he wanna be! -event(lobby_transport_request) -> - send_0c08(true); - -%% @todo Handle all different Lumilass. -event(lumilass_options_request) -> - send_1a03(); - -%% @todo Probably replenish the player HP when entering a non-mission area rather than when aborting the mission? -event(mission_abort) -> - send_1006(11, 0), - {ok, User} = egs_user_model:read(get(gid)), - %% delete the mission - if User#egs_user_model.instancepid =:= undefined -> ignore; - true -> psu_instance:stop(User#egs_user_model.instancepid) - end, - %% full hp - Character = User#egs_user_model.character, - MaxHP = Character#characters.maxhp, - NewCharacter = Character#characters{currenthp=MaxHP}, - NewUser = User#egs_user_model{character=NewCharacter, setid=0}, - egs_user_model:write(NewUser), - %% map change - if User#egs_user_model.areatype =:= mission -> - PrevArea = User#egs_user_model.prev_area, - event({area_change, PrevArea#psu_area.questid, PrevArea#psu_area.zoneid, PrevArea#psu_area.mapid, User#egs_user_model.prev_entryid}); - true -> ignore - end; - -%% @todo Forward the mission start to other players of the same party, whatever their location is. -event({mission_start, QuestID}) -> - log("mission start ~b", [QuestID]), - send_1020(), - send_1015(QuestID), - send_0c02(); - -%% @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}) -> - GID = get(gid), - {ok, User} = egs_user_model:read(GID), - %% Create NPC. - log("npc force invite ~p", [NPCid]), - TmpNPCUser = psu_npc:user_init(NPCid, ((User#egs_user_model.character)#characters.mainlevel)#level.number), - %% Create and join party. - case User#egs_user_model.partypid of - undefined -> - {ok, PartyPid} = psu_party:start_link(GID); - PartyPid -> - ignore - end, - {ok, PartyPos} = psu_party:join(PartyPid, npc, TmpNPCUser#egs_user_model.id), - #egs_user_model{instancepid=InstancePid, area=Area, entryid=EntryID, pos=Pos} = User, - NPCUser = TmpNPCUser#egs_user_model{lid=PartyPos, partypid=PartyPid, instancepid=InstancePid, areatype=mission, area=Area, entryid=EntryID, pos=Pos}, - egs_user_model:write(NPCUser), - egs_user_model:write(User#egs_user_model{partypid=PartyPid}), - %% Send stuff. - Character = NPCUser#egs_user_model.character, - SentNPCCharacter = Character#characters{gid=NPCid, npcid=NPCid}, - SentNPCUser = NPCUser#egs_user_model{character=SentNPCCharacter}, - psu_proto:send_010d(User, SentNPCUser), - psu_proto:send_0201(User, SentNPCUser), - psu_proto:send_0215(User, 0), - send_0a04(SentNPCUser#egs_user_model.id), - send_022c(0, 16#12), - send_1004(npc_mission, SentNPCUser, PartyPos), - send_100f((SentNPCUser#egs_user_model.character)#characters.npcid, PartyPos), - send_1601(PartyPos); - -%% @todo Also at the end send a 101a (NPC:16, PartyPos:16, ffffffff). Not sure about PartyPos. -event({npc_invite, NPCid}) -> - GID = get(gid), - {ok, User} = egs_user_model:read(GID), - %% Create NPC. - log("invited npcid ~b", [NPCid]), - TmpNPCUser = psu_npc:user_init(NPCid, ((User#egs_user_model.character)#characters.mainlevel)#level.number), - %% Create and join party. - case User#egs_user_model.partypid of - undefined -> - {ok, PartyPid} = psu_party:start_link(GID), - send_022c(0, 16#12); - PartyPid -> - ignore - end, - {ok, PartyPos} = psu_party:join(PartyPid, npc, TmpNPCUser#egs_user_model.id), - NPCUser = TmpNPCUser#egs_user_model{lid=PartyPos, partypid=PartyPid}, - egs_user_model:write(NPCUser), - egs_user_model:write(User#egs_user_model{partypid=PartyPid}), - %% Send stuff. - Character = NPCUser#egs_user_model.character, - SentNPCCharacter = Character#characters{gid=NPCid, npcid=NPCid}, - SentNPCUser = NPCUser#egs_user_model{character=SentNPCCharacter}, - send_1004(npc_invite, SentNPCUser, PartyPos), - send_101a(NPCid, PartyPos); - -%% @todo Should be 0115(money) 010a03(confirm sale). -%% @todo We probably need to save the ShopID somewhere since it isn't given back here. -event({npc_shop_buy, ShopItemIndex, Quantity}) -> - log("npc shop buy itemindex ~p quantity ~p", [ShopItemIndex, Quantity]); - -%% @todo Currently send the normal items shop for all shops, differentiate. -event({npc_shop_enter, ShopID}) -> - log("npc shop enter ~p", [ShopID]), - case proplists:get_value(ShopID, ?SHOPS) of - undefined -> %% @todo Temporary; prevent players from getting stuck. - GID = get(gid), - {ok, File} = file:read_file("p/itemshop.bin"), - send(<< 16#010a0300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, - GID:32/little-unsigned-integer, 0:64, GID:32/little-unsigned-integer, 0:32, File/binary >>); - ItemsList -> - send_010a(ItemsList) - end; - -event({npc_shop_leave, ShopID}) -> - log("npc shop leave ~p", [ShopID]), - GID = get(gid), - send(<< 16#010a0300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, - GID:32/little-unsigned-integer, 0:64, GID:32/little-unsigned-integer, 0:32 >>); - -%% @todo Should be 0115(money) 010a03(confirm sale). -event({npc_shop_sell, InventoryItemIndex, Quantity}) -> - log("npc shop sell itemindex ~p quantity ~p", [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}) -> - log("npc shop request ~p", [ShopID]), - case ShopID of - 80 -> send_1a02(0, 17, 17, 3, 9); %% lumilass - 90 -> send_1a02(0, 5, 1, 4, 5); %% parum weapon grinding - 91 -> send_1a02(0, 5, 5, 4, 7); %% tenora weapon grinding - 92 -> send_1a02(0, 5, 0, 4, 0); %% yohmei weapon grinding - 93 -> send_1a02(0, 5, 18, 4, 0); %% kubara weapon grinding - _ -> send_1a02(0, 0, 1, 0, 0) - end; - -%% @todo Not sure what are those hardcoded values. -event({object_boss_gate_activate, ObjectID}) -> - send_1213(ObjectID, 0), - send_1215(2, 16#7008), - %% @todo Following sent after the warp? - send_1213(37, 0), - %% @todo Why resend this? - send_1213(ObjectID, 0); - -event({object_boss_gate_enter, ObjectID}) -> - send_1213(ObjectID, 1); - -%% @todo Do we need to send something back here? -event({object_boss_gate_leave, _ObjectID}) -> - ignore; - -%% @doc Server-side event. -event({object_box_destroy, ObjectID}) -> - send_1213(ObjectID, 3); - -%% @todo Second send_1211 argument should be User#egs_user_model.lid. Fix when it's correctly handled. -event({object_chair_sit, ObjectTargetID}) -> - %~ {ok, User} = egs_user_model:read(get(gid)), - send_1211(ObjectTargetID, 0, 8, 0); - -%% @todo Second send_1211 argument should be User#egs_user_model.lid. Fix when it's correctly handled. -event({object_chair_stand, ObjectTargetID}) -> - %~ {ok, User} = egs_user_model:read(get(gid)), - send_1211(ObjectTargetID, 0, 8, 2); - -event({object_crystal_activate, ObjectID}) -> - send_1213(ObjectID, 1); - -%% @doc Server-side event. -event({object_event_trigger, BlockID, EventID}) -> - send_1205(EventID, BlockID, 0); - -event({object_goggle_target_activate, ObjectID}) -> - {ok, User} = egs_user_model:read(get(gid)), - {BlockID, EventID} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), - send_1205(EventID, BlockID, 0), - send_1213(ObjectID, 8); - -event({object_key_console_enable, ObjectID}) -> - {ok, User} = egs_user_model:read(get(gid)), - {BlockID, [EventID|_]} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), - send_1205(EventID, BlockID, 0), - send_1213(ObjectID, 1); - -event({object_key_console_init, ObjectID}) -> - {ok, User} = egs_user_model:read(get(gid)), - {BlockID, [_, EventID, _]} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), - send_1205(EventID, BlockID, 0); - -event({object_key_console_open_gate, ObjectID}) -> - {ok, User} = egs_user_model:read(get(gid)), - {BlockID, [_, _, EventID]} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), - send_1205(EventID, BlockID, 0), - send_1213(ObjectID, 1); - -%% @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}) -> - {ok, User} = egs_user_model:read(get(gid)), - {BlockID, [EventID|_]} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), - send_1205(EventID, BlockID, 0), - send_1213(ObjectID, 1); - -%% @todo Some switch objects apparently work differently, like the light switch in Mines in MAG'. -event({object_switch_off, ObjectID}) -> - {ok, User} = egs_user_model:read(get(gid)), - {BlockID, EventID} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), - send_1205(EventID, BlockID, 1), - send_1213(ObjectID, 0); - -event({object_switch_on, ObjectID}) -> - {ok, User} = egs_user_model:read(get(gid)), - {BlockID, EventID} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), - send_1205(EventID, BlockID, 0), - send_1213(ObjectID, 1); - -event({object_vehicle_boost_enable, ObjectID}) -> - send_1213(ObjectID, 1); - -event({object_vehicle_boost_respawn, ObjectID}) -> - send_1213(ObjectID, 0); - -%% @todo Second send_1211 argument should be User#egs_user_model.lid. Fix when it's correctly handled. -event({object_warp_take, BlockID, ListNb, ObjectNb}) -> - {ok, User} = egs_user_model:read(get(gid)), - Pos = psu_instance:warp_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, BlockID, ListNb, ObjectNb), - NewUser = User#egs_user_model{pos=Pos}, - egs_user_model:write(NewUser), - send_0503(User#egs_user_model.pos), - send_1211(16#ffffffff, 0, 14, 0); - -event({party_remove_member, PartyPos}) -> - log("party remove member ~b", [PartyPos]), - {ok, DestUser} = egs_user_model:read(get(gid)), - {ok, RemovedGID} = psu_party:get_member(DestUser#egs_user_model.partypid, PartyPos), - psu_party:remove_member(DestUser#egs_user_model.partypid, PartyPos), - {ok, RemovedUser} = egs_user_model:read(RemovedGID), - case (RemovedUser#egs_user_model.character)#characters.type of - npc -> egs_user_model:delete(RemovedGID); - _ -> ignore - end, - send_1006(8, PartyPos), - send_0204(DestUser, RemovedUser, 1), - psu_proto:send_0215(DestUser, 0); - -event({player_options_change, Options}) -> - {ok, User} = egs_user_model:read(get(gid)), - file:write_file(io_lib:format("save/~s/~b-character.options", [User#egs_user_model.folder, (User#egs_user_model.character)#characters.slot]), Options); - -%% @todo If the player has a scape, use it! Otherwise red screen. -%% @todo Right now we force revive and don't update the player's HP. -event(player_death) -> - % @todo send_0115(get(gid), 16#ffffffff, LV=1, EXP=idk, Money=1000), % apparently sent everytime you die... - %% use scape: - NewHP = 10, - send_0117(NewHP), - send_1022(NewHP); - %% red screen with return to lobby choice: - %~ send_0111(3, 1); - -%% @todo Refill the player's HP to maximum, remove SEs etc. -event(player_death_return_to_lobby) -> - {ok, User} = egs_user_model:read(get(gid)), - PrevArea = User#egs_user_model.prev_area, - event({area_change, PrevArea#psu_area.questid, PrevArea#psu_area.zoneid, PrevArea#psu_area.mapid, User#egs_user_model.prev_entryid}); - -event(player_type_availability_request) -> - send_1a07(); - -event(player_type_capabilities_request) -> - send_0113(); - -event(ppcube_request) -> - send_1a04(); - -%% @doc Uni cube handler. -event(unicube_request) -> - send_021e(); - -%% @doc Uni selection handler. -%% @todo When selecting 'Your room', load a default room. -%% @todo When selecting 'Reload', reload the character in the current lobby. -%% @todo Delete NPC characters and stop the party on entering myroom too. -event({unicube_select, Selection, EntryID}) -> - case Selection of - cancel -> ignore; - 16#ffffffff -> - log("uni selection (my room)"), - send_0230(), - % 0220 - event({area_change, 1120000, 0, 100, 0}); - _UniID -> - log("uni selection (reload)"), - send_0230(), - % 0220 - %% force reloading the character and data files (@todo hack, uses myroom questid to do it) - {ok, User} = egs_user_model:read(get(gid)), - case User#egs_user_model.partypid of - undefined -> - ignore; - PartyPid -> - %% @todo Replace stop by leave when leaving stops the party correctly when nobody's there anymore. - %~ psu_party:leave(User#egs_user_model.partypid, User#egs_user_model.id) - {ok, NPCList} = psu_party:get_npc(PartyPid), - [egs_user_model:delete(NPCGID) || {_Spot, NPCGID} <- NPCList], - psu_party:stop(PartyPid) - end, - Area = User#egs_user_model.area, - NewRow = User#egs_user_model{partypid=undefined, area=Area#psu_area{questid=1120000, zoneid=undefined}, entryid=EntryID}, - egs_user_model:write(NewRow), - event({area_change, Area#psu_area.questid, Area#psu_area.zoneid, Area#psu_area.mapid, EntryID}) - end. - - - - - -%% @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. -handle(16#0402, Data) -> - << SpawnID:32/little-unsigned-integer, _:64, Type:32/little-unsigned-integer, _:64 >> = Data, - case Type of - 7 -> % spawn cleared @todo 1201 sent back with same values apparently, but not always - log("cleared spawn ~b", [SpawnID]), - {ok, User} = egs_user_model:read(get(gid)), - {BlockID, EventID} = psu_instance:spawn_cleared_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, SpawnID), - if EventID =:= false -> ignore; - true -> send_1205(EventID, BlockID, 0) - end; - _ -> - ignore - end; - -%% @todo Handle this packet. -%% @todo 3rd Unsafe Passage C, EventID 10 BlockID 2 = mission cleared? -handle(16#0404, Data) -> - << EventID:8, BlockID:8, _:16, Value:8, _/bits >> = Data, - log("unknown command 0404: eventid ~b blockid ~b value ~b", [EventID, BlockID, Value]), - send_1205(EventID, BlockID, Value); - -%% @todo Used in the tutorial. Not sure what it does. Give an item (the PA) maybe? -handle(16#0a09, Data) -> - log("0a09 ~p", [Data]), - GID = get(gid), - send(<< 16#0a090300:32, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00003300:32, 0:32 >>); - -%% @todo Figure out this command. -handle(16#0c11, << A:32/little, B:32/little >>) -> - log("0c11 ~p ~p", [A, B]), - GID = get(gid), - send(<< 16#0c120300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 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. -handle(16#0d04, Data) -> - << Flag:128/bits, A:16/bits, _:8, B/bits >> = Data, - log("flag handler for ~s", [re:replace(Flag, "\\0+", "", [global, {return, binary}])]), - GID = get(gid), - send(<< 16#0d040300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, Flag/binary, A/binary, 1, B/binary >>); - -%% @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. -handle(16#0f00, Data) -> - << A:32/little-unsigned-integer, 0:16, B:16/little-unsigned-integer, 0:16, C:16/little-unsigned-integer, 0, Whut:8, D:16/little-unsigned-integer, 0:16, - E:16/little-unsigned-integer, 0:16, F:16/little-unsigned-integer, G:16/little-unsigned-integer, H:16/little-unsigned-integer, I:32/little-unsigned-integer >> = Data, - log("init vehicle: ~b ~b ~b ~b ~b ~b ~b ~b ~b ~b", [A, B, C, Whut, D, E, F, G, H, I]), - send(<< (header(16#1208))/binary, A:32/little-unsigned-integer, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, - 0:16, B:16/little-unsigned-integer, 0:16, C:16/little-unsigned-integer, 0:16, D:16/little-unsigned-integer, 0:112, - E:16/little-unsigned-integer, 0:16, F:16/little-unsigned-integer, H:16/little-unsigned-integer, 1, 0, 100, 0, 10, 0, G:16/little-unsigned-integer, 0:16 >>); - -%% @doc Enter vehicle. -%% @todo Separate the reply. -handle(16#0f02, Data) -> - << A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer >> = Data, - log("enter vehicle: ~b ~b ~b", [A, B, C]), - HP = 100, - send(<< (header(16#120a))/binary, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer, HP:32/little-unsigned-integer >>); - -%% @doc Sent right after entering the vehicle. Can't move without it. -%% @todo Separate the reply. -handle(16#0f07, Data) -> - << A:32/little-unsigned-integer, B:32/little-unsigned-integer >> = Data, - log("after enter vehicle: ~b ~b", [A, B]), - send(<< (header(16#120f))/binary, A:32/little-unsigned-integer, B:32/little-unsigned-integer >>); - -%% @todo Not sure yet. -handle(16#1019, _) -> - ignore; - %~ send(<< (header(16#1019))/binary, 0:192, 16#00200000:32, 0:32 >>); - -%% @todo Not sure about that one though. Probably related to 1112 still. -handle(16#1106, Data) -> - send_110e(Data); - -%% @doc Probably asking permission to start the video (used for syncing?). -handle(16#1112, Data) -> - send_1113(Data); - -%% @todo Not sure yet. Value is probably a TargetID. Used in Airboard Rally. Replying with the same value starts the race. -handle(16#1216, Data) -> - << Value:32/little-unsigned-integer >> = Data, - log("command 1216 with value ~b", [Value]), - send_1216(Value); - -%% @doc Unknown command handler. Do nothing. -handle(Command, _) -> - log("dismissed packet ~4.16.0b", [Command]). - %% @doc Build the packet header. header(Command) -> GID = get(gid), @@ -1759,11 +869,3 @@ send_1a04() -> send_1a07() -> send(<< (header(16#1a07))/binary, 16#085b5d0a:32, 16#3a200000:32, 0:32, 16#01010101:32, 16#01010101:32, 16#01010101:32, 16#01010101:32 >>). - -%% @doc Log message to the console. -log(Message) -> - io:format("game (~p): ~s~n", [get(gid), Message]). - -log(Message, Format) -> - FormattedMessage = io_lib:format(Message, Format), - log(FormattedMessage). diff --git a/src/psu/psu_proto.erl b/src/psu/psu_proto.erl index 00a02a8..a28e600 100644 --- a/src/psu/psu_proto.erl +++ b/src/psu/psu_proto.erl @@ -241,7 +241,6 @@ parse(Size, 16#020d, Channel, Data) -> ?ASSERT_EQ(VarK, 0), {system_key_auth_request, AuthGID, AuthKey}; -%% @doc This command should be safely ignored. Probably indicates that character loading was successful. 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), @@ -255,7 +254,7 @@ parse(Size, 16#021c, Channel, Data) -> ?ASSERT_EQ(VarG, 0), ?ASSERT_EQ(VarH, 0), ?ASSERT_EQ(VarI, 0), - ignore; + 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, @@ -1078,11 +1077,9 @@ parse(Size, 16#1a01, Channel, Data) -> _ -> log("unknown 1a01 EventID ~p", [EventID]) end; -parse(_Size, Command, Channel, Data) -> - %% @todo log unknown command? - %~ ignore. - << _:288, Rest/bits >> = Data, - {command, Command, Channel, Rest}. +%% @doc Unknown command, +parse(_Size, Command, Channel, _Data) -> + {command, Command, Channel}. %% @todo Many unknown vars in the hit values. parse_hits(<< >>, Acc) ->