patch: Add a fully working patch server and replace the old hack with it.
This commit is contained in:
parent
d043ab4d3d
commit
8526b5ab8f
1
priv/.gitignore
vendored
1
priv/.gitignore
vendored
@ -1 +1,2 @@
|
||||
egs_drv.so
|
||||
patch.conf
|
||||
|
25
priv/patch.conf
Normal file
25
priv/patch.conf
Normal file
@ -0,0 +1,25 @@
|
||||
%% 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 <http://www.gnu.org/licenses/>.
|
||||
|
||||
%% @doc List of folders where patched files are located.
|
||||
{folders, [root, "DATA"]}.
|
||||
|
||||
%% @doc Files in the root folder.
|
||||
{{folder, root}, [
|
||||
]}.
|
||||
|
||||
%% @doc Files in the DATA folder.
|
||||
{{folder, "DATA"}, [
|
||||
]}.
|
1
priv/patch/.gitignore
vendored
Normal file
1
priv/patch/.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
||||
*
|
Binary file not shown.
@ -1 +0,0 @@
|
||||
═╚и╡|7с[
|
Binary file not shown.
@ -1 +0,0 @@
|
||||
”§<EFBFBD><EFBFBD>±ינE<EFBFBD>%‚®³5%
|
@ -1,2 +0,0 @@
|
||||
ú;§¾x¢ýÀ´6çZ¥ë
|
||||
Ùí„Î!ÚU“ ÓC¥Ÿ¬(ä\ˆ–<>®ôÅRä$£a 6Ä+÷´Š…FVÞx9‘xü*°âÊ+².<2E>tþàBw]õÜÑ3ŒR¬ÙŒ
|
172
src/egs_patch_files_db.erl
Normal file
172
src/egs_patch_files_db.erl
Normal file
@ -0,0 +1,172 @@
|
||||
%% @author Loïc Hoguin <essen@dev-extend.eu>
|
||||
%% @copyright 2010 Loïc Hoguin.
|
||||
%% @doc EGS patch files database and cache manager.
|
||||
%%
|
||||
%% 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 <http://www.gnu.org/licenses/>.
|
||||
|
||||
-module(egs_patch_files_db).
|
||||
-behavior(gen_server).
|
||||
-export([start_link/0, stop/0, list/0, check/3, get_size/1, get_info/1, reload/0]). %% API.
|
||||
-export([init/1, handle_call/3, handle_cast/2, handle_info/2, terminate/2, code_change/3]). %% gen_server.
|
||||
|
||||
-include_lib("kernel/include/file.hrl").
|
||||
|
||||
-record(state, {list_bin=[], files=[]}).
|
||||
-record(file, {crc, size, folder, filename_bin, full_filename}).
|
||||
|
||||
%% Use the module name for the server's name.
|
||||
-define(SERVER, ?MODULE).
|
||||
|
||||
%% API.
|
||||
|
||||
%% @spec start_link() -> {ok,Pid::pid()}
|
||||
start_link() ->
|
||||
gen_server:start_link({local, ?SERVER}, ?MODULE, [], []).
|
||||
|
||||
%% @spec stop() -> stopped
|
||||
stop() ->
|
||||
gen_server:call(?SERVER, stop).
|
||||
|
||||
%% @spec list() -> binary()
|
||||
list() ->
|
||||
gen_server:call(?SERVER, list).
|
||||
|
||||
%% @spec check(FileNumber, CRC, Size) -> ok | invalid
|
||||
check(FileNumber, CRC, Size) ->
|
||||
gen_server:call(?SERVER, {check, FileNumber, CRC, Size}).
|
||||
|
||||
%% @spec get_size(FileNumber) -> Size
|
||||
get_size(FileNumber) ->
|
||||
gen_server:call(?SERVER, {get_size, FileNumber}).
|
||||
|
||||
%% @spec get_info(FileNumber) -> {CRC, Size, FilenameBin, FullFilename}
|
||||
get_info(FileNumber) ->
|
||||
gen_server:call(?SERVER, {get_info, FileNumber}).
|
||||
|
||||
%% @spec reload() -> ok
|
||||
reload() ->
|
||||
gen_server:cast(?SERVER, reload).
|
||||
|
||||
%% gen_server.
|
||||
|
||||
init([]) ->
|
||||
{ok, build_state()}.
|
||||
|
||||
handle_call(list, _From, State=#state{list_bin=Bin}) ->
|
||||
{reply, Bin, State};
|
||||
|
||||
handle_call({check, FileNumber, CRC, Size}, _From, State=#state{files=Files}) ->
|
||||
File = proplists:get_value(FileNumber, Files),
|
||||
case File of
|
||||
#file{crc=CRC, size=Size} -> {reply, ok, State};
|
||||
_Any -> {reply, invalid, State}
|
||||
end;
|
||||
|
||||
handle_call({get_size, FileNumber}, _From, State=#state{files=Files}) ->
|
||||
File = proplists:get_value(FileNumber, Files),
|
||||
{reply, File#file.size, State};
|
||||
|
||||
handle_call({get_info, FileNumber}, _From, State=#state{files=Files}) ->
|
||||
{reply, proplists:get_value(FileNumber, Files), State};
|
||||
|
||||
handle_call(stop, _From, State) ->
|
||||
{stop, normal, stopped, State};
|
||||
|
||||
handle_call(_Request, _From, State) ->
|
||||
{reply, ignored, State}.
|
||||
|
||||
handle_cast(reload, _State) ->
|
||||
{noreply, build_state()};
|
||||
|
||||
handle_cast(_Msg, State) ->
|
||||
{noreply, State}.
|
||||
|
||||
handle_info(_Info, State) ->
|
||||
{noreply, State}.
|
||||
|
||||
terminate(_Reason, _State) ->
|
||||
ok.
|
||||
|
||||
code_change(_OldVsn, State, _Extra) ->
|
||||
{ok, State}.
|
||||
|
||||
%% Internal.
|
||||
|
||||
build_state() ->
|
||||
{ok, Terms} = file:consult("priv/patch.conf"),
|
||||
Folders = proplists:get_value(folders, Terms),
|
||||
{ListBin, Files} = build_list_bin(Folders, Terms),
|
||||
#state{list_bin=ListBin, files=Files}.
|
||||
|
||||
%% The file number must start at 0.
|
||||
build_list_bin(Folders, Terms) ->
|
||||
build_list_bin(Folders, Terms, 0, [], []).
|
||||
build_list_bin([], _Terms, _N, Acc, FilesAcc) ->
|
||||
Bin = list_to_binary(lists:reverse(Acc)),
|
||||
Bin2 = << 16#08:32/little, 16#06:32/little, Bin/binary, 16#08:32/little, 16#08:32/little >>,
|
||||
{Bin2, lists:flatten(FilesAcc)};
|
||||
build_list_bin([Folder|Tail], Terms, N, Acc, FilesAcc) ->
|
||||
Filenames = proplists:get_value({folder, Folder}, Terms),
|
||||
{BinFiles, Files, N2} = build_files_bin(Folder, Filenames, N),
|
||||
BinFiles2 = case Folder of
|
||||
root -> BinFiles;
|
||||
_Any ->
|
||||
FolderBin = list_to_binary(Folder),
|
||||
Padding = 8 * (64 - length(Folder)),
|
||||
<< 16#48:32/little, 16#09:32/little, FolderBin/binary, 0:Padding,
|
||||
BinFiles/binary, 16#08:32/little, 16#0a:32/little >>
|
||||
end,
|
||||
build_list_bin(Tail, Terms, N2, [BinFiles2|Acc], [Files|FilesAcc]).
|
||||
|
||||
build_files_bin(Folder, Filenames, N) ->
|
||||
build_files_bin(Folder, Filenames, N, [], []).
|
||||
build_files_bin(_Folder, [], N, Acc, FilesAcc) ->
|
||||
Bin = list_to_binary(lists:reverse(Acc)),
|
||||
{Bin, FilesAcc, N};
|
||||
build_files_bin(Folder, [Filename|Tail], N, Acc, FilesAcc) ->
|
||||
FullFilename = case Folder of
|
||||
root -> ["priv/patch/"|Filename];
|
||||
_Any -> ["priv/patch/",Folder,"/"|Filename]
|
||||
end,
|
||||
Size = file_get_size(FullFilename),
|
||||
CRC = file_get_crc(FullFilename),
|
||||
FilenameBin = list_to_binary(Filename),
|
||||
Padding = 8 * (64 - length(Filename)),
|
||||
FilenameBin2 = << FilenameBin/binary, 0:Padding >>,
|
||||
Bin = << 16#4c:32/little, 16#07:32/little, N:32/little, FilenameBin2/binary >>,
|
||||
build_files_bin(Folder, Tail, N + 1, [Bin|Acc], [{N, #file{crc=CRC, size=Size, folder=Folder, filename_bin=FilenameBin2, full_filename=FullFilename}}|FilesAcc]).
|
||||
|
||||
file_get_size(Filename) ->
|
||||
{ok, FileInfo} = file:read_file_info(Filename),
|
||||
FileInfo#file_info.size.
|
||||
|
||||
file_get_crc(Filename) ->
|
||||
{ok, IoDevice} = file:open(Filename, [read, raw, binary]),
|
||||
case file:read(IoDevice, 524288) of
|
||||
eof -> 0;
|
||||
{ok, Data} ->
|
||||
CRC = erlang:crc32(Data),
|
||||
file_get_crc(IoDevice, CRC)
|
||||
end.
|
||||
file_get_crc(IoDevice, CRC) ->
|
||||
case file:read(IoDevice, 524288) of
|
||||
{ok, Data} ->
|
||||
CRC2 = erlang:crc32(CRC, Data),
|
||||
file_get_crc(IoDevice, CRC2);
|
||||
eof ->
|
||||
file:close(IoDevice),
|
||||
CRC
|
||||
end.
|
234
src/egs_patch_server.erl
Normal file
234
src/egs_patch_server.erl
Normal file
@ -0,0 +1,234 @@
|
||||
%% @author Loïc Hoguin <essen@dev-extend.eu>
|
||||
%% @copyright 2010 Loïc Hoguin.
|
||||
%% @doc Patch server 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 <http://www.gnu.org/licenses/>.
|
||||
|
||||
-module(egs_patch_server).
|
||||
-export([start_link/1]). %% External.
|
||||
-export([listen/1, accept/1, init/1, recv/3]). %% Internal
|
||||
|
||||
-define(OPTIONS, [binary, {active, true}, {packet, 0}, {reuseaddr, true}, {send_timeout, 5000}]).
|
||||
|
||||
-record(state, {step=start, files=[]}).
|
||||
|
||||
%% @spec start_link(Port) -> {ok,Pid::pid()}
|
||||
%% @doc Start the PSU patch server for inclusion in a supervisor tree.
|
||||
start_link(Port) ->
|
||||
Pid = spawn(?MODULE, listen, [Port]),
|
||||
{ok, Pid}.
|
||||
|
||||
%% @spec listen(Port) -> ok
|
||||
%% @doc Listen for connections.
|
||||
listen(Port) ->
|
||||
error_logger:info_report(io_lib:format("listener started for ~p on port ~b", [?MODULE, Port])),
|
||||
{ok, LSocket} = gen_tcp:listen(Port, ?OPTIONS),
|
||||
?MODULE:accept(LSocket).
|
||||
|
||||
%% @spec accept(LSocket) -> ok
|
||||
%% @doc Accept connections.
|
||||
accept(LSocket) ->
|
||||
case gen_tcp:accept(LSocket, 5000) of
|
||||
{ok, CSocket} ->
|
||||
Pid = spawn(?MODULE, init, [CSocket]),
|
||||
gen_tcp:controlling_process(CSocket, Pid);
|
||||
{error, timeout} ->
|
||||
reload
|
||||
end,
|
||||
?MODULE:accept(LSocket).
|
||||
|
||||
%% @spec init(CSocket) -> ok
|
||||
%% @doc Send the hello packet and move on to the main loop.
|
||||
init(CSocket) ->
|
||||
send_01(CSocket),
|
||||
recv(CSocket, << >>, #state{}).
|
||||
|
||||
%% @spec recv(CSocket, SoFar, State) -> ok
|
||||
%% @doc Receive commands from the client and process them.
|
||||
recv(CSocket, SoFar, State) ->
|
||||
receive
|
||||
{tcp, CSocket, Data} ->
|
||||
{Commands, Rest} = split(<< SoFar/bits, Data/bits >>, []),
|
||||
case handle(CSocket, Commands, State) of
|
||||
closed -> closed;
|
||||
State2 -> ?MODULE:recv(CSocket, Rest, State2)
|
||||
end;
|
||||
{tcp_closed, CSocket} ->
|
||||
tcp_closed;
|
||||
{tcp_error, CSocket, _Any} ->
|
||||
tcp_error;
|
||||
_ ->
|
||||
?MODULE:recv(CSocket, SoFar, State)
|
||||
end.
|
||||
|
||||
%% @spec split(Bin, Acc) -> {Commands, Rest}
|
||||
%% @doc Split the given binary into a list of commands.
|
||||
split(<< >>, Acc) ->
|
||||
{lists:reverse(Acc), << >>};
|
||||
split(Rest = << Size:32/little, Data/bits >>, Acc) when Data + 4 < Size ->
|
||||
{lists:reverse(Acc), Rest};
|
||||
split(<< Size:32/little, Cmd:16/little, _Junk:16, Rest/bits >>, Acc) ->
|
||||
BitSize = 8 * Size - 64,
|
||||
<< Data:BitSize/bits, Rest2/bits >> = Rest,
|
||||
split(Rest2, [{command, Cmd, Size, Data}|Acc]).
|
||||
|
||||
%% @spec handle(CSocket, CommandsList) -> closed | State
|
||||
%% @doc Handle the given commands.
|
||||
handle(_CSocket, [], State) ->
|
||||
State;
|
||||
%% Start of file info reply.
|
||||
handle(CSocket, [{command, 16#0c, 8, << >>}|Tail], State=#state{step=waitfileinfo}) ->
|
||||
handle(CSocket, Tail, State#state{step=recvfileinfo});
|
||||
%% File info.
|
||||
handle(CSocket, [{command, 16#0d, 20, << FileNumber:32/little, CRC:32/little, Size:32/little >>}|Tail], State=#state{step=recvfileinfo}) ->
|
||||
State2 = case egs_patch_files_db:check(FileNumber, CRC, Size) of
|
||||
ok -> State;
|
||||
invalid -> State#state{files=[FileNumber|State#state.files]}
|
||||
end,
|
||||
handle(CSocket, Tail, State2);
|
||||
%% End of file info reply. Nothing expected from the client afterward.
|
||||
handle(CSocket, [{command, 16#0e, 8, << >>}], #state{step=recvfileinfo, files=Files}) ->
|
||||
case Files of
|
||||
[] -> ok; %% No files to update.
|
||||
_List -> update(CSocket, lists:reverse(Files))
|
||||
end,
|
||||
send_13(CSocket),
|
||||
closed;
|
||||
%% Hello reply.
|
||||
%% @todo Figure out the remaining unknown values.
|
||||
handle(CSocket, [{command, 16#14, 52, << 16#e44c0915:32, UnknownA:32/little, UnknownB:32/little, UnknownC:32/little,
|
||||
UnknownD:32/little, _GameVersion:32/little, UnknownE:32/little, 0:128 >>}|Tail], State=#state{step=start}) ->
|
||||
io:format("patch #14: ~p ~p ~p ~p ~p~n", [UnknownA, UnknownB, UnknownC, UnknownD, UnknownE]),
|
||||
ListBin = egs_patch_files_db:list(),
|
||||
gen_tcp:send(CSocket, ListBin),
|
||||
handle(CSocket, Tail, State#state{step=waitfileinfo});
|
||||
%% Unknown command.
|
||||
handle(_CSocket, [{command, Cmd, _Size, Data}|_Tail], State) ->
|
||||
io:format("~p: dismissed command ~2.16.0b - ~p - ~p~n", [?MODULE, Cmd, Data, State]),
|
||||
closed.
|
||||
|
||||
%% @spec update(CSocket, Files) -> ok
|
||||
%% @doc Update the invalid client files.
|
||||
update(CSocket, Files) ->
|
||||
Size = update_size(Files),
|
||||
send_0f(CSocket, Size, length(Files)),
|
||||
update_files(CSocket, root, Files).
|
||||
|
||||
%% @spec update_files(CSocket, Files) -> ok
|
||||
%% @doc Send all the files the client needs to update.
|
||||
update_files(_CSocket, _CurrentFolder, []) ->
|
||||
ok;
|
||||
update_files(CSocket, CurrentFolder, [FileNumber|Tail]) ->
|
||||
{file, _CRC, Size, Folder, FilenameBin, FullFilename} = egs_patch_files_db:get_info(FileNumber),
|
||||
case CurrentFolder of
|
||||
Folder -> ok;
|
||||
_Any ->
|
||||
if CurrentFolder =/= root ->
|
||||
send_0a(CSocket);
|
||||
true -> ok
|
||||
end,
|
||||
if Folder =/= root ->
|
||||
send_09(CSocket, Folder);
|
||||
true -> ok
|
||||
end
|
||||
end,
|
||||
send_10(CSocket, Size, FilenameBin),
|
||||
update_send_file(CSocket, FullFilename),
|
||||
send_12(CSocket),
|
||||
update_files(CSocket, Folder, Tail).
|
||||
|
||||
%% @spec update_send_file(CSocket, Filename) -> ok
|
||||
%% @doc Send a file by fragmenting it into many chunks of fixed size.
|
||||
update_send_file(CSocket, Filename) ->
|
||||
{ok, IoDevice} = file:open(Filename, [read, raw, binary]),
|
||||
update_send_file(CSocket, IoDevice, 0).
|
||||
update_send_file(CSocket, IoDevice, N) ->
|
||||
case file:read(IoDevice, 24576) of
|
||||
{ok, Data} ->
|
||||
send_11(CSocket, Data, N),
|
||||
update_send_file(CSocket, IoDevice, N + 1);
|
||||
eof ->
|
||||
file:close(IoDevice)
|
||||
end.
|
||||
|
||||
%% @spec update_size(Files) -> Size
|
||||
%% @doc Return the total size for all the files given.
|
||||
update_size(Files) ->
|
||||
update_size(Files, 0).
|
||||
update_size([], Size) ->
|
||||
Size;
|
||||
update_size([FileNumber|Tail], Size) ->
|
||||
FileSize = egs_patch_files_db:get_size(FileNumber),
|
||||
update_size(Tail, Size + FileSize).
|
||||
|
||||
%% @spec send_01(CSocket) -> ok
|
||||
%% @doc Hello command sent when a client connects to the server. Encryption is disabled.
|
||||
send_01(CSocket) ->
|
||||
Bin = << 16#28:32/little, 16#01:32/little, 16#8b9f2dfa:32, 0:96, 1:32/little, 0:96 >>,
|
||||
gen_tcp:send(CSocket, Bin).
|
||||
|
||||
%% @spec send_09(CSocket, Folder) -> ok
|
||||
%% @doc Change folder command.
|
||||
send_09(CSocket, Folder) ->
|
||||
FolderBin = list_to_binary(Folder),
|
||||
Padding = 8 * (64 - length(Folder)),
|
||||
Bin = << 16#48:32/little, 16#09:32/little, FolderBin/binary, 0:Padding >>,
|
||||
gen_tcp:send(CSocket, Bin).
|
||||
|
||||
%% @spec send_0a(CSocket) -> ok
|
||||
%% @doc Back to root folder command.
|
||||
send_0a(CSocket) ->
|
||||
Bin = << 16#8:32/little, 16#0a:32/little >>,
|
||||
gen_tcp:send(CSocket, Bin).
|
||||
|
||||
%% @spec send_0f(CSocket, TotalSize, NbFiles) -> ok
|
||||
%% @doc General update information command. Prepare the update screen.
|
||||
send_0f(CSocket, Size, NbFiles) ->
|
||||
Bin = << 16#10:32/little, 16#0f:32/little, Size:32/little, NbFiles:32/little >>,
|
||||
gen_tcp:send(CSocket, Bin).
|
||||
|
||||
%% @spec send_10(CSocket, Size, FilenameBin) -> ok
|
||||
%% @doc File update begin command. Prepare sending an individual file.
|
||||
send_10(CSocket, Size, FilenameBin) ->
|
||||
Bin = << 16#50:32/little, 16#10:32/little, 0:32, Size:32/little, FilenameBin/binary >>,
|
||||
gen_tcp:send(CSocket, Bin).
|
||||
|
||||
%% @spec send_11(CSocket, Data, N) -> ok
|
||||
%% @doc Command to send a file fragment.
|
||||
send_11(CSocket, Data, N) ->
|
||||
DataSize = byte_size(Data),
|
||||
Padding = case DataSize rem 4 of
|
||||
0 -> 0;
|
||||
Rem -> 8 * (4 - Rem)
|
||||
end,
|
||||
Data2 = << Data/binary, 0:Padding >>,
|
||||
DataSize2 = DataSize + Padding div 8,
|
||||
Size = DataSize2 + 16#14,
|
||||
CRC = erlang:crc32(Data2),
|
||||
Bin = << Size:32/little, 16#11:32/little, N:32/little, CRC:32/little, DataSize:32/little, Data2/binary >>,
|
||||
gen_tcp:send(CSocket, Bin).
|
||||
|
||||
%% @spec send_12(CSocket) -> ok
|
||||
%% @doc File update end command.
|
||||
send_12(CSocket) ->
|
||||
Bin = << 16#8:32/little, 16#12:32/little >>,
|
||||
gen_tcp:send(CSocket, Bin).
|
||||
|
||||
%% @spec send_13(CSocket) -> ok
|
||||
%% @doc Update complete command. Usually followed by the server closing the connection.
|
||||
send_13(CSocket) ->
|
||||
Bin = << 16#8:32/little, 16#13:32/little >>,
|
||||
gen_tcp:send(CSocket, Bin).
|
@ -51,13 +51,14 @@ init([]) ->
|
||||
PatchPorts = egs_conf:read(patch_ports),
|
||||
LoginPorts = egs_conf:read(login_ports),
|
||||
{_ServerIP, GamePort} = egs_conf:read(game_server),
|
||||
PatchProcs = [{{egs_patch_server, Port}, {psu_patch, start_link, [Port]}, permanent, 5000, worker, dynamic} || Port <- PatchPorts],
|
||||
PatchProcs = [{{egs_patch_server, Port}, {egs_patch_server, start_link, [Port]}, permanent, 5000, worker, dynamic} || Port <- PatchPorts],
|
||||
LoginProcs = [{{egs_login_server, Port}, {egs_login_server, start_link, [Port]}, permanent, 5000, worker, dynamic} || Port <- LoginPorts],
|
||||
OtherProcs = [
|
||||
{egs_seasons, {egs_seasons, start_link, []}, permanent, 5000, worker, dynamic},
|
||||
{egs_counters_db, {egs_counters_db, start_link, []}, permanent, 5000, worker, dynamic},
|
||||
{egs_items_db, {egs_items_db, start_link, []}, permanent, 5000, worker, dynamic},
|
||||
{egs_npc_db, {egs_npc_db, start_link, []}, permanent, 5000, worker, dynamic},
|
||||
{egs_patch_files_db, {egs_patch_files_db, start_link, []}, permanent, 5000, worker, dynamic},
|
||||
{egs_quests_db, {egs_quests_db, start_link, []}, permanent, 5000, worker, dynamic},
|
||||
{egs_shops_db, {egs_shops_db, start_link, []}, permanent, 5000, worker, dynamic},
|
||||
{egs_accounts, {egs_accounts, start_link, []}, permanent, 5000, worker, dynamic},
|
||||
|
@ -1,67 +0,0 @@
|
||||
%% @author Loïc Hoguin <essen@dev-extend.eu>
|
||||
%% @copyright 2010 Loïc Hoguin.
|
||||
%% @doc Process patch requests.
|
||||
%%
|
||||
%% 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 <http://www.gnu.org/licenses/>.
|
||||
|
||||
-module(psu_patch).
|
||||
-export([start_link/1]). %% External.
|
||||
-export([listen/1, accept/1, process/1]). %% Internal
|
||||
|
||||
-define(OPTIONS, [binary, {send_timeout, 5000}, {packet, 0}, {active, false}, {reuseaddr, true}]).
|
||||
|
||||
%% @spec start_link(Port) -> {ok,Pid::pid()}
|
||||
%% @doc Start the PSU patch server for inclusion in a supervisor tree.
|
||||
start_link(Port) ->
|
||||
Pid = spawn(?MODULE, listen, [Port]),
|
||||
{ok, Pid}.
|
||||
|
||||
%% @spec listen(Port) -> ok
|
||||
%% @doc Listen for connections.
|
||||
listen(Port) ->
|
||||
error_logger:info_report(io_lib:format("psu_patch listening on port ~b", [Port])),
|
||||
{ok, LSocket} = gen_tcp:listen(Port, ?OPTIONS),
|
||||
?MODULE:accept(LSocket).
|
||||
|
||||
%% @spec accept(LSocket) -> ok
|
||||
%% @doc Accept connections.
|
||||
accept(LSocket) ->
|
||||
case gen_tcp:accept(LSocket, 5000) of
|
||||
{ok, CSocket} ->
|
||||
spawn(?MODULE, process, [CSocket]);
|
||||
{error, timeout} ->
|
||||
reload
|
||||
end,
|
||||
?MODULE:accept(LSocket).
|
||||
|
||||
%% @spec process(CSocket) -> ok
|
||||
%% @doc Fake the patch server by sending what the game wants to hear: no updates available.
|
||||
process(CSocket) ->
|
||||
io:format("faking patch server: no updates~n"),
|
||||
send_packet(CSocket, "priv/patch/patch-0.bin"),
|
||||
gen_tcp:recv(CSocket, 0, 5000),
|
||||
send_packet(CSocket, "priv/patch/patch-1.bin"),
|
||||
send_packet(CSocket, "priv/patch/patch-2.bin"),
|
||||
gen_tcp:recv(CSocket, 0, 5000),
|
||||
send_packet(CSocket, "priv/patch/patch-3.bin"),
|
||||
send_packet(CSocket, "priv/patch/patch-4.bin"),
|
||||
gen_tcp:close(CSocket).
|
||||
|
||||
%% @spec send_packet(CSocket, PacketFilename) -> ok
|
||||
%% @doc Send a packet from a file.
|
||||
send_packet(CSocket, PacketFilename) ->
|
||||
{ok, Packet} = file:read_file(PacketFilename),
|
||||
gen_tcp:send(CSocket, Packet).
|
Loading…
Reference in New Issue
Block a user