machi/test/machi_flu_psup_test.erl

181 lines
7.7 KiB
Erlang
Raw Permalink Normal View History

%% -------------------------------------------------------------------
%%
%% Copyright (c) 2007-2015 Basho Technologies, Inc. All Rights Reserved.
%%
%% This file is provided to you under the Apache License,
%% Version 2.0 (the "License"); you may not use this file
%% except in compliance with the License. You may obtain
%% a copy of the License at
%%
%% http://www.apache.org/licenses/LICENSE-2.0
%%
%% Unless required by applicable law or agreed to in writing,
%% software distributed under the License is distributed on an
%% "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
%% KIND, either express or implied. See the License for the
%% specific language governing permissions and limitations
%% under the License.
%%
%% -------------------------------------------------------------------
-module(machi_flu_psup_test).
-ifdef(TEST).
2015-06-02 13:13:15 +00:00
-ifndef(PULSE).
-include_lib("eunit/include/eunit.hrl").
-include("machi_projection.hrl").
%% smoke_test2() will try repeatedly to make a TCP connection to ports
%% on localhost that have no listener.
%% If you use 'sysctl -w net.inet.icmp.icmplim=3' before running this
%% test, you'll get to exercise some timeout handling in
%% machi_chain_manager1:perhaps_call_t().
%% The default for net.inet.icmp.icmplim is 50.
smoke_test_() ->
{timeout, 5*60, fun() -> smoke_test2() end}.
smoke_test2() ->
Ps = [{a,#p_srvr{name=a, address="localhost", port=5550, props="./data.a"}},
{b,#p_srvr{name=b, address="localhost", port=5551, props="./data.b"}},
{c,#p_srvr{name=c, address="localhost", port=5552, props="./data.c"}}
],
[os:cmd("rm -rf " ++ P#p_srvr.props) || {_,P} <- Ps],
{ok, SupPid} = machi_sup:start_link(),
try
2015-05-02 07:59:28 +00:00
%% Only run a, don't run b & c so we have 100% failures talking to them
[begin
#p_srvr{name=Name, port=Port, props=Dir} = P,
{ok, _} = machi_flu_psup:start_flu_package(Name, Port, Dir, [])
end || {_,P} <- [hd(Ps)]],
2015-05-01 05:51:42 +00:00
[machi_chain_manager1:trigger_react_to_env(a_chmgr) || _ <-lists:seq(1,5)],
machi_chain_manager1:set_chain_members(a_chmgr, orddict:from_list(Ps)),
[machi_chain_manager1:trigger_react_to_env(a_chmgr) || _ <-lists:seq(1,5)],
2015-05-02 07:59:28 +00:00
ok
after
exit(SupPid, normal),
[os:cmd("rm -rf " ++ P#p_srvr.props) || {_,P} <- Ps],
machi_util:wait_for_death(SupPid, 100),
ok
end.
2015-05-08 06:36:53 +00:00
partial_stop_restart_test_() ->
{timeout, 5*60, fun() -> partial_stop_restart2() end}.
2015-05-02 07:59:28 +00:00
2015-05-08 06:36:53 +00:00
partial_stop_restart2() ->
Ps = [{a,#p_srvr{name=a, address="localhost", port=5560, props="./data.a"}},
{b,#p_srvr{name=b, address="localhost", port=5561, props="./data.b"}},
{c,#p_srvr{name=c, address="localhost", port=5562, props="./data.c"}}
2015-05-02 07:59:28 +00:00
],
2015-05-08 06:36:53 +00:00
ChMgrs = [machi_flu_psup:make_mgr_supname(P#p_srvr.name) || {_,P} <-Ps],
PStores = [machi_flu_psup:make_proj_supname(P#p_srvr.name) || {_,P} <-Ps],
Dict = orddict:from_list(Ps),
2015-05-02 07:59:28 +00:00
[os:cmd("rm -rf " ++ P#p_srvr.props) || {_,P} <- Ps],
{ok, SupPid} = machi_sup:start_link(),
2015-07-21 08:29:33 +00:00
DbgProps = [{initial_wedged, true}],
Start = fun({_,P}) ->
#p_srvr{name=Name, port=Port, props=Dir} = P,
{ok, _} = machi_flu_psup:start_flu_package(
Name, Port, Dir, [{active_mode,false}|DbgProps])
end,
WedgeStatus = fun({_,#p_srvr{address=Addr, port=TcpPort}}) ->
machi_flu1_client:wedge_status(Addr, TcpPort)
end,
Append = fun({_,#p_srvr{address=Addr, port=TcpPort}}, EpochID) ->
machi_flu1_client:append_chunk(Addr, TcpPort,
EpochID,
<<"prefix">>, <<"data">>)
end,
2015-05-02 07:59:28 +00:00
try
2015-05-08 06:36:53 +00:00
[Start(P) || P <- Ps],
[{ok, {true, _}} = WedgeStatus(P) || P <- Ps], % all are wedged
[{error,wedged} = Append(P, ?DUMMY_PV1_EPOCH) || P <- Ps], % all are wedged
2015-05-02 07:59:28 +00:00
[machi_chain_manager1:set_chain_members(ChMgr, Dict) ||
ChMgr <- ChMgrs ],
{ok, {false, EpochID1}} = WedgeStatus(hd(Ps)),
[{ok, {false, EpochID1}} = WedgeStatus(P) || P <- Ps], % *not* wedged
[{ok,_} = Append(P, EpochID1) || P <- Ps], % *not* wedged
{ok, {_,_,File1}} = Append(hd(Ps), EpochID1),
2015-05-02 07:59:28 +00:00
{_,_,_} = machi_chain_manager1:trigger_react_to_env(hd(ChMgrs)),
2015-05-02 07:59:28 +00:00
[begin
_QQa = machi_chain_manager1:trigger_react_to_env(ChMgr)
end || _ <- lists:seq(1,125), ChMgr <- ChMgrs],
2015-05-02 07:59:28 +00:00
2015-05-08 06:36:53 +00:00
%% All chain managers & projection stores should be using the
2015-05-02 07:59:28 +00:00
%% same projection which is max projection in each store.
{no_change,_,Epoch_m} = machi_chain_manager1:trigger_react_to_env(
2015-05-08 06:36:53 +00:00
hd(ChMgrs)),
[{no_change,_,Epoch_m} = machi_chain_manager1:trigger_react_to_env(
2015-05-02 07:59:28 +00:00
ChMgr )|| ChMgr <- ChMgrs],
2015-05-08 06:36:53 +00:00
{ok, Proj_m} = machi_projection_store:read_latest_projection(
2015-05-02 07:59:28 +00:00
hd(PStores), public),
2015-04-30 14:16:08 +00:00
[begin
{ok, Proj_m2} = machi_projection_store:read_latest_projection(
PStore, ProjType),
true = (machi_projection:update_dbg2(Proj_m, []) ==
machi_projection:update_dbg2(Proj_m2, []))
2015-05-02 07:59:28 +00:00
end || ProjType <- [public, private], PStore <- PStores ],
2015-05-08 06:36:53 +00:00
Epoch_m = Proj_m#projection_v1.epoch_number,
%% Confirm that all FLUs are *not* wedged, with correct proj & epoch
Proj_mCSum = Proj_m#projection_v1.epoch_csum,
[{ok, {false, {Epoch_m, Proj_mCSum}}} = WedgeStatus(P) || % *not* wedged
P <- Ps],
{ok, {false, EpochID1}} = WedgeStatus(hd(Ps)),
[{ok,_} = Append(P, EpochID1) || P <- Ps], % *not* wedged
2015-05-08 06:36:53 +00:00
%% Stop all but 'a'.
[ok = machi_flu_psup:stop_flu_package(Name) || {Name,_} <- tl(Ps)],
%% Stop and restart a.
{FluName_a, _} = hd(Ps),
ok = machi_flu_psup:stop_flu_package(FluName_a),
{ok, _} = Start(hd(Ps)),
timer:sleep(123), % TODO fix server socket available race condition in better way
2015-05-08 06:36:53 +00:00
%% Remember: 'a' is not in active mode.
{ok, Proj_m3} = machi_projection_store:read_latest_projection(
hd(PStores), private),
true = (machi_projection:update_dbg2(Proj_m, []) ==
machi_projection:update_dbg2(Proj_m3, [])),
2015-05-08 10:07:57 +00:00
%% Confirm that 'a' is wedged
{error, wedged} = Append(hd(Ps), EpochID1),
2015-05-08 10:50:47 +00:00
{_, #p_srvr{address=Addr_a, port=TcpPort_a}} = hd(Ps),
{error, wedged} = machi_flu1_client:read_chunk(
Addr_a, TcpPort_a, ?DUMMY_PV1_EPOCH,
<<>>, 99999999, 1, []),
2015-05-08 10:50:47 +00:00
{error, wedged} = machi_flu1_client:checksum_list(
Addr_a, TcpPort_a, ?DUMMY_PV1_EPOCH, <<>>),
%% list_files() is permitted despite wedged status
{ok, _} = machi_flu1_client:list_files(
2015-05-08 10:50:47 +00:00
Addr_a, TcpPort_a, ?DUMMY_PV1_EPOCH),
2015-05-08 10:07:57 +00:00
%% Iterate through humming consensus once
{now_using,_,Epoch_n} = machi_chain_manager1:trigger_react_to_env(
2015-05-08 06:36:53 +00:00
hd(ChMgrs)),
true = (Epoch_n > Epoch_m),
{ok, {false, EpochID3}} = WedgeStatus(hd(Ps)),
%% The file we're assigned should be different with the epoch change.
{ok, {_,_,File3}} = Append(hd(Ps), EpochID3),
true = (File1 /= File3),
2015-05-08 10:07:57 +00:00
%% Confirm that 'a' is *not* wedged
{ok, _} = Append(hd(Ps), EpochID3),
2015-05-08 06:36:53 +00:00
ok
after
2015-05-01 05:51:42 +00:00
exit(SupPid, normal),
[os:cmd("rm -rf " ++ P#p_srvr.props) || {_,P} <- Ps],
2015-05-01 05:51:42 +00:00
machi_util:wait_for_death(SupPid, 100),
ok
end.
2015-06-02 13:13:15 +00:00
-endif. % !PULSE
-endif. % TEST