Skip to content

Commit 0faae29

Browse files
committed
Merge remote-tracking branch 'contrib-sargun/master' into slf-merge-sargun-master
2 parents 042e8ea + 4c92409 commit 0faae29

7 files changed

+274
-47
lines changed

examples/riakc_pb_distributed.config

+42
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
{mode, max}.
2+
3+
{duration, 10}.
4+
{report_interval,1}.
5+
6+
{concurrent, 50}.
7+
8+
{driver, basho_bench_driver_riakc_pb}.
9+
10+
{key_generator, {int_to_bin_bigendian, {uniform_int, 10000}}}.
11+
12+
{value_generator, {fixed_bin, 10000}}.
13+
14+
{riakc_pb_ips, [{127,0,0,1}]}.
15+
16+
{riakc_pb_replies, 1}.
17+
18+
%%% {operations, [{get, 1}]}.
19+
{operations, [{get, 1}, {update, 1}]}.
20+
21+
%% Use {auto_reconnect, false} to get "old" behavior (prior to April 2013).
22+
%% See deps/riakc/src/riakc_pb_socket.erl for all valid socket options.
23+
{pb_connect_options, [{auto_reconnect, true}]}.
24+
25+
%% Overrides for the PB client's default 60 second timeout, on a
26+
%% per-type-of-operation basis. All timeout units are specified in
27+
%% milliseconds. The pb_timeout_general config item provides a
28+
%% default timeout if the read/write/listkeys/mapreduce timeout is not
29+
%% specified.
30+
31+
{pb_timeout_general, 30000}.
32+
{pb_timeout_read, 5000}.
33+
{pb_timeout_write, 5000}.
34+
{pb_timeout_listkeys, 50000}.
35+
%% The general timeout will be used because this specific item is commented:
36+
%% {pb_timeout_mapreduce, 50000}.
37+
38+
39+
%% Remote_nodes must be in the format of [{fqdn, nodename}]
40+
%% basho_bench / distributed Erlang use longnames
41+
{remote_nodes, [{'3c075477e55e-2.local', 'bb25'}]}.
42+
{distribute_work, true}.

rebar.config

+2-2
Original file line numberDiff line numberDiff line change
@@ -35,7 +35,7 @@
3535

3636
{escript_incl_apps, [lager, getopt, bear, folsom, ibrowse, riakc, riak_pb, mochiweb, protobuffs, velvet, goldrush]}.
3737

38-
{escript_emu_args, "%%! +K true\n"}.
38+
{escript_emu_args, "%%! +K true -rsh ssh\n"}.
3939
%% Use this for the Java client bench driver
4040
%% {escript_emu_args, "%%! +K true -name [email protected] -setcookie YOUR_ERLANG_COOKIE\n"}.
41-
{escript_emu_args, "%%! +K true -name [email protected] -setcookie YOUR_ERLANG_COOKIE\n"}.
41+
{escript_emu_args, "%%! +K true -name [email protected] -setcookie YOUR_ERLANG_COOKIE -rsh ssh\n"}.

src/basho_bench.erl

+72-4
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,6 @@
2222
-module(basho_bench).
2323

2424
-export([main/1, md5/1]).
25-
2625
-include("basho_bench.hrl").
2726

2827
%% ====================================================================
@@ -53,6 +52,8 @@ main(Args) ->
5352
{error, {already_loaded, basho_bench}} -> ok
5453
end,
5554
register(basho_bench, self()),
55+
%% TODO: Move into a proper supervision tree, janky for now
56+
{ok, _Pid} = basho_bench_config:start_link(),
5657
basho_bench_config:set(test_id, BenchName),
5758

5859
application:load(lager),
@@ -93,15 +94,16 @@ main(Args) ->
9394
%% Copy the config into the test dir for posterity
9495
[ begin {ok, _} = file:copy(Config, filename:join(TestDir, filename:basename(Config))) end
9596
|| Config <- Configs ],
96-
97+
case basho_bench_config:get(distribute_work, false) of
98+
true -> setup_distributed_work();
99+
false -> ok
100+
end,
97101
%% Set our CWD to the test dir
98102
ok = file:set_cwd(TestDir),
99-
100103
log_dimensions(),
101104

