From 8ffc8f2069e02cc6f4c6940f938a579312c2108b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Lo=C3=AFc=20Hoguin?= Date: Thu, 3 Jun 2010 22:53:34 +0200 Subject: [PATCH] login: Move all parsing functions out of proto and do it inline for clarity. --- src/egs_login.erl | 33 +++++++++++++++++++++++---------- src/egs_proto.erl | 36 ------------------------------------ 2 files changed, 23 insertions(+), 46 deletions(-) diff --git a/src/egs_login.erl b/src/egs_login.erl index b19cd0e..e9348a1 100644 --- a/src/egs_login.erl +++ b/src/egs_login.erl @@ -85,31 +85,44 @@ loop(CSocket, SessionID) -> handle(16#0217, CSocket, SessionID, _) -> log(SessionID, "send game server info"), - egs_proto:send_game_server_info(CSocket, SessionID, ?GAME_IP, ?GAME_PORT), + IP = ?GAME_IP, + Port = ?GAME_PORT, + Packet = << 16#02160300:32, 0:192, SessionID:32/little-unsigned-integer, 0:64, IP/binary, Port:32/little-unsigned-integer >>, + egs_proto:packet_send(CSocket, Packet), ssl:close(CSocket), closed; %% @doc Authentication request handler. Currently always succeed. %% Use the temporary session ID as the GID for now. %% Use username and password as a folder name for saving character data. -%% @todo Handle real GIDs whenever there's real authentication. +%% @todo Handle real GIDs whenever there's real authentication. GID is the second SessionID in the reply. +%% @todo Apparently it's possible to ask a question in the reply here. Used for free course on JP. -handle(16#0219, CSocket, SessionID, Packet) -> - [{username, Username}, {password, Password}] = egs_proto:parse_auth_request(Packet), +handle(16#0219, CSocket, SessionID, Orig) -> + << _:352, UsernameBlob:192/bits, PasswordBlob:192/bits, _/bits >> = Orig, + Username = re:replace(UsernameBlob, "\\0", "", [global, {return, binary}]), + Password = re:replace(PasswordBlob, "\\0", "", [global, {return, binary}]), log(SessionID, io_lib:format("auth success for ~s ~s", [Username, Password])), Auth = crypto:rand_bytes(4), Folder = << Username/binary, "-", Password/binary >>, Time = calendar:datetime_to_gregorian_seconds(calendar:universal_time()), egs_db:users_insert(#users{gid=SessionID, pid=self(), socket=CSocket, auth=Auth, time=Time, folder=Folder}), - egs_proto:send_auth_success(CSocket, SessionID, SessionID, Auth); + Packet = << 16#02230300:32, 0:192, SessionID:32/little-unsigned-integer, 0:64, SessionID:32/little-unsigned-integer, Auth:32/bits >>, + egs_proto:packet_send(CSocket, Packet); -%% @doc MOTD request handler. Handles both forms of MOTD requests, US and JP. -%% Currently ignore the language and send the same MOTD file to everyone. +%% @doc MOTD request handler. Handles both forms of MOTD requests, US and JP. Page number starts at 0. +%% @todo Currently ignore the language and send the same MOTD file to everyone. Language is 8 bits next to Page. +%% @todo Use a normal ASCII file rather than an UCS2 one? -handle(Command, CSocket, SessionID, Packet) when Command =:= 16#0226; Command =:= 16#023f -> - [{page, Page}, {language, _}] = egs_proto:parse_motd_request(Packet), +handle(Command, CSocket, SessionID, Orig) when Command =:= 16#0226; Command =:= 16#023f -> + << _:352, Page:8/little-unsigned-integer, _/bits >> = Orig, log(SessionID, io_lib:format("send MOTD page ~.10b", [Page + 1])), - egs_proto:send_motd(CSocket, Page); + {ok, File} = file:read_file("conf/motd.txt"), + Tokens = re:split(File, "\n."), + MOTD = << << Line/binary, "\n", 0 >> || Line <- lists:sublist(Tokens, 1 + Page * 15, 15) >>, + NbPages = 1 + length(Tokens) div 15, + Packet = << 16#0225:16, 0:304, NbPages:8, Page:8, 16#8200:16/unsigned-integer, MOTD/binary, 0:16 >>, + egs_proto:packet_send(CSocket, Packet); %% @doc Unknown command handler. Do nothing. diff --git a/src/egs_proto.erl b/src/egs_proto.erl index 5d7a0fd..6366cac 100644 --- a/src/egs_proto.erl +++ b/src/egs_proto.erl @@ -119,13 +119,6 @@ packet_split(Packet, Result) -> end end. -%% @doc Parse a login authentication command. Return the username and password. - -parse_auth_request(Packet) -> - << _:352, Username:192/bits, Password:192/bits, _/bits >> = Packet, - [{username, re:replace(Username, "\\0", "", [global, {return, binary}])}, - {password, re:replace(Password, "\\0", "", [global, {return, binary}])}]. - %% @doc Parse a character creation command. Return the character number and data. parse_character_create(Packet) -> @@ -163,12 +156,6 @@ parse_lobby_change(Packet) -> MapNumber:16/little-unsigned-integer, MapEntry:16/little-unsigned-integer, _/bits >> = Packet, [{quest, Quest}, {maptype, MapType}, {mapnumber, MapNumber}, {mapentry, MapEntry}]. -%% @doc Parse the MOTD request command. - -parse_motd_request(Packet) -> - << _:352, Page:8/little-unsigned-integer, Language:8/little-unsigned-integer, _/bits >> = Packet, - [{page, Page}, {language, Language}]. - %% @doc Parse the options change command. Retrieve the options for saving. parse_options_change(Packet) -> @@ -187,13 +174,6 @@ parse_uni_select(Packet) -> << _:352, Uni:32/little-unsigned-integer, _/bits >> = Packet, [{uni, Uni}]. -%% @doc Indicate that the authentication was successful and send the real GUARDIANS ID. -%% @todo Apparently it's possible to ask a question here too. Used for free course on JP. - -send_auth_success(CSocket, SessionID, GID, Auth) -> - Packet = << 16#0223:16, 0:208, SessionID:32/little-unsigned-integer, 0:64, GID:32/little-unsigned-integer, Auth:32/bits >>, - packet_send(CSocket, Packet). - %% @doc Center the camera on the player, if possible. %% @todo Probably. @@ -242,12 +222,6 @@ send_flags(CSocket, GID) -> Size = 4 + byte_size(Packet), ssl:send(CSocket, << Size:32/little-unsigned-integer, Packet/binary >>). -%% @doc Send the IP and port of the game server the player is sent to. - -send_game_server_info(CSocket, GID, IP, Port) -> - Packet = << 16#0216:16, 0:208, GID:32/little-unsigned-integer, 0:64, IP/binary, Port:32/little-unsigned-integer >>, - packet_send(CSocket, Packet). - %% @doc Shortcut for send_global/4. send_global(CSocket, Type, Message) -> @@ -324,16 +298,6 @@ send_map(CSocket, Quest, MapType, MapNumber, MapEntry) -> MapNumber:32/little-unsigned-integer, MapEntry:32/little-unsigned-integer, 0:64 >>, packet_send(CSocket, Packet). -%% @doc Send the requested MOTD page to the client. Pages start at 0. - -send_motd(CSocket, Page) -> - {ok, File} = file:read_file("conf/motd.txt"), - Tokens = re:split(File, "\n."), - MOTD = << << Line/binary, "\n", 0 >> || Line <- lists:sublist(Tokens, 1 + Page * 15, 15) >>, - NbPages = 1 + length(Tokens) div 15, - Packet = << 16#0225:16, 0:304, NbPages:8, Page:8, 16#8200:16/unsigned-integer, MOTD/binary, 0:16 >>, - packet_send(CSocket, Packet). - %% @doc Send the player's NPC and PM information. send_npc_info(CSocket, GID) ->