egs/src/egs_game.erl

1508 lines
55 KiB
Erlang
Raw Normal View History

% EGS: Erlang Game Server
% Copyright (C) 2010 Loic Hoguin
%
% This file is part of EGS.
%
% EGS is free software: you can redistribute it and/or modify
% it under the terms of the GNU General Public License as published by
% the Free Software Foundation, either version 3 of the License, or
% (at your option) any later version.
%
% EGS is distributed in the hope that it will be useful,
% but WITHOUT ANY WARRANTY; without even the implied warranty of
% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
% GNU General Public License for more details.
%
% You should have received a copy of the GNU General Public License
% along with EGS. If not, see <http://www.gnu.org/licenses/>.
-module(egs_game).
-export([start/0]). % external
-export([supervisor_init/0, supervisor/0, listen/1, accept/2, process_init/2, process/0, char_select/0, area_load/4, loop/1]). % internal
-include("include/records.hrl").
-include("include/network.hrl").
-include("include/maps.hrl").
%% @doc Start the game server.
start() ->
SPid = spawn(?MODULE, supervisor_init, []),
LPid = spawn(?MODULE, listen, [SPid]),
[{listener, LPid}, {supervisor, SPid}].
%% @doc Game processes supervisor initialization.
supervisor_init() ->
process_flag(trap_exit, true),
supervisor().
%% @doc Game processes supervisor. Make sure everything is cleaned up when an unexpected error occurs.
supervisor() ->
receive
{'EXIT', Pid, _} ->
supervisor_close(Pid);
_ ->
reload
after 5000 ->
reload
end,
?MODULE:supervisor().
%% @doc Close the connection for the given user and cleanup.
supervisor_close(Pid) ->
try
User = egs_db:users_select_by_pid(Pid),
egs_db:users_delete(User#users.gid),
lists:foreach(fun(Other) -> Other#users.pid ! {psu_player_unspawn, User} end, egs_db:users_select_others_in_area(User)),
io:format("game (~p): quit~n", [User#users.gid])
catch _:_ ->
ignore
end.
%% @doc Listen for connections.
listen(SPid) ->
{ok, LSocket} = ssl:listen(?GAME_PORT, ?GAME_LISTEN_OPTIONS),
?MODULE:accept(LSocket, SPid).
%% @doc Accept connections.
accept(LSocket, SPid) ->
case ssl:transport_accept(LSocket, 5000) of
{ok, CSocket} ->
ssl:ssl_accept(CSocket),
Pid = spawn(?MODULE, process_init, [CSocket, SPid]),
ssl:controlling_process(CSocket, Pid);
_ ->
reload
end,
?MODULE:accept(LSocket, SPid).
%% @doc Initialize the client process by saving the socket to the process dictionary.
process_init(CSocket, SPid) ->
link(SPid),
put(socket, CSocket),
send_0202(),
process().
%% @doc Process the new connections.
%% Send an hello packet, authenticate the user and send him to character select.
process() ->
case egs_proto:packet_recv(get(socket), 5000) of
{ok, Orig} ->
{command, Command, _, Data} = egs_proto:packet_parse(Orig),
process_handle(Command, Data);
{error, timeout} ->
reload,
?MODULE:process();
{error, closed} ->
2010-06-04 05:36:34 +08:00
closed
end.
%% @doc Game server auth request handler. Save the GID in the process dictionary after checking it.
process_handle(16#020d, << GID:32/little-unsigned-integer, Auth:32/bits, _/bits >>) ->
CSocket = get(socket),
case egs_db:users_select(GID) of
error ->
log("can't find user, closing"),
ssl:close(CSocket);
User ->
case User#users.auth of
Auth ->
put(gid, GID),
log("auth success"),
LID = 1 + egs_db:next(lobby) rem 1023,
Time = calendar:datetime_to_gregorian_seconds(calendar:universal_time()),
egs_db:users_insert(#users{gid=GID, pid=self(), socket=CSocket, auth=success, time=Time, folder=User#users.folder, lid=LID}),
send_0d05(),
?MODULE:char_select();
_ ->
log("quit, auth failed"),
egs_db:users_delete(GID),
ssl:close(CSocket)
end
end;
%% @doc Platform information handler. Obtain the game version and save it into the process dictionary.
process_handle(16#080e, << _:64, Version:32/little-unsigned-integer, _/bits >>) ->
put(version, Version),
?MODULE:process();
%% @doc Unknown command handler. Do nothing.
process_handle(Command, _) ->
log("(process) dismissed packet ~4.16.0b", [Command]),
?MODULE:process().
%% @doc Character selection screen loop.
char_select() ->
case egs_proto:packet_recv(get(socket), 5000) of
{ok, Orig} ->
{command, Command, _, Data} = egs_proto:packet_parse(Orig),
char_select_handle(Command, Data);
{error, timeout} ->
egs_proto:send_keepalive(get(socket)),
reload,
?MODULE:char_select();
{error, closed} ->
log("quit"),
egs_db:users_delete(get(gid))
end.
%% @doc Character selection handler.
char_select_handle(16#020b, << Number:32/little-unsigned-integer, _/bits >>) ->
log("selected character ~b", [Number]),
char_select_load(Number);
%% @doc Character creation handler.
char_select_handle(16#0d02, << Number:32/little-unsigned-integer, Char/bits >>) ->
log("character creation"),
% check for valid character appearance
<< _Name:512, RaceID:8, GenderID:8, _TypeID:8, AppearanceBin:776/bits, _/bits >> = Char,
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
User = egs_db:users_select(get(gid)),
_ = file:make_dir(io_lib:format("save/~s", [User#users.folder])),
file:write_file(io_lib:format("save/~s/~b-character", [User#users.folder, Number]), Char),
2010-07-10 02:40:44 +08:00
file:write_file(io_lib:format("save/~s/~b-character.options", [User#users.folder, Number]), << 0:128, 4, 0:56 >>), % default 0 to everything except brightness 4
char_select_load(Number);
%% @doc Character selection screen request.
char_select_handle(16#0d06, _) ->
User = egs_db:users_select(get(gid)),
send_0d03(data_load(User#users.folder, 0), data_load(User#users.folder, 1), data_load(User#users.folder, 2), data_load(User#users.folder, 3)),
?MODULE:char_select();
%% @doc Silently ignore packet 0818. Gives CPU/GPU information.
char_select_handle(16#0818, _) ->
?MODULE:char_select();
%% @doc Unknown command handler. Do nothing.
char_select_handle(Command, _) ->
log("(char_select) dismissed packet ~4.16.0b", [Command]),
?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) ->
OldUser = egs_db:users_select(get(gid)),
[{status, 1}, {char, CharBin}, {options, OptionsBin}] = data_load(OldUser#users.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
User = OldUser#users{character=Character},
egs_db:users_insert(User),
char_load(User),
send_021b(),
area_load(1100000, 0, 4, 6),
ssl:setopts(get(socket), [{active, true}]),
?MODULE:loop(<< >>).
%% @doc Load the given character's data.
2010-06-08 23:04:54 +08:00
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.
char_load(User) ->
send_0d01(User),
% 0246
send_0a0a(),
send_1006(5),
send_1005((User#users.character)#characters.name),
send_1006(12),
send_0210(),
send_0222(),
send_1500(User),
send_1501(),
send_1512(),
2010-06-04 23:02:13 +08:00
% 0303
send_1602().
%% @doc Load the given map as a mission counter.
counter_load(QuestID, ZoneID, MapID, EntryID) ->
OldUser = egs_db:users_select(get(gid)),
User = OldUser#users{areatype=counter, questid=QuestID, zoneid=ZoneID, mapid=MapID, entryid=EntryID,
savedquestid=OldUser#users.questid, savedzoneid=OldUser#users.zoneid, savedmapid=ZoneID, savedentryid=MapID},
egs_db:users_insert(User),
AreaName = "Mission counter",
QuestFile = "data/lobby/counter.quest.nbl",
ZoneFile = "data/lobby/counter.zone.nbl",
% broadcast unspawn to other people
lists:foreach(fun(Other) -> Other#users.pid ! {psu_player_unspawn, User} end, egs_db:users_select_others_in_area(OldUser)),
% load counter
send_0c00(16#7fffffff),
send_020e(QuestFile),
send_0a05(),
send_010d(User),
send_0200(mission),
send_020f(ZoneFile, 0, 16#ff),
send_0205(0, 0, 0, 0),
send_100e(16#7fffffff, 0, 0, AreaName, EntryID),
send_0215(0),
send_0215(0),
send_020c(),
send_1202(),
send_1204(),
send_1206(),
send_1207(),
send_1212(),
send_0201(User),
send_0a06(),
send_0208(),
send_0236().
%% @doc Return the current season information.
area_get_season(QuestID) ->
{{_, Month, Day}, _} = calendar:universal_time(),
[IsSeasonal, SeasonID, SeasonQuestIDs] = if
Month =:= 1, Day =< 14 -> ?SEASON_NEWYEAR;
Month =:= 1, Day >= 25 -> ?SEASON_WINTER;
Month =:= 2, Day =< 7 -> ?SEASON_WINTER;
Month =:= 2, Day >= 14 -> ?SEASON_VALENTINE;
Month =:= 3, Day =< 6 -> ?SEASON_VALENTINE;
Month =:= 3, Day >= 14 -> ?SEASON_WHITEDAY;
Month =:= 4, Day =< 3 -> ?SEASON_WHITEDAY;
Month =:= 4, Day >= 4, Day =< 24 -> ?SEASON_EASTER;
Month =:= 4, Day >= 25 -> ?SEASON_SPRING;
Month =:= 5, Day =< 8 -> ?SEASON_SPRING;
Month =:= 5, Day >= 17, Day =< 30 -> ?SEASON_WEDDING;
Month =:= 6, Day >= 3, Day =< 16 -> ?SEASON_PARUMUNIF;
Month =:= 6, Day >= 23 -> ?SEASON_SONIC;
Month =:= 7, Day =< 13 -> ?SEASON_SONIC;
Month =:= 7, Day >= 18 -> ?SEASON_HOLYLIGHT;
Month =:= 8, Day =< 21 -> ?SEASON_FIREWORKS;
Month =:= 8, Day >= 28 -> ?SEASON_NATIVE;
Month =:= 9, Day =< 10 -> ?SEASON_NATIVE;
Month =:= 9, Day >= 24 -> ?SEASON_AUTUMN;
Month =:= 10, Day =< 7 -> ?SEASON_AUTUMN;
Month =:= 10, Day >= 15, Day =< 28 -> ?SEASON_PARTY;
Month =:= 10, Day >= 31 -> ?SEASON_HALLOWEEN;
Month =:= 11, Day =< 20 -> ?SEASON_HALLOWEEN;
Month =:= 12, Day >= 11 -> ?SEASON_CHRISTMAS;
true -> ?SEASON_NONE
end,
if IsSeasonal =:= 1 ->
case lists:member(QuestID, SeasonQuestIDs) of
true -> [{status, IsSeasonal}, {season, SeasonID}];
false -> [{status, 0}, {season, 255}]
end;
true ->
[{status, 0}, {season, 255}]
end.
%% @doc Load the given map as a standard lobby.
area_load(QuestID, ZoneID, MapID, EntryID) ->
OldUser = egs_db:users_select(get(gid)),
[{type, AreaType}, {file, QuestFile}|StartInfo] = proplists:get_value(QuestID, ?QUESTS, [{type, undefined}, {file, undefined}]),
[IsStart, InstanceID, RealZoneID, RealMapID, RealEntryID] = case AreaType of
mission ->
if ZoneID =:= 65535 ->
[{start, [TmpZoneID, TmpMapID, TmpEntryID]}] = StartInfo,
[true, OldUser#users.gid, TmpZoneID, TmpMapID, TmpEntryID];
true -> [false, OldUser#users.gid, ZoneID, MapID, EntryID]
end;
myroom ->
if ZoneID =:= 0 ->
[false, undefined, 0, 423, EntryID];
true -> [false, undefined, ZoneID, MapID, EntryID]
end;
_ ->
[false, undefined, ZoneID, MapID, EntryID]
end,
[{file, ZoneFile}] = proplists:get_value([QuestID, RealZoneID], ?ZONES, [{file, undefined}]),
if AreaType =:= myroom ->
AreaName = "Your Room";
true ->
[{name, AreaName}] = proplists:get_value([QuestID, RealMapID], ?MAPS, [{name, "dammy"}])
end,
User = OldUser#users{instanceid=InstanceID, areatype=AreaType, questid=QuestID, zoneid=RealZoneID, mapid=RealMapID, entryid=RealEntryID},
egs_db:users_insert(User),
%% @todo Don't recalculate SetID when leaving the mission and back (this changes the spawns).
SetID = if IsStart =:= true -> crypto:rand_uniform(0, 4); true -> 0 end,
if IsStart =:= true -> % initialize the mission
psu_missions:start(InstanceID, QuestID, SetID);
true -> ignore
end,
area_load(AreaType, IsStart, SetID, OldUser, User, QuestFile, ZoneFile, AreaName).
area_load(AreaType, IsStart, SetID, OldUser, User, QuestFile, ZoneFile, AreaName) ->
QuestChange = if OldUser#users.questid /= User#users.questid, QuestFile /= undefined -> true; true -> false end,
if ZoneFile =:= undefined ->
ZoneChange = false;
true ->
ZoneChange = if OldUser#users.questid =:= User#users.questid, OldUser#users.zoneid =:= User#users.zoneid -> false; true -> true end
end,
[{status, IsSeasonal}, {season, SeasonID}] = area_get_season(User#users.questid),
% broadcast spawn and unspawn to other people
lists:foreach(fun(Other) -> Other#users.pid ! {psu_player_unspawn, User} end, egs_db:users_select_others_in_area(OldUser)),
if AreaType =:= lobby ->
lists:foreach(fun(Other) -> Other#users.pid ! {psu_player_spawn, User} end, egs_db:users_select_others_in_area(User));
true -> ignore
end,
% load area
if QuestChange =:= true ->
% reload the character if entering or leaving the room quest
if OldUser#users.questid =:= 1120000; User#users.questid =:= 1120000 ->
char_load(User);
true -> ignore
end,
% load new quest
send_0c00(User#users.questid),
send_020e(QuestFile);
true -> ignore
end,
if IsStart =:= true ->
send_0215(16#ffffffff);
true -> ignore
end,
if ZoneChange =:= true ->
% load new zone
send_0a05(),
if AreaType =:= lobby ->
send_0111(6, 0);
true -> ignore
end,
send_010d(User),
send_0200(AreaType),
send_020f(ZoneFile, SetID, SeasonID);
true -> ignore
end,
send_0205(User#users.zoneid, User#users.mapid, User#users.entryid, IsSeasonal),
send_100e(User#users.questid, User#users.zoneid, User#users.mapid, AreaName, 16#ffffffff),
if AreaType =:= mission ->
send_0215(0),
if IsStart =:= true ->
send_0215(0),
send_0c09();
true -> ignore
end;
true ->
send_020c()
end,
case AreaType of
myroom ->
myroom_send_packet("p/packet1332.bin"),
send_1202(),
send_1204(),
send_1206();
mission ->
send_1202(),
send_1204(),
send_1206(),
send_1207();
_ -> ignore
end,
if AreaType /= spaceport ->
send_1212();
true -> ignore
end,
if AreaType =:= myroom ->
myroom_send_packet("p/packet1309.bin");
true -> ignore
end,
send_0201(User),
if ZoneChange =:= true ->
send_0a06();
true -> ignore
end,
send_0233(egs_db:users_select_others_in_area(User)),
send_0208(),
send_0236().
myroom_send_packet(Filename) ->
{ok, << _:32, File/bits >>} = file:read_file(Filename),
send(File).
%% @doc Game's main loop.
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, ChatGID, ChatName, ChatModifiers, ChatMessage} ->
send_0304(ChatGID, ChatName, ChatModifiers, ChatMessage),
?MODULE:loop(SoFar);
{psu_keepalive} ->
egs_proto:send_keepalive(get(socket)),
?MODULE:loop(SoFar);
2010-06-28 18:10:31 +08:00
{psu_player_spawn, _Spawn} ->
% Should be something along the lines of 203 201 204 or something.
send_0233(egs_db:users_select_others_in_area(egs_db:users_select(get(gid)))),
?MODULE:loop(SoFar);
{psu_player_unspawn, Spawn} ->
send_0204(Spawn#users.gid, Spawn#users.lid, 5),
?MODULE:loop(SoFar);
{psu_warp, QuestID, ZoneID, MapID, EntryID} ->
area_load(QuestID, ZoneID, MapID, EntryID),
?MODULE:loop(SoFar);
{ssl, _, Data} ->
{Packets, Rest} = egs_proto:packet_split(<< SoFar/bits, Data/bits >>),
[dispatch(Orig) || Orig <- Packets],
?MODULE:loop(Rest);
{ssl_closed, _} ->
exit(ssl_closed);
{ssl_error, _, _} ->
exit(ssl_error);
_ ->
?MODULE:loop(SoFar)
after 1000 ->
reload,
?MODULE:loop(SoFar)
end.
%% @doc Dispatch the command to the right handler.
dispatch(Orig) ->
{command, Command, Channel, Data} = egs_proto:packet_parse(Orig),
case Channel of
ignore -> ignore;
1 -> broadcast(Command, Orig);
_ -> handle(Command, Data)
end.
%% @doc Position change broadcast handler. Save the position and then dispatch it.
broadcast(16#0503, Orig) ->
<< 100:32/little-unsigned-integer, 16#050301:24/unsigned-integer, _:360, Direction:32/bits, Coords:96/bits, _:96,
QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, MapID:32/little-unsigned-integer,
EntryID:32/little-unsigned-integer, _:32 >> = Orig,
User = egs_db:users_select(get(gid)),
NewUser = User#users{direction=Direction, coords=Coords, questid=QuestID, zoneid=ZoneID, mapid=MapID, entryid=EntryID},
egs_db:users_insert(NewUser),
broadcast(default, Orig);
%% @doc Stand still broadcast handler. Save the position and then dispatch it.
broadcast(16#0514, Orig) ->
<< _:320, _:96, Direction:32/bits, Coords:96/bits, QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer,
MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer, _/bits >> = Orig,
User = egs_db:users_select(get(gid)),
NewUser = User#users{direction=Direction, coords=Coords, questid=QuestID, zoneid=ZoneID, mapid=MapID, entryid=EntryID},
egs_db:users_insert(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_db:users_select(GID) of
error ->
ignore;
Self ->
LID = Self#users.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 >>,
lists:foreach(fun(User) -> User#users.pid ! {psu_broadcast, Packet} end, egs_db:users_select_others_in_area(Self))
end.
%% @doc Movement (non-broadcast) handler. Do nothing.
handle(16#0102, _) ->
ignore;
%% @doc Weapon equip, unequip, item drop, and more... handler. Do what we can.
%% 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.
handle(16#0105, Data) ->
<< _:32, A:32/little-unsigned-integer, ItemID:8, Action:8, _:8, B:8, C:32/little-unsigned-integer, _/bits >> = Data,
log("0105 action ~b item ~b (~b ~b ~b)", [Action, ItemID, A, B, C]),
GID = get(gid),
Category = case ItemID 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,
case Action of
1 -> % equip item
Filename = case ItemID of
% weapons
16 -> "p/packet0105_sword.bin";
13 -> "p/packet0105_twindaggers.bin";
15 -> "p/packet0105_dagger.bin";
9 -> "p/packet0105_rcsm.bin";
14 -> "p/packet0105_saber.bin";
8 -> "p/packet0105_mgun.bin";
X when X =:= 17; X =:= 18 ->
"p/packet0105_twinguns.bin";
% armor
19 -> "p/packet0105_armor.bin";
% clothes
X when X =:= 5; X =:= 6; X =:= 7 ->
none;
_ -> % default: do nothing
none
end,
case Filename of
none -> File = << >>;
_ -> {ok, File} = file:read_file(Filename)
end,
send(<< 16#01050300: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, A:32/little-unsigned-integer, ItemID, Action, Category, B, C:32/little-unsigned-integer,
File/binary >>);
2 -> % unequip item
send(<< 16#01050300: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, A:32/little-unsigned-integer, ItemID, Action, Category, B, C:32/little-unsigned-integer >>);
5 -> % drop item
ignore;
_ ->
ignore
end;
2010-06-04 18:23:18 +08:00
%% @doc Shop listing request. Currently return the normal item shop for everything.
%% @todo Return the other shops appropriately.
handle(16#010a, Data) ->
<< _:32, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer >> = Data,
log("shop listing request (~b, ~b, ~b)", [A, B, C]),
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 >>);
%% @doc Character death, and more, handler. Warp to 4th floor for now.
%% @todo Recover from death correctly.
handle(16#0110, Data) ->
<< _:32, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer >> = Data,
2010-06-14 05:42:56 +08:00
case B of
2 -> % triggered when looking at the type menu
send_0113();
3 -> % type change
log("changed type to ~b", [C]);
7 -> % player death: if the player has a scape, use it! otherwise red screen @todo Right now we force revive and don't reset the HP.
% @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);
8 -> % return to lobby after death
log("return to lobby after death"),
area_load(1100000, 0, 4, 6); %% @todo temporary handler
2010-06-14 05:42:56 +08:00
10 -> % online status change
log("changed status to ~b", [C]);
2010-06-14 05:42:56 +08:00
_ ->
log("unknown 0110 (~b, ~b, ~b)", [A, B, C])
2010-06-14 05:42:56 +08:00
end;
%% @doc Uni cube handler.
handle(16#021d, _) ->
send_021e();
%% @doc Uni selection handler.
%% When selecting 'Your room', load a default room.
%% When selecting 'Reload', reload the character in the current lobby.
%% @todo There's probably an entryid given in the uni selection packet.
handle(16#021f, << Uni:32/little-unsigned-integer, _/bits >>) ->
2010-06-04 17:23:15 +08:00
case Uni of
0 -> % cancelled uni selection
2010-06-04 05:36:34 +08:00
ignore;
2010-06-04 17:23:15 +08:00
16#ffffffff ->
log("uni selection (my room)"),
send_0230(),
% 0220
area_load(1120000, 0, 100, 0);
_ ->
log("uni selection (reload)"),
send_0230(),
% 0220
% force reloading the character and data files (hack)
User = egs_db:users_select(get(gid)),
NewRow = User#users{questid=1120000, zoneid=undefined},
egs_db:users_insert(NewRow),
area_load(User#users.questid, User#users.zoneid, User#users.mapid, User#users.entryid)
end;
%% @doc Shortcut changes handler. Do nothing.
%% @todo Save it.
handle(16#0302, _) ->
log("dismissed shortcut changes");
%% @doc Chat broadcast handler. Dispatch the message to everyone (for now).
%% We must take extra precautions to handle different versions of the game correctly.
%% Disregard the name sent by the server in later versions of the game. Use the name saved in memory instead, to prevent client-side editing.
%% @todo Only broadcast to people in the same map.
handle(16#0304, Data) ->
User = egs_db:users_select(get(gid)),
case get(version) of
2010-06-04 18:18:12 +08:00
0 -> % AOTI v2.000
<< _:64, Modifiers:128/bits, Message/bits >> = Data;
2010-06-04 18:18:12 +08:00
_ -> % Above
<< _:64, Modifiers:128/bits, _:512, Message/bits >> = Data
end,
[LogName|_] = re:split((User#users.character)#characters.name, "\\0\\0", [{return, binary}]),
2010-06-04 18:18:12 +08:00
[TmpMessage|_] = re:split(Message, "\\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}])]]),
lists:foreach(fun(X) -> X#users.pid ! {psu_chat, get(gid), (User#users.character)#characters.name, Modifiers, Message} end, egs_db:users_select_all());
%% @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.
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]),
User = egs_db:users_select(get(gid)),
[EventID, BlockID] = psu_missions:spawn_cleared(User#users.instanceid, 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);
%% @doc Map change handler.
%% Rooms are handled differently than normal lobbies.
%% @todo When changing lobby to the room, 0230 must also be sent. Same when going from room to lobby.
handle(16#0807, Data) ->
<< QuestID:32/little-unsigned-integer, ZoneID:16/little-unsigned-integer,
MapID:16/little-unsigned-integer, EntryID:16/little-unsigned-integer, _/bits >> = Data,
log("map change (~b,~b,~b,~b)", [QuestID, ZoneID, MapID, EntryID]),
area_load(QuestID, ZoneID, MapID, EntryID);
2010-05-24 19:10:05 +08:00
%% @doc Mission counter handler.
handle(16#0811, Data) ->
<< QuestID:32/little-unsigned-integer, ZoneID:16/little-unsigned-integer,
MapID:16/little-unsigned-integer, EntryID:16/little-unsigned-integer, _/bits >> = Data,
log("mission counter (~b,~b,~b,~b)", [QuestID, ZoneID, MapID, EntryID]),
counter_load(QuestID, ZoneID, MapID, EntryID);
%% @doc Leave mission counter handler. Lobby values depend on which counter was entered.
handle(16#0812, _) ->
User = egs_db:users_select(get(gid)),
area_load(User#users.savedquestid, User#users.savedzoneid, User#users.zoneid, User#users.mapid);
2010-06-14 21:53:51 +08:00
%% @doc Item description request.
%% @todo Send something other than just "dammy".
handle(16#0a10, << ItemID:32/unsigned-integer >>) ->
send_0a11(ItemID, "dammy");
2010-06-14 21:53:51 +08:00
%% @doc Start mission handler.
%% @todo Forward the mission start to other players of the same party, whatever their location is.
handle(16#0c01, << QuestID:32/little-unsigned-integer >>) ->
log("start mission ~b", [QuestID]),
send_170c(),
send_1020(),
send_1015(QuestID),
send_0c02();
%% @doc Counter quests files request handler? Send huge number of quest files.
%% @todo Handle correctly.
handle(16#0c05, _) ->
User = egs_db:users_select(get(gid)),
[{quests, Filename}, {bg, _}, {options, _}] = proplists:get_value(User#users.entryid, ?COUNTERS),
send_0c06(Filename);
%% @doc Lobby transport handler? Just ignore the meseta price for now and send the player where he wanna be!
%% @todo Handle correctly.
handle(16#0c07, _) ->
send_0c08(true);
%% @doc Abort mission handler.
%% Replenish the player HP.
handle(16#0c0e, _) ->
send_1006(11),
User = egs_db:users_select(get(gid)),
%% full hp
Character = User#users.character,
MaxHP = Character#characters.maxhp,
NewCharacter = Character#characters{currenthp=MaxHP},
NewUser = User#users{character=NewCharacter},
egs_db:users_insert(NewUser),
%% map change (temporary)
if User#users.areatype =:= mission ->
area_load(User#users.savedquestid, User#users.savedzoneid, User#users.savedmapid, User#users.savedentryid);
true -> ignore
end;
%% @doc Counter available mission list request handler.
%% @todo Temporarily allow rare mission and LL all difficulties to all players.
handle(16#0c0f, _) ->
User = egs_db:users_select(get(gid)),
[{quests, _}, {bg, _}, {options, Options}] = proplists:get_value(User#users.entryid, ?COUNTERS),
send_0c10(Options);
2010-05-24 19:10:05 +08:00
2010-05-24 19:30:36 +08:00
%% @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 Options changes handler.
handle(16#0d07, Data) ->
log("options changes"),
% Translate options into a tuple, validate them and do nothing with it for now
Options = psu_characters:options_binary_to_tuple(Data),
psu_characters:validate_options(Options),
% End of validation
User = egs_db:users_select(get(gid)),
file:write_file(io_lib:format("save/~s/~b-character.options", [User#users.folder, (User#users.character)#characters.slot]), Data);
%% @doc Hit handler.
%% @todo Finish the work on it.
%% @todo First value at 2C is the number of hits. We don't need to know it though.
handle(16#0e00, Data) ->
<< _:96, Hits/bits >> = Data,
handle_hits(Hits);
2010-06-14 21:38:57 +08:00
%% @doc Object event handler.
%% @todo Handle all events appropriately.
handle(16#0f0a, Data) ->
<< _:48, _MapID:16/little-unsigned-integer, ObjectID:16/little-unsigned-integer, _:16, A:32/little-unsigned-integer, _:64, B:32/little-unsigned-integer, _:272, Action:8, _/bits >> = Data,
log("object event handler: action ~b object ~b", [Action, ObjectID]),
2010-06-14 21:38:57 +08:00
case Action of
0 -> % warp
ignore;
2010-06-19 00:57:10 +08:00
3 -> % crystal activation
send_1213(ObjectID, 1);
12 -> % pick/use key
User = egs_db:users_select(get(gid)),
[[EventID|_], BlockID] = psu_missions:key_event(User#users.instanceid, ObjectID),
send_1205(EventID, BlockID, 0),
send_1213(ObjectID, 1);
2010-06-14 21:38:57 +08:00
13 -> % button on
ignore;
14 -> % button off
ignore;
%~ 19 -> % @todo (somewhere in phantom ruins block 4)
%~ ignore;
20 -> % enter counter/elevator/room/spaceport/pick key/use key
2010-06-14 21:38:57 +08:00
ignore;
23 -> % initialize key slots (called when picking a key or checking the gate directly with no key)
User = egs_db:users_select(get(gid)),
[[_, EventID, _], BlockID] = psu_missions:key_event(User#users.instanceid, ObjectID),
send_1205(EventID, BlockID, 0); % in block 1, 202 = key [1] x1, 203 = key [-] x1
24 -> % open gate (only when client has key)
User = egs_db:users_select(get(gid)),
[[_, _, EventID], BlockID] = psu_missions:key_event(User#users.instanceid, ObjectID),
send_1205(EventID, BlockID, 0),
send_1213(ObjectID, 1);
2010-06-14 21:38:57 +08:00
25 -> % sit on chair
send_1211(A, B, 8, 0);
2010-06-14 21:38:57 +08:00
26 -> % sit out of chair
send_1211(A, B, 8, 2);
2010-06-14 21:38:57 +08:00
%~ 30 -> % @todo (phantom ruins block 4)
%~ ignore;
_ ->
log("object event ~b", [Action])
2010-06-14 21:38:57 +08:00
end;
%% @doc Party information recap request.
%% @todo Handle when the party already exists! And stop doing it wrong.
handle(16#1705, _) ->
User = egs_db:users_select(get(gid)),
send_1706((User#users.character)#characters.name);
%% @doc Mission selected handler. Send the currently selected mission.
%% @todo Probably need to dispatch that info to other party members in the same counter.
handle(16#1707, _) ->
ignore;
%% @doc Party settings request handler. Item distribution is random for now.
%% @todo Handle correctly.
handle(16#1709, _) ->
send_170a();
%% @doc Counter-related handler.
handle(16#170b, _) ->
send_170c();
%% @doc Counter initialization handler? Send the code for the background image to use.
%% @todo Handle correctly.
handle(16#1710, _) ->
User = egs_db:users_select(get(gid)),
[{quests, _}, {bg, Background}, {options, _}] = proplists:get_value(User#users.entryid, ?COUNTERS),
send_1711(Background);
%% @doc Dialog request handler. Do what we can.
%% @todo Handle correctly.
handle(16#1a01, Data) ->
<< _:32, A:8, B:8, _:16, C:8, _/bits >> = Data,
case [A, B, C] of
[ 0, 0, 2] ->
log("lumilass (and more?)"),
send_1a03();
[ 0, 0, 3] ->
log("pp cube"),
send_1a04();
2010-06-14 05:42:56 +08:00
[ 0, 0, 9] ->
log("types menu"),
send_1a07();
2010-06-14 05:42:56 +08:00
[80, 0, _] ->
log("npc dialog choice"),
send_1a02(0, 17, 17, 3, 9);
[90, 0, _] -> % All the replies from here are consistent but their effect is unknown.
log("1a01 unknown (~b ~b ~b)", [A, B, C]),
send_1a02(0, 5, 1, 4, 5);
[91, 0, _] ->
log("1a01 unknown (~b ~b ~b)", [A, B, C]),
send_1a02(0, 5, 5, 4, 7);
[92, 0, _] ->
log("1a01 unknown (~b ~b ~b)", [A, B, C]),
send_1a02(0, 5, 0, 4, 0);
[93, 0, _] ->
log("1a01 unknown (~b ~b ~b)", [A, B, C]),
send_1a02(0, 5, 18, 4, 0);
[ _, 2, _] ->
log("1a01 unknown (~b ~b ~b)", [A, B, C]),
send_1a02(0, 0, 1, 0, 0);
_ ->
log("1a01 unknown (~b ~b ~b) - do nothing", [A, B, C])
end;
%% @doc Unknown command handler. Do nothing.
handle(Command, _) ->
log("dismissed packet ~4.16.0b", [Command]).
%% @doc Handle all hits received.
%% @todo Finish the work on it.
2010-07-02 04:47:02 +08:00
%% @todo Type EXP.
%% @todo Boxes give EXP too but it doesn't show up on the screen.
%~ log_hits(Data) ->
%~ << A:32/unsigned-integer, B:32/unsigned-integer, C:32/unsigned-integer, D:32/unsigned-integer,
%~ E:32/unsigned-integer, F:32/unsigned-integer, G:32/unsigned-integer, H:32/unsigned-integer,
%~ I:32/unsigned-integer, J:32/unsigned-integer, K:32/unsigned-integer, L:32/unsigned-integer,
%~ M:32/unsigned-integer, N:32/unsigned-integer, O:32/unsigned-integer, P:32/unsigned-integer,
%~ Q:32/unsigned-integer, R:32/unsigned-integer, S:32/unsigned-integer, T:32/unsigned-integer, _/bits >> = Data,
%~ log("hit!~n ~8.16.0b ~8.16.0b ~8.16.0b ~8.16.0b~n ~8.16.0b ~8.16.0b ~8.16.0b ~8.16.0b~n ~8.16.0b ~8.16.0b ~8.16.0b ~8.16.0b~n ~8.16.0b ~8.16.0b ~8.16.0b ~8.16.0b~n ~8.16.0b ~8.16.0b ~8.16.0b ~8.16.0b", [A, B, C, D, E, F, G, H, I, J, K, L, M, N, O, P, Q, R, S, T]).
handle_hits(<< >>) ->
ok;
handle_hits(Data) ->
%~ log_hits(Data),
2010-07-02 04:47:02 +08:00
% parse
<< A:224/bits, B:128/bits, _:288, Rest/bits >> = Data,
<< _:96, SourceID:32/little-unsigned-integer, TargetID:32/little-unsigned-integer, _/bits >> = A,
2010-07-02 04:47:02 +08:00
% retrieve
GID = get(gid),
2010-07-02 04:47:02 +08:00
User = egs_db:users_select(GID),
% hit!
#hit_response{type=Type, user=NewUser, exp=HasEXP, damage=Damage, targethp=TargetHP, targetse=TargetSE, events=Events} = psu_missions:object_hit(User, SourceID, TargetID),
case Type of
box ->
% TODO: also has a hit sent, we should send it too
handle_events(Events);
_ ->
PlayerHP = (NewUser#users.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#users.character,
Level = Character#characters.mainlevel,
send_0115(GID, TargetID, Level#level.number, Level#level.exp, Character#characters.money);
true -> ignore
end,
% save
2010-07-02 04:47:02 +08:00
egs_db:users_insert(NewUser),
% next
handle_hits(Rest).
%% @doc Handle a list of events.
handle_events([]) ->
ok;
handle_events([{explode, ObjectID}|Tail]) ->
send_1213(ObjectID, 3),
handle_events(Tail);
handle_events([{event, [EventID, BlockID]}|Tail]) ->
send_1205(EventID, BlockID, 0),
handle_events(Tail).
%% @doc Build the packet header.
header(Command) ->
GID = get(gid),
<< Command:16/unsigned-integer, 16#0300:16, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64 >>.
%% @doc Send the given packet to the client.
%% @todo Consolidate the receive and send functions better.
send(Packet) ->
egs_proto:packet_send(get(socket), Packet).
%% @todo Figure out what this does compared to 0201(self).
%% @todo Figure out the unknown values.
%% @todo Probably don't pattern match the data like this...
send_010d(User) ->
GID = get(gid),
CharGID = User#users.gid,
<< _:640, CharBin/bits >> = psu_characters:character_user_to_binary(User#users{lid=0}),
send(<< 16#010d0300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64,
1:32/little-unsigned-integer, 0:32, 16#00000300:32, 16#ffff0000:32, 0:32, CharGID:32/little-unsigned-integer,
0:192, CharGID:32/little-unsigned-integer, 0:32, 16#ffffffff:32, CharBin/binary >>).
%% @todo Possibly related to 010d. Just send seemingly safe values.
send_0111(A, B) ->
GID = get(gid),
send(<< 16#01110300: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, A:32/little-unsigned-integer, B:32/little-unsigned-integer >>).
2010-06-14 05:42:56 +08:00
%% @todo Types capability list.
send_0113() ->
2010-06-14 05:42:56 +08:00
{ok, File} = file:read_file("p/typesinfo.bin"),
GID = get(gid),
send(<< 16#01130300: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, File/binary >>).
2010-06-14 05:42:56 +08:00
2010-07-02 04:47:02 +08:00
%% @doc Update the character's EXP.
send_0115(GID, TargetID, LV, EXP, Money) ->
send(<< 16#01150300: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, TargetID:32/little-unsigned-integer, LV:32/little-unsigned-integer, 0:32, 0:32, EXP:32/little-unsigned-integer, 0:32, Money:32/little-unsigned-integer, 16#f5470500:32, 0:96, 0:64,
16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32,
16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32 >>).
%% @doc Revive player?
%% @todo Figure out more of it.
send_0117(HP) ->
GID = get(gid),
send(<< 16#01170300: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:96, HP:32/little-unsigned-integer, 0:32 >>).
%% @doc Send the zone initialization notification.
send_0200(ZoneType) ->
case ZoneType of
mission ->
Var = << 16#06000500:32, 16#01000000:32, 0:64, 16#00040000:32, 16#00010000:32, 16#00140000:32 >>;
myroom ->
Var = << 16#06000000:32, 16#02000000:32, 0:64, 16#40000000:32, 16#00010000:32, 16#00010000:32 >>;
_ ->
Var = << 16#00040000:32, 0:160, 16#00140000:32 >>
end,
send(<< (header(16#0200))/binary, 0:32, 16#01000000:32, 16#ffffffff:32, Var/binary, 16#ffffffff:32, 16#ffffffff:32 >>).
%% @todo Figure out what the other things are.
%% @todo Handle LID correctly (should be ffffffff for self, apparently).
send_0201(User) ->
GID = get(gid),
CharGID = User#users.gid,
CharBin = psu_characters:character_user_to_binary(User#users{lid=0}),
IsGM = 0,
OnlineStatus = 0,
GameVersion = 0,
send(<< 16#02010300:32, 0:32, 16#00001200:32, CharGID:32/little-unsigned-integer, 0:64, 16#00011300:32,
GID:32/little-unsigned-integer, 0:64, CharBin/binary, IsGM:8, 0:8, OnlineStatus:8, GameVersion:8, 0:608 >>).
%% @doc Hello packet, always sent on client connection.
send_0202() ->
send(<< 16#02020300:32, 0:352 >>).
%% @todo Not sure. Used for unspawning, and more.
send_0204(PlayerGID, PlayerLID, Action) ->
GID = get(gid),
send(<< 16#02040300:32, 0:32, 16#00001200:32, PlayerGID:32/little-unsigned-integer, 0:64,
16#00011300:32, GID:32/little-unsigned-integer, 0:64, PlayerGID:32/little-unsigned-integer,
PlayerLID:32/little-unsigned-integer, Action:32/little-unsigned-integer >>).
%% @doc Send the map ID to be loaded by the client.
%% @todo Last two values are unknown.
send_0205(MapType, MapNumber, MapEntry, IsSeasonal) ->
send(<< 16#02050300:32, 0:288, 16#ffffffff:32, MapType:32/little-unsigned-integer,
MapNumber:32/little-unsigned-integer, MapEntry:32/little-unsigned-integer, 0:56, IsSeasonal >>).
%% @doc Indicate to the client that loading should finish.
%% @todo Last value seems to be 2 most of the time. Never 0 though. Apparently counters have it at 4.
send_0208() ->
send(<< (header(16#0208))/binary, 2:32/little-unsigned-integer >>).
%% @todo No idea what this one does. For unknown reasons it uses channel 2.
send_020c() ->
send(<< 16#020c020c:32, 16#fffff20c:32, 0:256 >>).
%% @doc Send the quest file to be loaded.
%% @todo Probably should try sending the checksum like value (right before the file) and see if it magically fixes anything.
send_020e(Filename) ->
{ok, File} = file:read_file(Filename),
Size = byte_size(File),
send(<< 16#020e0300:32, 0:288, Size:32/little-unsigned-integer, 0:32, File/binary, 0:32 >>).
%% @doc Send the zone file to be loaded.
send_020f(Filename, SetID, SeasonID) ->
{ok, File} = file:read_file(Filename),
Size = byte_size(File),
send(<< 16#020f0300:32, 0:288, SetID, SeasonID, 0:16, Size:32/little-unsigned-integer, File/binary >>).
%% @doc Send the current UNIX time.
send_0210() ->
CurrentTime = calendar:datetime_to_gregorian_seconds(calendar:now_to_universal_time(now()))
- calendar:datetime_to_gregorian_seconds({{1970, 1, 1}, {0, 0, 0}}),
send(<< (header(16#0210))/binary, 0:32, CurrentTime:32/little-unsigned-integer >>).
%% @todo No idea what this do. Nor why it's sent twice when loading a counter.
send_0215(N) ->
send(<< (header(16#0215))/binary, N:32/little-unsigned-integer >>).
%% @todo End of character loading. Just send it.
send_021b() ->
send(header(16#021b)).
%% @doc Send the list of available universes.
send_021e() ->
{ok, << File:1184/bits, _/bits >>} = file:read_file("p/unicube.bin"),
[StrCount] = io_lib:format("~b", [egs_db:users_count()]),
UCS2Count = << << X:8, 0:8 >> || X <- StrCount >>,
PaddingSize = (12 - byte_size(UCS2Count)) * 8,
send(<< 16#021e0300:32, 0:288, File/binary, UCS2Count/binary, 0:PaddingSize >>).
%% @doc Send the current universe name and number.
%% @todo Currently only have universe number 2, named EGS Test.
send_0222() ->
UCS2Name = << << X:8, 0:8 >> || X <- "EGS Test" >>,
GID = get(gid),
send(<< 16#02220300:32, 0:32, 16#00001200:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64,
2:32/little-unsigned-integer, 0:32, UCS2Name/binary, 0:16 >>).
%% @todo Not sure. Sent when going to or from room.
send_0230() ->
GID = get(gid),
send(<< 16#02300300:32, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64 >>).
%% @todo Figure out what the other things are.
send_0233(Users) ->
NbUsers = length(Users),
case NbUsers of
0 ->
ignore;
_ ->
GID = get(gid),
Header = << 16#02330300:32, 0:32, 16#00001200:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32,
GID:32/little-unsigned-integer, 0:64, NbUsers:32/little-unsigned-integer >>,
Contents = build_0233_contents(Users),
send(<< Header/binary, Contents/binary >>)
end.
%% @todo God this function is ugly. Use tail recursion!
%% @todo Do it properly without relying on the temporary file.
build_0233_contents([]) ->
<< >>;
build_0233_contents(Users) ->
[User|Rest] = Users,
LID = 16#010000 + User#users.lid, % @todo The LID must be 16 bits and 0233 seems to (almost always) require that 01 right there...
CharBin = psu_characters:character_user_to_binary(User#users{lid=LID}),
IsGM = 0,
GameVersion = 0,
Chunk = << CharBin/binary, IsGM:8, 0:8, GameVersion:8, 0:8 >>,
Next = build_0233_contents(Rest),
<< Chunk/binary, Next/binary >>.
%% @doc Center the camera on the player, if possible.
%% @todo Probably.
2010-06-05 01:36:56 +08:00
send_0236() ->
send(header(16#0236)).
%% @doc Send a chat command. Handled differently at v2.0000 and all versions starting somewhere above that.
send_0304(FromGID, FromName, Modifiers, Message) ->
case get(version) of
0 -> send(<< 16#03040300:32, 0:288, 16#00001200:32, FromGID:32/little-unsigned-integer, Modifiers:128/bits, Message/bits >>);
_ -> send(<< 16#03040300:32, 0:288, 16#00001200:32, FromGID:32/little-unsigned-integer, Modifiers:128/bits, FromName:512/bits, Message/bits >>)
end.
%% @todo Inventory related. No idea what it does.
send_0a05() ->
send(header(16#0a05)).
%% @todo Inventory related. Figure out everything in this packet and handle it correctly.
2010-06-05 01:46:17 +08:00
%% @todo It sends 60 values so it's probably some kind of options for all 60 items in the inventory?
send_0a06() ->
{ok, << _:32, A:96/bits, _:32, B:96/bits, _:32, C:1440/bits, _:32, D/bits >>} = file:read_file("p/packet0a06.bin"),
GID = get(gid),
send(<< A/binary, GID:32/little-unsigned-integer, B/binary, GID:32/little-unsigned-integer, C/binary, GID:32/little-unsigned-integer, D/binary >>).
%% @todo Inventory. Figure out everything in this packet and handle it correctly.
send_0a0a() ->
{ok, << _:32, A:224/bits, _:32, B/bits >>} = file:read_file("p/packet0a0a.bin"),
GID = get(gid),
send(<< A/binary, GID:32/little-unsigned-integer, B/binary >>).
2010-06-14 21:53:51 +08:00
%% @doc Item description.
send_0a11(ItemID, ItemDesc) ->
2010-06-14 21:53:51 +08:00
Size = 1 + length(ItemDesc),
UCS2Desc = << << X:8, 0:8 >> || X <- ItemDesc >>,
send(<< (header(16#0a11))/binary, ItemID:32/unsigned-integer, Size:32/little-unsigned-integer, UCS2Desc/binary, 0:16 >>).
2010-06-14 21:53:51 +08:00
%% @doc Init quest.
send_0c00(QuestID) ->
send(<< (header(16#0c00))/binary, QuestID:32/little-unsigned-integer,
16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32,
16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32,
16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32,
16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32 >>).
%% @todo Figure out last 4 bytes!
send_0c02() ->
send(<< (header(16#0c02))/binary, 0:32 >>).
%% @doc Send the huge pack of quest files available in the counter.
send_0c06(Filename) ->
{ok, << File/bits >>} = file:read_file(Filename),
send(<< 16#0c060300:32, 0:288, 1:32/little-unsigned-integer, File/binary >>).
%% @doc Reply whether the player is allowed to use the transport option.
%% Use true for allowing it, and false otherwise.
send_0c08(Response) ->
Value = if Response =:= true -> 0; true -> 1 end,
send(<< (header(16#0c08))/binary, Value:32 >>).
%% @doc Send the trial start notification.
send_0c09() ->
send(<< (header(16#0c09))/binary, 0:64 >>).
%% @doc Send the counter's mission options (0 = invisible, 2 = disabled, 3 = available).
send_0c10(Options) ->
GID = get(gid),
send(<< 16#0c100300:32, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64,
16#00011300:32, GID:32/little-unsigned-integer, 0:64, Options/binary >>).
%% @doc Send the data for the selected character.
%% @todo The large chunk of 0s can have some values set... but what are they used for?
%% @todo The values after the Char variable are the flags. Probably use bits to define what flag is and isn't set. Handle correctly.
send_0d01(User) ->
CharBin = psu_characters:character_tuple_to_binary(User#users.character),
OptionsBin = psu_characters:options_tuple_to_binary((User#users.character)#characters.options),
send(<< (header(16#0d01))/binary, 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.
send_0d03(Data0, Data1, Data2, Data3) ->
[{status, Status0}, {char, Char0}|_] = Data0,
[{status, Status1}, {char, Char1}|_] = Data1,
[{status, Status2}, {char, Char2}|_] = Data2,
[{status, Status3}, {char, Char3}|_] = Data3,
GID = get(gid),
send(<< 16#0d030300:32/unsigned-integer, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:104,
Status0:8/unsigned-integer, 0:48, Char0/binary, 0:520,
Status1:8/unsigned-integer, 0:48, Char1/binary, 0:520,
Status2:8/unsigned-integer, 0:48, Char2/binary, 0:520,
Status3:8/unsigned-integer, 0:48, Char3/binary, 0:512 >>).
%% @doc Send the character 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() ->
{ok, Flags} = file:read_file("p/flags.bin"),
GID = get(gid),
Packet = << 16#0d050300:32, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, Flags/binary >>,
Size = 4 + byte_size(Packet),
ssl:send(get(socket), << Size:32/little-unsigned-integer, Packet/binary >>).
%% @todo Figure out what the packet is.
send_1005(Name) ->
{ok, File} = file:read_file("p/packet1005.bin"),
<< _:352, Before:160/bits, _:608, After/bits >> = File,
GID = get(gid),
send(<< (header(16#1005))/binary, Before/binary, GID:32/little-unsigned-integer, 0:64, Name/binary, After/binary >>).
%% @doc Party-related command probably controlling the party state.
%% Value 11 aborts the mission.
%% @todo Figure out what the packet is.
send_1006(N) ->
send(<< (header(16#1006))/binary, N:32/little-unsigned-integer >>).
%% @doc Send the player's current location.
send_100e(QuestID, ZoneID, MapID, Location, CounterID) ->
UCS2Location = << << X:8, 0:8 >> || X <- Location >>,
Packet = << (header(16#100e))/binary, 1:32/little-unsigned-integer, MapID:16/little-unsigned-integer,
ZoneID:16/little-unsigned-integer, QuestID:32/little-unsigned-integer, UCS2Location/binary >>,
PaddingSize = (128 - byte_size(Packet) - 8) * 8,
case CounterID of
16#ffffffff ->
Footer = << CounterID:32/little-unsigned-integer, 0:32 >>;
_ ->
Footer = << CounterID:32/little-unsigned-integer, 1:32/little-unsigned-integer >>
end,
send(<< Packet/binary, 0:PaddingSize, Footer/binary >>).
2010-06-09 01:09:54 +08:00
%% @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.
send_1015(QuestID) ->
[{type, _}, {file, QuestFile}|_] = proplists:get_value(QuestID, ?QUESTS),
2010-06-09 01:09:54 +08:00
{ok, File} = file:read_file(QuestFile),
Size = byte_size(File),
send(<< (header(16#1015))/binary, QuestID:32/little-unsigned-integer, 16#01010000:32, 0:32, Size:32/little-unsigned-integer, File/binary >>).
%% @todo Totally unknown.
send_1020() ->
send(header(16#1020)).
%% @doc Update HP in the party members information on the left.
%% @todo Figure out more of it.
send_1022(HP) ->
GID = get(gid),
send(<< 16#10220300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, HP:32/little-unsigned-integer, 0:32 >>).
%% @todo Figure out what this packet does. Sane values for counter and missions for now.
send_1202() ->
send(<< (header(16#1202))/binary, 0:32, 16#10000000:32, 0:64, 16#14000000:32, 0:32 >>).
%% @todo Figure out what this packet does. Seems it's the same values all the time.
send_1204() ->
send(<< (header(16#1204))/binary, 0:32, 16#20000000:32, 0:256 >>).
%% @doc Object events response?
%% @todo Not sure what Value does exactly. It's either 0 or 1.
send_1205(EventID, BlockID, Value) ->
send(<< (header(16#1205))/binary, EventID, BlockID, 0:16, Value, 0:24 >>).
%% @todo Figure out what this packet does. Sane values for counter and missions for now.
send_1206() ->
send(<< (header(16#1206))/binary, 0:32, 16#80020000:32, 0:5120 >>).
%% @todo Figure out what this packet does. Sane values for counter and missions for now.
send_1207() ->
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 >>,
send(<< (header(16#1207))/binary, Chunk/binary, Chunk/binary, Chunk/binary, Chunk/binary, Chunk/binary, Chunk/binary >>).
2010-06-14 21:38:57 +08:00
%% @todo Object interaction? Figure out. C probably the interaction type.
send_1211(A, B, C, D) ->
send(<< (header(16#1211))/binary, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer, D:32/little-unsigned-integer >>).
2010-06-14 21:38:57 +08:00
%% @doc Make the client load the quest previously sent.
send_1212() ->
send(<< (header(16#1212))/binary, 0:19200 >>).
%% @todo Not sure. Related to keys.
send_1213(A, B) ->
send(<< (header(16#1213))/binary, A:32/little-unsigned-integer, B:32/little-unsigned-integer >>).
%% @doc Send the player's partner card.
%% @todo Find out the remaining values.
send_1500(User) ->
#characters{slot=Slot, name=Name, race=Race, gender=Gender, class=Class} = User#users.character,
RaceBin = psu_characters:race_atom_to_binary(Race),
GenderBin = psu_characters:gender_atom_to_binary(Gender),
ClassBin = psu_characters:class_atom_to_binary(Class),
send(<< (header(16#1500))/binary, Name/binary, RaceBin:8, GenderBin:8, ClassBin:8, 0:3112, 16#010401:24, Slot:8, 0:64 >>).
%% @todo Send an empty partner card list.
send_1501() ->
send(<< (header(16#1501))/binary, 0:32 >>).
2010-06-04 23:02:13 +08:00
%% @todo Send an empty blacklist.
send_1512() ->
send(<< (header(16#1512))/binary, 0:46080 >>).
2010-06-04 23:02:13 +08:00
%% @doc Send the player's NPC and PM information.
send_1602() ->
{ok, File} = file:read_file("p/npc.bin"),
send(<< (header(16#1602))/binary, 0:32, File/binary >>).
%% @doc Party information.
%% @todo Handle existing parties.
send_1706(CharName) ->
send(<< (header(16#1706))/binary, 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 >>).
%% @doc Party settings. Item distribution is random for now.
%% @todo Handle correctly.
send_170a() ->
send(<< (header(16#170a))/binary, 16#01010c08:32 >>).
%% @todo Find what the heck this packet is.
send_170c() ->
{ok, File} = file:read_file("p/packet170c.bin"),
send(<< (header(16#170c))/binary, File/binary >>).
%% @doc Send the background to use for the counter.
%% @todo Background has more info past the first byte.
send_1711(Background) ->
send(<< (header(16#1711))/binary, Background:32/little-unsigned-integer >>).
%% @doc Unknown dialog-related handler.
%% @todo Everything!
send_1a02(A, B, C, D, E) ->
send(<< (header(16#1a02))/binary, A:32/little-unsigned-integer, B:16/little-unsigned-integer,
C:16/little-unsigned-integer, D:16/little-unsigned-integer, E:16/little-unsigned-integer >>).
%% @doc Lumilass handler. Possibly more.
%% @todo Figure out how Lumilass work exactly. The 4 bytes before the file may vary.
send_1a03() ->
{ok, File} = file:read_file("p/lumilassA.bin"),
send(<< (header(16#1a03))/binary, 0:32, File/binary >>).
%% @doc PP cube handler.
%% @todo The 4 bytes before the file may vary. Everything past that is the same. Figure things out.
send_1a04() ->
{ok, File} = file:read_file("p/ppcube.bin"),
send(<< (header(16#1a04))/binary, 0:32, File/binary >>).
2010-06-14 05:42:56 +08:00
%% @doc Types menu handler.
%% @todo Handle correctly.
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]).
2010-06-04 05:36:34 +08:00
log(Message, Format) ->
FormattedMessage = io_lib:format(Message, Format),
log(FormattedMessage).