Merge branches 'kallfaktorn_rewrite' and 'master' of github.com:jeena/GGS
This commit is contained in:
commit
32d9622da8
15 changed files with 369 additions and 158 deletions
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -2,3 +2,4 @@
|
||||||
*.dump
|
*.dump
|
||||||
*.beam
|
*.beam
|
||||||
Mnesia.*
|
Mnesia.*
|
||||||
|
*.swo
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
Subproject commit 5350ed21606606dbee5ecb07e974f2abb9106270
|
Subproject commit 709b568efbc99c954507d1593bc5633f900bc5dc
|
|
@ -1,50 +0,0 @@
|
||||||
%%%%----------------------------------------------------
|
|
||||||
%%% @author Mattias Pettersson <mattiaspgames@gmail.com>
|
|
||||||
%%% @copyright 2011 Mattias Pettersson
|
|
||||||
%%% @doc Database for runtime game variable storage.
|
|
||||||
%%% @end
|
|
||||||
|
|
||||||
Test Mnesia
|
|
||||||
-module(gamedb).
|
|
||||||
-import(mnesia).
|
|
||||||
-export([init/0,insert_player/1,example_player/0,read_player/1,test_player/0]).
|
|
||||||
-include("gamedb.hrl").
|
|
||||||
|
|
||||||
%%-----------------------------------------------------
|
|
||||||
%% Creation
|
|
||||||
%%-----------------------------------------------------
|
|
||||||
init() ->
|
|
||||||
mnesia:create_table(player, [{attributes, record_info(fields, player)}]).
|
|
||||||
|
|
||||||
%%-----------------------------------------------------
|
|
||||||
%% Test
|
|
||||||
%%-----------------------------------------------------
|
|
||||||
test_player() ->
|
|
||||||
insert_player(example_player()),
|
|
||||||
read_player(0001).
|
|
||||||
|
|
||||||
example_player() ->
|
|
||||||
#player{id = 0001,
|
|
||||||
name = "Tux"}.
|
|
||||||
|
|
||||||
%%-----------------------------------------------------
|
|
||||||
%% Insertions
|
|
||||||
%%-----------------------------------------------------
|
|
||||||
insert_player(Player) ->
|
|
||||||
Fun = fun() ->
|
|
||||||
mnesia:write(Player)
|
|
||||||
end,
|
|
||||||
mnesia:transaction(Fun).
|
|
||||||
|
|
||||||
|
|
||||||
%%-----------------------------------------------------
|
|
||||||
%% Querries
|
|
||||||
%%-----------------------------------------------------
|
|
||||||
read_player(Player_Key) ->
|
|
||||||
Fun = fun() ->
|
|
||||||
[P] = mnesia:read(player, Player_Key),
|
|
||||||
Name = P#player.name,
|
|
||||||
io:format("Player name: ~s~n",[Name])
|
|
||||||
end,
|
|
||||||
mnesia:transaction(Fun).
|
|
||||||
|
|
Binary file not shown.
Binary file not shown.
|
@ -1,13 +1,20 @@
|
||||||
-module(ggs_coordinator).
|
-module(ggs_coordinator).
|
||||||
|
|
||||||
%% API Exports
|
%% API Exports
|
||||||
-export([start_link/0, stop/1]).
|
-export([start_link/0, stop/1, join_table/1, create_table/1, join_lobby/0,
|
||||||
|
respawn_player/2, respawn_table/1, remove_player/2]).
|
||||||
|
|
||||||
%% gen_server callback exports
|
%% gen_server callback exports
|
||||||
-export([init/1, handle_call/3, handle_cast/2, handle_info/2, terminate/2,
|
-export([init/1, handle_call/3, handle_cast/2, handle_info/2, terminate/2,
|
||||||
code_change/3]).
|
code_change/3]).
|
||||||
-define(SERVER, ?MODULE).
|
-define(SERVER, ?MODULE).
|
||||||
|
|
||||||
|
-record(co_state,
|
||||||
|
{players = [], % List of all player processes
|
||||||
|
player_table_map = [], % Players <-> Table map
|
||||||
|
table_state_map = [],
|
||||||
|
tables = []}). % Table <-> Table state map
|
||||||
|
|
||||||
%% @doc This module act as "the man in the middle".
|
%% @doc This module act as "the man in the middle".
|
||||||
%% Creates the starting connection between table and players.
|
%% Creates the starting connection between table and players.
|
||||||
|
|
||||||
|
@ -16,21 +23,22 @@ start_link() ->
|
||||||
gen_server:start_link({local, ?SERVER}, ?MODULE, [], []).
|
gen_server:start_link({local, ?SERVER}, ?MODULE, [], []).
|
||||||
|
|
||||||
%% @doc Terminates the coordinator process.
|
%% @doc Terminates the coordinator process.
|
||||||
stop(_Reason) ->
|
stop(Reason) ->
|
||||||
ggs_logger:not_implemented().
|
gen_server:cast(ggs_coordinator, {stop, Reason}).
|
||||||
|
|
||||||
%% @doc Joins table with specified token
|
%% @doc Joins table with specified token, returns {error, no_such_table}
|
||||||
join_table(_Token) ->
|
%% if the specified table token does not exist
|
||||||
ggs_logger:not_implemented().
|
join_table(Token) ->
|
||||||
|
gen_server:call(ggs_coordinator, {join_table, Token}).
|
||||||
|
|
||||||
%% @doc Create a new table
|
%% @doc Create a new table, return {error, Reason} or {ok, TableToken}
|
||||||
create_table(_Params) ->
|
create_table(Params) ->
|
||||||
ggs_logger:not_implemented().
|
gen_server:call(ggs_coordinator, {create_table, Params}).
|
||||||
|
|
||||||
%% @doc This is the first function run by a newly created players.
|
%% @doc This is the first function run by a newly created players.
|
||||||
%% Generates a unique token that we use to identify the player.
|
%% Generates a unique token that we use to identify the player.
|
||||||
join_lobby() ->
|
join_lobby() ->
|
||||||
ggs_logger:not_implemented().
|
gen_server:call(ggs_coordinator, join_lobby).
|
||||||
|
|
||||||
%% @doc Act as a supervisor to player and respawns player when it gets bad data.
|
%% @doc Act as a supervisor to player and respawns player when it gets bad data.
|
||||||
respawn_player(_Player, _Socket) ->
|
respawn_player(_Player, _Socket) ->
|
||||||
|
@ -47,11 +55,38 @@ remove_player(_From, _Player) ->
|
||||||
%% gen_server callbacks
|
%% gen_server callbacks
|
||||||
|
|
||||||
init([]) ->
|
init([]) ->
|
||||||
{ok, ok}.
|
{ok, #co_state{}}.
|
||||||
|
|
||||||
|
handle_call(join_lobby, _From, State) ->
|
||||||
|
Token = helpers:get_new_token(),
|
||||||
|
{reply, {ok, Token}, State};
|
||||||
|
|
||||||
|
handle_call({join_table, Table}, From, State) ->
|
||||||
|
{FromPlayer, _Ref} = From,
|
||||||
|
Tables = State#co_state.tables,
|
||||||
|
case lists:keyfind(Table, 1, Tables) of
|
||||||
|
{TableID, TablePID} ->
|
||||||
|
ggs_table:add_player(TablePID, FromPlayer),
|
||||||
|
{reply, {ok, TablePID}, State};
|
||||||
|
false ->
|
||||||
|
{reply, {error, no_such_table}, State}
|
||||||
|
end;
|
||||||
|
|
||||||
|
handle_call({create_table, {force, TableID}}, From, State) ->
|
||||||
|
TableIDMap = State#co_state.player_table_map,
|
||||||
|
Tables = State#co_state.tables,
|
||||||
|
NewTableProc = ggs_table:start_link(),
|
||||||
|
{reply, {ok, TableID}, State#co_state{
|
||||||
|
player_table_map = [{From, TableID} | TableIDMap],
|
||||||
|
tables = [{TableID, NewTableProc} | Tables]
|
||||||
|
}};
|
||||||
|
|
||||||
handle_call(_Message, _From, State) ->
|
handle_call(_Message, _From, State) ->
|
||||||
{noreply, State}.
|
{noreply, State}.
|
||||||
|
|
||||||
|
handle_cast({stop, Reason}, State) ->
|
||||||
|
{stop, normal, state};
|
||||||
|
|
||||||
handle_cast(_Message, State) ->
|
handle_cast(_Message, State) ->
|
||||||
{noreply, State}.
|
{noreply, State}.
|
||||||
|
|
||||||
|
|
|
@ -1,24 +1,34 @@
|
||||||
-module(ggs_gamevm).
|
|
||||||
-export([start_link/0, define/2, user_command/4]).
|
|
||||||
%% @doc This module is responsible for running the game VM:s. You can issue
|
%% @doc This module is responsible for running the game VM:s. You can issue
|
||||||
%% commands to a vm using this module.
|
%% commands to a vm using this module.
|
||||||
|
|
||||||
|
-module(ggs_gamevm).
|
||||||
|
-behaviour(gen_server).
|
||||||
|
|
||||||
|
%% gen_server callbacks
|
||||||
|
-export([init/1, handle_call/3, handle_cast/2, handle_info/2,
|
||||||
|
terminate/2, code_change/3]).
|
||||||
|
|
||||||
|
-record(state, { port, table } ).
|
||||||
|
|
||||||
|
%% API
|
||||||
|
-export([start_link/1, define/2, user_command/4, stop/1, call_js/2]).
|
||||||
|
|
||||||
|
-include_lib("eunit/include/eunit.hrl").
|
||||||
|
|
||||||
|
|
||||||
|
%% ----------------------------------------------------------------------
|
||||||
|
% API implementation
|
||||||
|
|
||||||
%% @doc Create a new VM process. The process ID is returned and can be used
|
%% @doc Create a new VM process. The process ID is returned and can be used
|
||||||
%% with for example the define method of this module.
|
%% with for example the define method of this module.
|
||||||
start_link() ->
|
start_link(Table) ->
|
||||||
erlang_js:start(), %% @TODO: should only be done once
|
erlang_js:start(), %% @TODO: should only be done once
|
||||||
PortPid = spawn_link( fun() ->
|
{ok, Pid} = gen_server:start_link(?MODULE, [Table], []),
|
||||||
process_flag(trap_exit, true),
|
Pid.
|
||||||
{ok, Port} = js_driver:new(),
|
|
||||||
js:define(Port, <<"function userCommand(user, command, args){return 'Hello world';}">>),
|
|
||||||
loop(Port)
|
|
||||||
end ),
|
|
||||||
PortPid.
|
|
||||||
|
|
||||||
%% @doc Define some new code on the specified VM, returns the atom ok.
|
%% @doc Define some new code on the specified VM, returns the atom ok.
|
||||||
define(GameVM, SourceCode) ->
|
define(GameVM, SourceCode) ->
|
||||||
GameVM ! {define,SourceCode},
|
gen_server:cast(GameVM, {define, SourceCode}).
|
||||||
ok.
|
|
||||||
|
|
||||||
%% @doc Execute a user command on the specified VM. This function is
|
%% @doc Execute a user command on the specified VM. This function is
|
||||||
%% asynchronous, and returns ok.
|
%% asynchronous, and returns ok.
|
||||||
|
@ -28,23 +38,93 @@ define(GameVM, SourceCode) ->
|
||||||
%% Command = a game command to run
|
%% Command = a game command to run
|
||||||
%% Args = arguments for the Command parameter
|
%% Args = arguments for the Command parameter
|
||||||
user_command(GameVM, Player, Command, Args) ->
|
user_command(GameVM, Player, Command, Args) ->
|
||||||
Ref = make_ref(),
|
gen_server:cast(GameVM, {user_command, Player, Command, Args}).
|
||||||
GameVM ! {user_command, Player, Command, Args, self(), Ref},
|
|
||||||
|
%% @private
|
||||||
|
% only for tests
|
||||||
|
call_js(GameVM, SourceCode) ->
|
||||||
|
gen_server:call(GameVM, {eval, SourceCode}).
|
||||||
|
|
||||||
|
% @doc stops the gamevm process
|
||||||
|
stop(GameVM) ->
|
||||||
|
gen_server:cast(GameVM, stop).
|
||||||
|
|
||||||
|
|
||||||
|
%% ----------------------------------------------------------------------
|
||||||
|
|
||||||
|
%% @private
|
||||||
|
init([Table]) ->
|
||||||
|
process_flag(trap_exit, true),
|
||||||
|
{ok, Port} = js_driver:new(),
|
||||||
|
%% @TODO: add here default JS API instead
|
||||||
|
{ok, #state { port = Port, table = Table }}.
|
||||||
|
|
||||||
|
%% private
|
||||||
|
% only needed for the tests
|
||||||
|
handle_call({eval, SourceCode}, _From, #state { port = Port } = State) ->
|
||||||
|
{ok, Ret} = js:eval(Port, list_to_binary(SourceCode)),
|
||||||
|
{reply, Ret, State}.
|
||||||
|
|
||||||
|
%% @private
|
||||||
|
handle_cast({define, SourceCode}, #state { port = Port } = State) ->
|
||||||
|
ok = js:define(Port, list_to_binary(SourceCode)),
|
||||||
|
{noreply, State};
|
||||||
|
handle_cast({user_command, Player, Command, Args}, #state { port = Port } = State) ->
|
||||||
|
Arguments = string:concat("'", string:concat(
|
||||||
|
string:join([js_escape(Player), js_escape(Command), js_escape(Args)], "','"), "'")),
|
||||||
|
Js = list_to_binary(string:concat(string:concat("userCommand(", Arguments), ");")),
|
||||||
|
js_driver:define_js(Port, Js),
|
||||||
|
{noreply, State};
|
||||||
|
handle_cast(stop, State) ->
|
||||||
|
{stop, normal, State};
|
||||||
|
handle_cast(Msg, S) ->
|
||||||
|
error_logger:error_report([unknown_msg, Msg]),
|
||||||
|
{noreply, S}.
|
||||||
|
|
||||||
|
%% @private
|
||||||
|
handle_info(Msg, S) ->
|
||||||
|
error_logger:error_report([unknown_msg, Msg]),
|
||||||
|
{noreply, S}.
|
||||||
|
|
||||||
|
%% @private
|
||||||
|
terminate(_Reason, _State) ->
|
||||||
ok.
|
ok.
|
||||||
|
|
||||||
%% Helper functions
|
%% @private
|
||||||
|
code_change(_OldVsn, State, _Extra) ->
|
||||||
|
{ok, State}.
|
||||||
|
|
||||||
|
js_escape(S) ->
|
||||||
|
lists:flatmap(fun($\') -> [$\\, $\']; (X) -> [X] end, S).
|
||||||
|
|
||||||
|
%% ----------------------------------------------------------------------
|
||||||
|
% Tests
|
||||||
|
|
||||||
|
start_link_test() ->
|
||||||
|
erlang_js:start(), %% @TODO: should only be done once
|
||||||
|
GameVM = start_link(test_table),
|
||||||
|
?assertNot(GameVM =:= undefined).
|
||||||
|
|
||||||
|
define_test() ->
|
||||||
|
GameVM = start_link(test_table),
|
||||||
|
define(GameVM, "function hello(test) { return test; }"),
|
||||||
|
?assertMatch(<<"jeena">>, gen_server:call(GameVM, {eval, "hello('jeena')"})).
|
||||||
|
|
||||||
|
stop_test() ->
|
||||||
|
GameVM = start_link(test_table),
|
||||||
|
ok = stop(GameVM).
|
||||||
|
|
||||||
|
user_command_test() ->
|
||||||
|
GameVM = start_link(test_table),
|
||||||
|
define(GameVM, "var t = '';\nfunction userCommand(user, command, args) { t = user + command + args; }\n"),
|
||||||
|
user_command(GameVM, "'jeena", "thecommand", "theargs'"),
|
||||||
|
?assertMatch(<<"'jeenathecommandtheargs'">>, gen_server:call(GameVM, {eval, "t;"})).
|
||||||
|
|
||||||
|
js_erlang_test() ->
|
||||||
|
GameVM = start_link(test_table),
|
||||||
|
define(GameVM, "var t = '';\nfunction userCommand(user, command, args) { t = callErlang('erlang time') + ''; }\n"),
|
||||||
|
user_command(GameVM, "", "", ""),
|
||||||
|
{A, B, C} = erlang:time(),
|
||||||
|
T = "{" ++ integer_to_list(A) ++ ", " ++ integer_to_list(B) ++ ", " ++ integer_to_list(C) ++ "}",
|
||||||
|
?assertMatch(T, binary_to_list(gen_server:call(GameVM, {eval, "t;"}))).
|
||||||
|
|
||||||
loop(Port) ->
|
|
||||||
receive
|
|
||||||
{define, SourceCode} ->
|
|
||||||
ok = js:define(Port, list_to_binary(SourceCode)),
|
|
||||||
loop(Port);
|
|
||||||
{user_command, User, Command, Args, From, Ref} ->
|
|
||||||
{ok, Ret} = js:call(Port, <<"userCommand">>,
|
|
||||||
[ list_to_binary(User),
|
|
||||||
list_to_binary(Command),
|
|
||||||
list_to_binary(Args)
|
|
||||||
]),
|
|
||||||
From ! {Ref, Ret},
|
|
||||||
loop(Port)
|
|
||||||
end.
|
|
||||||
|
|
41
src/ggs_gamevm_e.erl
Normal file
41
src/ggs_gamevm_e.erl
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
-module(ggs_gamevm_e).
|
||||||
|
-export([start_link/1, define/2, user_command/4]).
|
||||||
|
%% @doc This module is responsible for running the game VM:s. You can issue
|
||||||
|
%% commands to a vm using this module.
|
||||||
|
|
||||||
|
%% @doc Create a new VM process. The process ID is returned and can be used
|
||||||
|
%% with for example the define method of this module.
|
||||||
|
start_link(Table) ->
|
||||||
|
PortPid = spawn( fun() ->
|
||||||
|
loop(Table)
|
||||||
|
end ),
|
||||||
|
PortPid.
|
||||||
|
|
||||||
|
%% @doc Define some new code on the specified VM, returns the atom ok.
|
||||||
|
define(GameVM, SourceCode) ->
|
||||||
|
GameVM ! {define,SourceCode},
|
||||||
|
ok.
|
||||||
|
|
||||||
|
%% @doc Execute a user command on the specified VM. This function is
|
||||||
|
%% asynchronous, and returns ok.
|
||||||
|
%% @spec user_command(GameVM, User, Command, Args) -> ok
|
||||||
|
%% GameVM = process IS of VM
|
||||||
|
%% Player = the player running the command
|
||||||
|
%% Command = a game command to run
|
||||||
|
%% Args = arguments for the Command parameter
|
||||||
|
user_command(GameVM, Player, Command, Args) ->
|
||||||
|
Ref = make_ref(),
|
||||||
|
GameVM ! {user_command, Player, Command, Args, self(), Ref},
|
||||||
|
ok.
|
||||||
|
|
||||||
|
%% Helper functions
|
||||||
|
|
||||||
|
loop(Table) ->
|
||||||
|
receive
|
||||||
|
{define, SourceCode} ->
|
||||||
|
loop(Table);
|
||||||
|
{user_command, _User, Command, _Args, _From, _Ref} ->
|
||||||
|
io:format("GameVM received a message~n"),
|
||||||
|
ggs_table:notify_all_players(Table, Command),
|
||||||
|
loop(Table)
|
||||||
|
end.
|
|
@ -1,5 +1,9 @@
|
||||||
-module(ggs_player).
|
-module(ggs_player).
|
||||||
-export([start_link/1, notify/3, get_token/1, stop/2]).
|
-export([start_link/1, notify/3, get_token/1, stop/2]).
|
||||||
|
-record(pl_state,
|
||||||
|
{token, % Player's token
|
||||||
|
socket, % Player's socket
|
||||||
|
table}). % Player's table
|
||||||
|
|
||||||
%% @doc This module handles communication between a player and GGS. This module is
|
%% @doc This module handles communication between a player and GGS. This module is
|
||||||
%%responsible for:
|
%%responsible for:
|
||||||
|
@ -13,14 +17,28 @@
|
||||||
%% identifying the player.
|
%% identifying the player.
|
||||||
%% @spec start_link(Socket::socket()) -> {ok, Pid} | {error, Reason}
|
%% @spec start_link(Socket::socket()) -> {ok, Pid} | {error, Reason}
|
||||||
start_link(Socket) ->
|
start_link(Socket) ->
|
||||||
loop(Socket).
|
% The socket is in 'active' mode, and that means we are pushed any data
|
||||||
|
% that arrives on it, we do not need to recv() manually. Since the socket
|
||||||
|
% was opened in our parent process, we need to change the owner of it to
|
||||||
|
% us, otherwise these messages end up in our parent.
|
||||||
|
erlang:port_connect(Socket, self()),
|
||||||
|
{ok, Token} = ggs_coordinator:join_lobby(),
|
||||||
|
TableStatus = ggs_coordinator:join_table(1337),
|
||||||
|
case TableStatus of
|
||||||
|
{ok, Table} ->
|
||||||
|
loop(#pl_state{socket = Socket, token = Token, table = Table});
|
||||||
|
{error, no_such_table} ->
|
||||||
|
ggs_coordinator:create_table({force, 1337}),
|
||||||
|
{ok, Table} = ggs_coordinator:join_table(1337),
|
||||||
|
loop(#pl_state{socket = Socket, token = Token, table = Table})
|
||||||
|
end.
|
||||||
|
|
||||||
%% @doc Handles incoming messages from the GGS and forwards them through the player
|
%% @doc Handles incoming messages from the GGS and forwards them through the player
|
||||||
%% socket to the player.
|
%% socket to the player.
|
||||||
%% @spec notify(Player::Pid(), From::Pid(),
|
%% @spec notify(Player::Pid(), From::Pid(),
|
||||||
%% {Command::String(), Message::string()}) -> ok
|
%% {Command::String(), Message::string()}) -> ok
|
||||||
notify(Player, From, Message) ->
|
notify(Player, From, Message) ->
|
||||||
ggs_logger:not_implemented().
|
Player ! {notify, From, Message}.
|
||||||
|
|
||||||
%% @doc Get the player token uniquely representing the player.
|
%% @doc Get the player token uniquely representing the player.
|
||||||
%% @spec get_token() -> string()
|
%% @spec get_token() -> string()
|
||||||
|
@ -36,13 +54,13 @@ stop(_Player,_Table) ->
|
||||||
|
|
||||||
%% Internals
|
%% Internals
|
||||||
|
|
||||||
loop(Socket) ->
|
loop(#pl_state{token = Token, socket = Socket, table = Table} = State) ->
|
||||||
% The socket is in 'active' mode, and that means we are pushed any data
|
receive
|
||||||
% that arrives on it, we do not need to recv() manually. Since the socket
|
{tcp, Socket, Data} -> % Just echo for now..
|
||||||
% was opened in our parent process, we need to change the owner of it to
|
io:format("Notifying table..~n"),
|
||||||
% us, otherwise these messages end up in our parent.
|
ggs_table:notify_game(Table, Token, Data),
|
||||||
erlang:port_connect(Socket, self()),
|
loop(State);
|
||||||
receive {tcp, Socket, Data} -> % Just echo for now..
|
{notify, From, Message} ->
|
||||||
gen_tcp:send(Socket,Data),
|
gen_tcp:send(Socket, Message),
|
||||||
loop(Socket)
|
loop(State)
|
||||||
end.
|
end.
|
||||||
|
|
|
@ -1,31 +1,29 @@
|
||||||
%% @doc This module represents a Player with a Socket and a Token
|
%% @doc This module represents a table with players
|
||||||
|
|
||||||
-module(ggs_table).
|
-module(ggs_table).
|
||||||
-behaviour(gen_server).
|
-behaviour(gen_server).
|
||||||
|
|
||||||
%% gen_server callbacks
|
%% gen_server callbacks
|
||||||
-export([init/1, handle_call/3, handle_cast/2, handle_info/2,
|
-export([init/1, handle_call/3, handle_cast/2, handle_info/2,
|
||||||
terminate/2, code_change/3]).
|
terminate/2, code_change/3, notify_all_players/2, notify_game/3,
|
||||||
|
add_player/2]).
|
||||||
|
|
||||||
-record(state, { token, players, socket, game_vm } ).
|
-record(state, { players, game_vm } ).
|
||||||
|
|
||||||
%% API
|
%% API
|
||||||
-export([start_link/2,
|
-export([start_link/0,
|
||||||
add_player/2,
|
add_player/2,
|
||||||
remove_player/2,
|
remove_player/2,
|
||||||
stop/1,
|
stop/1,
|
||||||
notify/3]).
|
notify/3]).
|
||||||
|
|
||||||
-include_lib("eunit/include/eunit.hrl").
|
|
||||||
|
|
||||||
%% ----------------------------------------------------------------------
|
%% ----------------------------------------------------------------------
|
||||||
% API implementation
|
% API implementation
|
||||||
|
|
||||||
% @doc returns a new table
|
% @doc returns a new table
|
||||||
start_link(Token, Socket) ->
|
start_link() ->
|
||||||
GameVM = ggs_gamevm:start_link(),
|
{ok, Pid} = gen_server:start_link(?MODULE, [], []),
|
||||||
{ok, Pid} = gen_server:start_link(?MODULE, [Token, Socket, GameVM], []),
|
|
||||||
Pid.
|
|
||||||
|
|
||||||
%% @private
|
%% @private
|
||||||
call(Pid, Msg) ->
|
call(Pid, Msg) ->
|
||||||
|
@ -47,22 +45,34 @@ stop(Table) ->
|
||||||
notify(Table, Player, Message) ->
|
notify(Table, Player, Message) ->
|
||||||
gen_server:cast(Table, {notify, Player, Message}).
|
gen_server:cast(Table, {notify, Player, Message}).
|
||||||
|
|
||||||
|
notify_all_players(Table, Message) ->
|
||||||
|
gen_server:cast(Table, {notify_all_players, Message}).
|
||||||
|
|
||||||
|
notify_game(Table, From, Message) ->
|
||||||
|
io:format("Notify game called on"),
|
||||||
|
erlang:display(Table),
|
||||||
|
io:format("~n"),
|
||||||
|
gen_server:cast(Table, {notify_game, Message, From}).
|
||||||
|
|
||||||
%% ----------------------------------------------------------------------
|
%% ----------------------------------------------------------------------
|
||||||
|
|
||||||
%% @private
|
%% @private
|
||||||
init([Token, Socket, GameVM]) ->
|
init([]) ->
|
||||||
{ok, #state { token = Token,
|
GameVM = ggs_gamevm_e:start_link(self()), %% @TODO: Temporary erlang gamevm
|
||||||
socket = Socket,
|
{ok, #state {
|
||||||
game_vm = GameVM,
|
game_vm = GameVM,
|
||||||
players = [] }}.
|
players = [] }}.
|
||||||
|
|
||||||
%% @private
|
%% @private
|
||||||
handle_call({add_player, Player}, _From, #state { players = Players } = State) ->
|
handle_call({add_player, Player}, _From, #state { players = Players } = State) ->
|
||||||
{reply, ok, State#state { players = [Player | Players] }};
|
{reply, ok, State#state { players = [Player | Players] }};
|
||||||
|
|
||||||
handle_call({remove_player, Player}, _From, #state { players = Players } = State) ->
|
handle_call({remove_player, Player}, _From, #state { players = Players } = State) ->
|
||||||
{reply, ok, State#state { players = Players -- [Player] }};
|
{reply, ok, State#state { players = Players -- [Player] }};
|
||||||
|
|
||||||
handle_call(get_player_list, _From, #state { players = Players } = State) ->
|
handle_call(get_player_list, _From, #state { players = Players } = State) ->
|
||||||
{reply, {ok, Players}, State};
|
{reply, {ok, Players}, State};
|
||||||
|
|
||||||
handle_call(Msg, _From, State) ->
|
handle_call(Msg, _From, State) ->
|
||||||
error_logger:error_report([unknown_msg, Msg]),
|
error_logger:error_report([unknown_msg, Msg]),
|
||||||
{reply, ok, State}.
|
{reply, ok, State}.
|
||||||
|
@ -71,11 +81,25 @@ handle_call(Msg, _From, State) ->
|
||||||
handle_cast({notify, Player, Message}, #state { game_vm = GameVM } = State) ->
|
handle_cast({notify, Player, Message}, #state { game_vm = GameVM } = State) ->
|
||||||
case Message of
|
case Message of
|
||||||
{server, define, Args} ->
|
{server, define, Args} ->
|
||||||
ggs_gamevm:define(GameVM, Args);
|
ggs_gamevm_e:define(GameVM, Args);
|
||||||
{game, Command, Args} ->
|
{game, Command, Args} ->
|
||||||
ggs_gamevm:user_command(GameVM, Player, Command, Args)
|
ggs_gamevm_e:user_command(GameVM, Player, Command, Args)
|
||||||
end,
|
end,
|
||||||
{noreply, State};
|
{noreply, State};
|
||||||
|
|
||||||
|
handle_cast({notify_game, Message, From}, #state { game_vm = GameVM } = State) ->
|
||||||
|
io:format("notify_game message received~n"),
|
||||||
|
ggs_gamevm_e:user_command(GameVM, From, Message, ""),
|
||||||
|
{noreply, State};
|
||||||
|
|
||||||
|
handle_cast({notify_all_players, Message}, #state{players = Players} = State) ->
|
||||||
|
io:format("Notifying all players... ~p~n", [Players]),
|
||||||
|
lists:foreach(fun(P) ->
|
||||||
|
io:format("Notifying ~p~n", [P]),
|
||||||
|
ggs_player:notify(P, "Server", Message)
|
||||||
|
end, Players),
|
||||||
|
{noreply, State};
|
||||||
|
|
||||||
handle_cast(stop, State) ->
|
handle_cast(stop, State) ->
|
||||||
{stop, normal, State};
|
{stop, normal, State};
|
||||||
handle_cast(Msg, S) ->
|
handle_cast(Msg, S) ->
|
||||||
|
@ -95,44 +119,3 @@ terminate(_Reason, _State) ->
|
||||||
code_change(_OldVsn, State, _Extra) ->
|
code_change(_OldVsn, State, _Extra) ->
|
||||||
{ok, State}.
|
{ok, State}.
|
||||||
|
|
||||||
|
|
||||||
%% ----------------------------------------------------------------------
|
|
||||||
|
|
||||||
% Tests
|
|
||||||
|
|
||||||
start_link_test() ->
|
|
||||||
Table = start_link("123", none),
|
|
||||||
?assertNot(Table =:= undefined).
|
|
||||||
|
|
||||||
add_player_test() ->
|
|
||||||
Table = start_link("123", none),
|
|
||||||
Player = test_player,
|
|
||||||
add_player(Table, Player),
|
|
||||||
{ok, [Player]} = gen_server:call(Table, get_player_list).
|
|
||||||
|
|
||||||
remove_player_test() ->
|
|
||||||
Table = start_link("123", none),
|
|
||||||
Player = test_player,
|
|
||||||
Player2 = test_player2,
|
|
||||||
add_player(Table, Player),
|
|
||||||
{ok, [Player]} = gen_server:call(Table, get_player_list),
|
|
||||||
add_player(Table, Player2),
|
|
||||||
{ok, [Player2, Player]} = gen_server:call(Table, get_player_list),
|
|
||||||
remove_player(Table, Player),
|
|
||||||
{ok, [Player2]} = gen_server:call(Table, get_player_list),
|
|
||||||
remove_player(Table, Player2),
|
|
||||||
{ok, []} = gen_server:call(Table, get_player_list).
|
|
||||||
|
|
||||||
stop_test() ->
|
|
||||||
Table = start_link("123", none),
|
|
||||||
ok = stop(Table).
|
|
||||||
|
|
||||||
% @private
|
|
||||||
notify_test() ->
|
|
||||||
Table = start_link("123", none),
|
|
||||||
Player = test_player,
|
|
||||||
Message = {server, define, "function helloWorld(x) { }"},
|
|
||||||
ok = notify(Table, Player, Message).
|
|
||||||
%Message2 = {game, "helloWorld", "test"},
|
|
||||||
%ok = notify(Table, Player, Message2).
|
|
||||||
|
|
|
@ -1,4 +1,8 @@
|
||||||
-module(helpers).
|
-module(helpers).
|
||||||
|
-export([not_implemented/0, get_new_token/0]).
|
||||||
|
|
||||||
not_implemented() ->
|
not_implemented() ->
|
||||||
exit("Not implemented").
|
exit("Not implemented").
|
||||||
|
|
||||||
|
get_new_token() ->
|
||||||
|
string:strip(os:cmd("uuidgen"), right, $\n ).
|
||||||
|
|
|
@ -1,3 +1,3 @@
|
||||||
#!/usr/bin/env bash
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
erl -boot start_sasl -pa ebin_test -pa erlang_js/ebin/ -pa erlv8/ebin -pa ebin -pa src -eval 'ggs_protocol_test:test_parse().'
|
erl -boot start_sasl -pa ebin_test -pa erlang_js/ebin/ -pa ebin -pa src -eval 'ggs_coordinator_test:test().'
|
||||||
|
|
53
tests/ggs_coordinator_test.erl
Normal file
53
tests/ggs_coordinator_test.erl
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
-module(ggs_coordinator_test).
|
||||||
|
-include_lib("eunit/include/eunit.hrl").
|
||||||
|
|
||||||
|
coordinator_test_() ->
|
||||||
|
{foreach,
|
||||||
|
fun() ->
|
||||||
|
{ok, _Coord} = ggs_coordinator:start_link(),
|
||||||
|
timer:sleep(100)
|
||||||
|
end,
|
||||||
|
fun(_X) ->
|
||||||
|
ggs_coordinator:stop("End of test"),
|
||||||
|
timer:sleep(100)
|
||||||
|
end,
|
||||||
|
[
|
||||||
|
fun test_start_link/0,
|
||||||
|
fun test_stop/0,
|
||||||
|
fun test_join_bad_table/0,
|
||||||
|
fun test_join_lobby/0
|
||||||
|
]
|
||||||
|
}.
|
||||||
|
|
||||||
|
test_start_link() ->
|
||||||
|
% Check process info
|
||||||
|
PInfo = whereis(ggs_coordinator),
|
||||||
|
?assert((PInfo /= undefined)). % Did the server start?
|
||||||
|
|
||||||
|
test_stop() ->
|
||||||
|
ok = ggs_coordinator:stop(""), % Extra cleaning
|
||||||
|
timer:sleep(100),
|
||||||
|
% Did it stop?
|
||||||
|
?assert((whereis(ggs_coordinator)) == undefined).
|
||||||
|
|
||||||
|
test_join_bad_table() ->
|
||||||
|
Response = ggs_coordinator:join_table("Nonexistant table"),
|
||||||
|
?assert(Response == {error, no_such_table}).
|
||||||
|
|
||||||
|
test_join_lobby() ->
|
||||||
|
{Response, _} = ggs_coordinator:join_lobby(),
|
||||||
|
?assert(Response /= error).
|
||||||
|
|
||||||
|
%% 'Manual' tests
|
||||||
|
|
||||||
|
create_table_test() ->
|
||||||
|
{ok, _Coord} = ggs_coordinator:start_link(),
|
||||||
|
timer:sleep(100),
|
||||||
|
% Forcibly create a table. This functionality should be disabled
|
||||||
|
% in the production system, but is pretty nice for testing.
|
||||||
|
Response = ggs_coordinator:create_table({force, 1337}),
|
||||||
|
?assert(Response == {ok, 1337}).
|
||||||
|
|
||||||
|
join_good_table_test() ->
|
||||||
|
Response = ggs_coordinator:join_table(1337),
|
||||||
|
?assert(Response == {ok, 1337}).
|
|
@ -1,3 +1,4 @@
|
||||||
|
-module(ggs_player_test).
|
||||||
-include_lib("eunit/include/eunit.hrl").
|
-include_lib("eunit/include/eunit.hrl").
|
||||||
-import(ggs_player).
|
-import(ggs_player).
|
||||||
|
|
||||||
|
|
45
tests/ggs_table_test.erl
Normal file
45
tests/ggs_table_test.erl
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
-module(ggs_table_test).
|
||||||
|
-include_lib("eunit/include/eunit.hrl").
|
||||||
|
-import(ggs_table).
|
||||||
|
|
||||||
|
|
||||||
|
% @private
|
||||||
|
start_link_test() ->
|
||||||
|
Table = start_link(),
|
||||||
|
?assertNot(Table =:= undefined).
|
||||||
|
|
||||||
|
% @private
|
||||||
|
add_player_test() ->
|
||||||
|
Table = start_link(),
|
||||||
|
Player = test_player,
|
||||||
|
add_player(Table, Player),
|
||||||
|
{ok, [Player]} = gen_server:call(Table, get_player_list).
|
||||||
|
|
||||||
|
% @private
|
||||||
|
remove_player_test() ->
|
||||||
|
Table = start_link(),
|
||||||
|
Player = test_player,
|
||||||
|
Player2 = test_player2,
|
||||||
|
add_player(Table, Player),
|
||||||
|
{ok, [Player]} = gen_server:call(Table, get_player_list),
|
||||||
|
add_player(Table, Player2),
|
||||||
|
{ok, [Player2, Player]} = gen_server:call(Table, get_player_list),
|
||||||
|
remove_player(Table, Player),
|
||||||
|
{ok, [Player2]} = gen_server:call(Table, get_player_list),
|
||||||
|
remove_player(Table, Player2),
|
||||||
|
{ok, []} = gen_server:call(Table, get_player_list).
|
||||||
|
|
||||||
|
% @private
|
||||||
|
stop_test() ->
|
||||||
|
Table = start_link(),
|
||||||
|
ok = stop(Table).
|
||||||
|
|
||||||
|
% @private
|
||||||
|
notify_test() ->
|
||||||
|
Table = start_link(),
|
||||||
|
Player = test_player,
|
||||||
|
Message = {server, define, "function helloWorld(x) { }"},
|
||||||
|
ok = notify(Table, Player, Message).
|
||||||
|
Message2 = {game, "helloWorld", "test"},
|
||||||
|
ok = notify(Table, Player, Message2).
|
||||||
|
|
Reference in a new issue