102105
%% Run pre_hook for user code preconditions
103106
run_pre_hook(),
104-
105107
%% Spin up the application
106108
ok = basho_bench_app:start(),
107109

@@ -139,6 +141,7 @@ maybe_net_node(Opts) ->
139141
case lists:keyfind(net_node, 1, Opts) of
140142
{_, Node} ->
141143
{_, Cookie} = lists:keyfind(net_cookie, 1, Opts),
144+
os:cmd("epmd -daemon"),
142145
net_kernel:start([Node, longnames]),
143146
erlang:set_cookie(Node, Cookie),
144147
ok;
@@ -256,6 +259,7 @@ load_source_files(Dir) ->
256259
case compile:file(F, [report, binary]) of
257260
{ok, Mod, Bin} ->
258261
{module, Mod} = code:load_binary(Mod, F, Bin),
262+
deploy_module(Mod),
259263
?INFO("Loaded ~p (~s)\n", [Mod, F]),
260264
ok;
261265
Error ->
@@ -276,6 +280,70 @@ run_hook({Module, Function}) ->
276280
run_hook(no_op) ->
277281
no_op.
278282

283+
get_addr_args() ->
284+
{ok, IfAddrs} = inet:getifaddrs(),
285+
FlattAttrib = lists:flatten([IfAttrib || {_Ifname, IfAttrib} <- IfAddrs]),
286+
Addrs = proplists:get_all_values(addr, FlattAttrib),
287+
% If inet:ntoa is unavailable, it probably means that you're running <R16
288+
StrAddrs = [inet:ntoa(Addr) || Addr <- Addrs],
289+
string:join(StrAddrs, " ").
290+
setup_distributed_work() ->
291+
case node() of
292+
'nonode@nohost' ->
293+
?STD_ERR("Basho bench not started in distributed mode, and distribute_work = true~n", []),
294+
halt(1);
295+
_ -> ok
296+
end,
297+
{ok, _Pid} = erl_boot_server:start([]),
298+
%% Allow anyone to boot from me...I might want to lock this down this down at some point
299+
erl_boot_server:add_subnet({0,0,0,0}, {0,0,0,0}),
300+
%% This is cheating, horribly, but it's the only simple way to bypass net_adm:host_file()
301+
gen_server:start({global, pool_master}, pool, [], []),
302+
RemoteSpec = basho_bench_config:get(remote_nodes, []),
303+
Cookie = lists:flatten(erlang:atom_to_list(erlang:get_cookie())),
304+
Args = "-setcookie " ++ Cookie ++ " -loader inet -hosts " ++ get_addr_args(),
305+
Slaves = [ slave:start_link(Host, Name, Args) || {Host, Name} <- RemoteSpec],
306+
SlaveNames = [SlaveName || {ok, SlaveName} <- Slaves],
307+
[pool:attach(SlaveName) || SlaveName <- SlaveNames],
308+
CodePaths = code:get_path(),
309+
rpc:multicall(SlaveNames, code, set_path, [CodePaths]),
310+
Apps = [lager, basho_bench, getopt, bear, folsom, ibrowse, riakc, riak_pb, mochiweb, protobuffs, velvet, goldrush],
311+
[distribute_app(App) || App <- Apps].
312+
313+
314+
deploy_module(Module) ->
315+
case basho_bench_config:get(distribute_work, false) of
316+
true ->
317+
Nodes = nodes(),
318+
{Module, Binary, Filename} = code:get_object_code(Module),
319+
rpc:multicall(Nodes, code, load_binary, [Module, Filename, Binary]);
320+
false -> ok
321+
end.
322+
323+
distribute_app(App) ->
324+
% :(. This is super hackish, it depends on a bunch of assumptions
325+
% But, unfortunately there are negative interactions with escript and slave nodes
326+
CodeExtension = code:objfile_extension(),
327+
LibDir = code:lib_dir(App),
328+
% Get what paths are in the code path that start with LibDir
329+
LibDirLen = string:len(LibDir),
330+
EbinsDir = lists:filter(fun(CodePathDir) -> string:substr(CodePathDir, 1, LibDirLen) == LibDir end, code:get_path()),
331+
StripEndFun = fun(Path) ->
332+
PathLen = string:len(Path),
333+
case string:substr(Path, PathLen - string:len(CodeExtension) + 1, string:len(Path)) of
334+
CodeExtension ->
335+
{true, string:substr(Path, 1, PathLen - string:len(CodeExtension))};
336+
_ -> false
337+
end
338+
end,
339+
EbinDirDistributeFun = fun(EbinDir) ->
340+
{ok, Beams} = erl_prim_loader:list_dir(EbinDir),
341+
Modules = lists:filtermap(StripEndFun, Beams),
342+
ModulesLoaded = [code:load_abs(filename:join(EbinDir, ModFileName)) || ModFileName <- Modules],
343+
lists:foreach(fun({module, Module}) -> deploy_module(Module) end, ModulesLoaded)
344+
end,
345+
lists:foreach(EbinDirDistributeFun, EbinsDir),
346+
ok.
279347
%% just a utility, should be in basho_bench_utils.erl
280348
%% but 's' is for multiple utilities, and so far this
281349
%% is the only one.

src/basho_bench_app.erl

+2
Original file line numberDiff line numberDiff line change
@@ -79,6 +79,8 @@ stop_or_kill() ->
7979
%%===================================================================
8080

8181
start(_StartType, _StartArgs) ->
82+
%% TODO: Move into a proper supervision tree, janky for now
83+
basho_bench_config:start_link(),
8284
{ok, Pid} = basho_bench_sup:start_link(),
8385
application:set_env(basho_bench_app, is_running, true),
8486
ok = basho_bench_stats:run(),

src/basho_bench_config.erl

+93-32
Original file line numberDiff line numberDiff line change
@@ -20,27 +20,63 @@
2020
%%
2121
%% -------------------------------------------------------------------
2222
-module(basho_bench_config).
23+
-behaviour(gen_server).
24+
25+
26+
-ifdef(TEST).
27+
-include_lib("eunit/include/eunit.hrl").
28+
-compile(export_all).
29+
-endif.
2330

2431
-export([load/1,
2532
normalize_ips/2,
2633
set/2,
2734
get/1, get/2]).
2835

36+
-export([start_link/0]).
37+
38+
% Gen server callbacks
39+
-export([code_change/3, init/1, terminate/2, handle_call/3, handle_cast/2, handle_info/2]).
40+
2941
-include("basho_bench.hrl").
3042

43+
-record(basho_bench_config_state, {}).
44+
45+
-type state() :: #basho_bench_config_state{}.
3146
%% ===================================================================
3247
%% Public API
3348
%% ===================================================================
3449

50+
%% Todo: ensure_started before calling on any gen_server APIs.
51+
ensure_started() ->
52+
start_link().
53+
54+
start_link() ->
55+
gen_server:start_link({global, ?MODULE}, ?MODULE, [], []).
56+
57+
3558
load(Files) ->
36-
TermsList =
37-
[ case file:consult(File) of
38-
{ok, Terms} ->
39-
Terms;
40-
{error, Reason} ->
41-
?FAIL_MSG("Failed to parse config file ~s: ~p\n", [File, Reason])
42-
end || File <- Files ],
43-
load_config(lists:append(TermsList)).
59+
ensure_started(),
60+
gen_server:call({global, ?MODULE}, {load_files, Files}).
61+
62+
set(Key, Value) ->
63+
gen_server:call({global, ?MODULE}, {set, Key, Value}).
64+
65+
get(Key) ->
66+
case gen_server:call({global, ?MODULE}, {get, Key}) of
67+
{ok, Value} ->
68+
Value;
69+
undefined ->
70+
erlang:error("Missing configuration key", [Key])
71+
end.
72+
73+
get(Key, Default) ->
74+
case gen_server:call({global, ?MODULE}, {get, Key}) of
75+
{ok, Value} ->
76+
Value;
77+
undefined ->
78+
Default
79+
end.
4480

4581
%% @doc Normalize the list of IPs and Ports.
4682
%%
@@ -58,42 +94,67 @@ normalize_ips(IPs, DefultPort) ->
5894
end,
5995
lists:foldl(F, [], IPs).
6096

61-
set(Key, Value) ->
62-
ok = application:set_env(basho_bench, Key, Value).
6397

64-
get(Key) ->
65-
case application:get_env(basho_bench, Key) of
66-
{ok, Value} ->
67-
Value;
68-
undefined ->
69-
erlang:error("Missing configuration key", [Key])
70-
end.
7198

72-
get(Key, Default) ->
73-
case application:get_env(basho_bench, Key) of
74-
{ok, Value} ->
75-
Value;
76-
_ ->
77-
Default
78-
end.
7999

80100

81101
%% ===================================================================
82102
%% Internal functions
83103
%% ===================================================================
84104

85-
load_config([]) ->
86-
ok;
87-
load_config([{Key, Value} | Rest]) ->
88-
?MODULE:set(Key, Value),
89-
load_config(Rest);
90-
load_config([ Other | Rest]) ->
91-
?WARN("Ignoring non-tuple config value: ~p\n", [Other]),
92-
load_config(Rest).
93105

94106
normalize_ip_entry({IP, Ports}, Normalized, _) when is_list(Ports) ->
95107
[{IP, Port} || Port <- Ports] ++ Normalized;
96108
normalize_ip_entry({IP, Port}, Normalized, _) ->
97109
[{IP, Port}|Normalized];
98110
normalize_ip_entry(IP, Normalized, DefaultPort) ->
99111
[{IP, DefaultPort}|Normalized].
112+
113+
114+
%% ===
115+
%% Gen_server Functions
116+
%% ===
117+
118+
-spec init(term()) -> {ok, state()}.
119+
init(_Args) ->
120+
State = #basho_bench_config_state{},
121+
{ok, State}.
122+
123+
-spec code_change(term(), state(), term()) -> {ok, state()}.
124+
code_change(_OldVsn, State, _Extra) ->
125+
{ok, State}.
126+
127+
-spec terminate(term(), state()) -> 'ok'.
128+
terminate(_Reason, _State) ->
129+
ok.
130+
131+
handle_call({load_files, FileNames}, _From, State) ->
132+
set_keys_from_files(FileNames),
133+
{reply, ok, State};
134+
135+
handle_call({set, Key, Value}, _From, State) ->
136+
application:set_env(basho_bench, Key, Value),
137+
{reply, ok, State};
138+
handle_call({get, Key}, _From, State) ->
139+
Value = application:get_env(basho_bench, Key),
140+
{reply, Value, State}.
141+
142+
handle_cast(_Cast, State) ->
143+
{noreply, State}.
144+
145+
handle_info(_Info, State) ->
146+
{noreply, State}.
147+
148+
set_keys_from_files(Files) ->
149+
KVs = [
150+
case file:consult(File) of
151+
{ok, Terms} ->
152+
Terms;
153+
{error, Reason} ->
154+
?FAIL_MSG("Failed to parse config file ~s: ~p\n", [File, Reason]),
155+
throw(invalid_config),
156+
notokay
157+
end || File <- Files ],
158+
FlatKVs = lists:flatten(KVs),
159+
[application:set_env(basho_bench, Key, Value) || {Key, Value} <- FlatKVs].
160+

0 commit comments

Comments
 (0)