%% %% %CopyrightBegin% %% %% Copyright Ericsson AB 1998-2017. All Rights Reserved. %% %% Licensed 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. %% %% %CopyrightEnd% %% %% %%%---------------------------------------------------------------------- %%% File : mnesia_evil_backup.erl %%% Author : Dan Gudmundsson <dgud@legolas> %%% Purpose : Evil backup tests %%% Created : 3 Jun 1998 by Dan Gudmundsson <dgud@erix.ericsson.se> %%%---------------------------------------------------------------------- -module(mnesia_evil_backup). -author('dgud@erix.ericsson.se'). -compile(export_all). -include("mnesia_test_lib.hrl"). %%-export([Function/Arity, ...]). init_per_testcase(Func, Conf) -> mnesia_test_lib:init_per_testcase(Func, Conf). end_per_testcase(Func, Conf) -> mnesia_test_lib:end_per_testcase(Func, Conf). %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% all() -> [backup, bad_backup, global_backup_checkpoint, {group, restore_tables}, traverse_backup, selective_backup_checkpoint, incremental_backup_checkpoint, install_fallback, uninstall_fallback, local_fallback, sops_with_checkpoint]. groups() -> [{restore_tables, [], [restore_errors, restore_clear, restore_keep, restore_recreate, restore_clear_ram]}]. init_per_group(_GroupName, Config) -> Config. end_per_group(_GroupName, Config) -> Config. backup(doc) -> ["Checking the interface to the function backup", "We don't check that the backups can be used here", "That is checked in install_fallback and in restore"]; backup(suite) -> []; backup(Config) when is_list(Config) -> [Node1, Node2] = _Nodes = ?acquire_nodes(2, Config), Tab = backup_tab, Def = [{disc_copies, [Node1]}, {ram_copies, [Node2]}], ?match({atomic, ok}, mnesia:create_table(Tab, Def)), ?match(ok, mnesia:dirty_write({Tab, 1, test_ok})), File = "backup_test.BUP", ?match(ok, mnesia:backup(File)), File2 = "backup_test2.BUP", Tab2 = backup_tab2, Def2 = [{disc_only_copies, [Node2]}], ?match({atomic, ok}, mnesia:create_table(Tab2, Def2)), ?match(ok, mnesia:backup(File2, mnesia_backup)), File3 = "backup_test3.BUP", mnesia_test_lib:kill_mnesia([Node2]), ?match({error, _}, mnesia:backup(File3, mnesia_backup)), ?match(ok, file:delete(File)), ?match(ok, file:delete(File2)), ?match({error, _}, file:delete(File3)), ?verify_mnesia([Node1], [Node2]). bad_backup(suite) -> []; bad_backup(Config) when is_list(Config) -> [Node1] = ?acquire_nodes(1, Config), Tab = backup_tab, Def = [{disc_copies, [Node1]}], ?match({atomic, ok}, mnesia:create_table(Tab, Def)), ?match(ok, mnesia:dirty_write({Tab, 1, test_ok})), File = "backup_test.BUP", ?match(ok, mnesia:backup(File)), file:write_file(File, "trash", [append]), ?match(ok, mnesia:dirty_write({Tab, 1, test_bad})), ?match({atomic,[Tab]}, mnesia:restore(File, [{clear_tables, [Tab]}])), ?match([{Tab,1,test_ok}], mnesia:dirty_read(Tab, 1)), ?match(ok, file:delete(File)), ?verify_mnesia([Node1], []). global_backup_checkpoint(doc) -> ["Checking the interface to the function backup_checkpoint", "We don't check that the backups can be used here", "That is checked in install_fallback and in restore"]; global_backup_checkpoint(suite) -> []; global_backup_checkpoint(Config) when is_list(Config) -> [Node1, Node2] = Nodes = ?acquire_nodes(2, Config), Tab = backup_cp, Def = [{disc_copies, [Node1]}, {ram_copies, [Node2]}], File = "backup_checkpoint.BUP", File2 = "backup_checkpoint2.BUP", ?match({atomic, ok}, mnesia:create_table(Tab, Def)), ?match(ok, mnesia:dirty_write({Tab, 1, test_ok})), ?match({error, _}, mnesia:backup_checkpoint(cp_name, File)), Spec = [{name, cp_name}, {max, mnesia:system_info(tables)}], ?match({ok, _Name, _Ns}, mnesia:activate_checkpoint(Spec)), ?match(ok, mnesia:backup_checkpoint(cp_name, File)), ?match({error, _}, mnesia:backup_checkpoint(cp_name_nonexist, File)), ?match(ok, mnesia:backup_checkpoint(cp_name, File2, mnesia_backup)), ?match({error, _}, file:delete(File)), ?match(ok, file:delete(File2)), ?verify_mnesia(Nodes, []). restore_errors(suite) -> []; restore_errors(Config) when is_list(Config) -> [_Node] = ?acquire_nodes(1, Config), ?match({aborted, enoent}, mnesia:restore(notAfile, [])), ?match({aborted, {badarg, _}}, mnesia:restore(notAfile, not_a_list)), ?match({aborted, {badarg, _}}, mnesia:restore(notAfile, [test_badarg])), ?match({aborted, {badarg, _}}, mnesia:restore(notAfile, [{test_badarg, xxx}])), ?match({aborted, {badarg, _}}, mnesia:restore(notAfile, [{skip_tables, xxx}])), ?match({aborted, {badarg, _}}, mnesia:restore(notAfile, [{recreate_tables, [schema]}])), ?match({aborted, {badarg, _}}, mnesia:restore(notAfile, [{default_op, asdklasd}])), MnesiaDir = mnesia_lib:dir(), ?match({aborted, {not_a_log_file, _}}, mnesia:restore(filename:join(MnesiaDir, "schema.DAT"), [])), ?match({aborted, _}, mnesia:restore(filename:join(MnesiaDir, "LATEST.LOG"), [])), ok. restore_clear(suite) -> []; restore_clear(Config) when is_list(Config) -> restore(Config, clear_tables). restore_keep(suite) -> []; restore_keep(Config) when is_list(Config) -> restore(Config, keep_tables). restore_recreate(suite) -> []; restore_recreate(Config) when is_list(Config) -> restore(Config, recreate_tables). check_tab(Records, Line) -> Verify = fun({Table, Key, Val}) -> case catch mnesia:dirty_read({Table, Key}) of [{Table, Key, Val}] -> ok; Else -> mnesia_test_lib:error("Not matching on Node ~p ~n" " Expected ~p~n Actual ~p~n", [node(), {Table, Key, Val}, Else], ?MODULE, Line), exit(error) end; (Recs) -> [{Tab, Key, _}, _] = Recs, SRecs = lists:sort(Recs), R_Recs = lists:sort(catch mnesia:dirty_read({Tab, Key})), case R_Recs of SRecs -> ok; Else -> mnesia_test_lib:error("Not matching on Node ~p ~n" " Expected ~p~n Actual ~p~n", [node(), SRecs, Else], ?MODULE, Line), exit(error) end end, lists:foreach(Verify, Records). restore(Config, Op) -> [Node1, Node2, _Node3] = Nodes = ?acquire_nodes(3, Config), Tab1 = ram_snmp, Def1 = [{snmp, [{key, integer}]}, {ram_copies, [Node1]}], Tab2 = disc_index, Def2 = [{index, [val]}, {disc_copies, [Node1, Node2]}], Tab3 = dionly_bag, Def3 = [{type, bag}, {disc_only_copies, Nodes}], ?match({atomic, ok}, mnesia:create_table(Tab1, Def1)), ?match({atomic, ok}, mnesia:create_table(Tab2, Def2)), ?match({atomic, ok}, mnesia:create_table(Tab3, Def3)), File1 = "restore1.BUP", File2 = "restore2.BUP", Restore = fun(O, A) -> case mnesia:restore(O, A) of {atomic, Tabs} when is_list(Tabs) -> {atomic, lists:sort(Tabs)}; Other -> Other end end, Tabs = lists:sort([Tab1, Tab2, Tab3]), [mnesia:dirty_write({Tab1, N, N+42}) || N <- lists:seq(1, 10)], [mnesia:dirty_write({Tab2, N, N+43}) || N <- lists:seq(1, 10)], [mnesia:dirty_write({Tab3, N, N+44}) || N <- lists:seq(1, 10)], Res1 = [{Tab1, N, N+42} || N <- lists:seq(1, 10)], Res2 = [{Tab2, N, N+43} || N <- lists:seq(1, 10)], Res3 = [{Tab3, N, N+44} || N <- lists:seq(1, 10)], {ok, Name, _} = mnesia:activate_checkpoint([{min, Tabs}, {ram_overrides_dump, true}]), file:delete(File1), %% Test standard Restore on one table on one node ?match(ok, mnesia:backup_checkpoint(Name, File1)), ?match(ok, mnesia:deactivate_checkpoint(Name)), ?match(ok, mnesia:backup(File2)), [mnesia:dirty_write({Tab1, N, N+1}) || N <- lists:seq(1, 11)], [mnesia:dirty_write({Tab2, N, N+1}) || N <- lists:seq(1, 11)], [mnesia:dirty_write({Tab3, N, N+1}) || N <- lists:seq(1, 11)], Res21 = [{Tab2, N, N+1} || N <- lists:seq(1, 11)], Res31 = [[{Tab3, N, N+1}, {Tab3, N, N+44}] || N <- lists:seq(1, 10)], Check = fun() -> [disk_log:pid2name(X) || X <- processes(), Data <- [process_info(X, [current_function])], Data =/= undefined, element(1, element(2, lists:keyfind(current_function, 1, Data)))=:= disk_log] end, Before = Check(), ?match({atomic, [Tab1]}, Restore(File1, [{Op, [Tab1]}, {skip_tables, Tabs -- [Tab1]}])), case Op of keep_tables -> ?match([{Tab1, 11, 12}], mnesia:dirty_read({Tab1, 11})); clear_tables -> ?match([], mnesia:dirty_read({Tab1, 11})); recreate_tables -> ?match([], mnesia:dirty_read({Tab1, 11})) end, [rpc:call(Node, ?MODULE, check_tab, [Res1, ?LINE]) || Node <- Nodes], [rpc:call(Node, ?MODULE, check_tab, [Res21, ?LINE]) || Node <- Nodes], [rpc:call(Node, ?MODULE, check_tab, [Res31, ?LINE]) || Node <- Nodes], %% Restore all tables on it's nodes mnesia:clear_table(Tab1), mnesia:clear_table(Tab2), mnesia:clear_table(Tab3), [mnesia:dirty_write({Tab1, N, N+1}) || N <- lists:seq(1, 11)], [mnesia:dirty_write({Tab2, N, N+1}) || N <- lists:seq(1, 11)], [mnesia:dirty_write({Tab3, N, N+1}) || N <- lists:seq(1, 11)], ?match({atomic, ok}, mnesia:del_table_copy(Tab2, Node1)), ?match({ok, Node1}, mnesia:subscribe({table, Tab1})), ?match({atomic, Tabs}, Restore(File1, [{default_op, Op}, {module, mnesia_backup}])), case Op of clear_tables -> ?match_receive({mnesia_table_event, {delete, {schema, Tab1}, _}}), ?match_receive({mnesia_table_event, {write, {schema, Tab1, _}, _}}), check_subscr(Tab1), [rpc:call(Node, ?MODULE, check_tab, [Res1, ?LINE]) || Node <- Nodes], [rpc:call(Node, ?MODULE, check_tab, [Res2, ?LINE]) || Node <- Nodes], [rpc:call(Node, ?MODULE, check_tab, [Res3, ?LINE]) || Node <- Nodes], ?match([], mnesia:dirty_read({Tab1, 11})), ?match([], mnesia:dirty_read({Tab2, 11})), ?match([], mnesia:dirty_read({Tab3, 11})), %% Check Index ?match([{Tab2, 10, 53}], mnesia:dirty_index_read(Tab2, 53, val)), ?match([], mnesia:dirty_index_read(Tab2, 11, val)), %% Check Snmp ?match({ok, [1]}, mnesia:snmp_get_next_index(Tab1,[])), ?match({ok, {Tab1, 1, 43}}, mnesia:snmp_get_row(Tab1, [1])), ?match(undefined, mnesia:snmp_get_row(Tab1, [11])), %% Check schema info ?match([Node2], mnesia:table_info(Tab2, where_to_write)); keep_tables -> check_subscr(Tab1), [rpc:call(Node, ?MODULE, check_tab, [Res1, ?LINE]) || Node <- Nodes], [rpc:call(Node, ?MODULE, check_tab, [Res2, ?LINE]) || Node <- Nodes], [rpc:call(Node, ?MODULE, check_tab, [Res31, ?LINE]) || Node <- Nodes], ?match([{Tab1, 11, 12}], mnesia:dirty_read({Tab1, 11})), ?match([{Tab2, 11, 12}], mnesia:dirty_read({Tab2, 11})), ?match([{Tab3, 11, 12}], mnesia:dirty_read({Tab3, 11})), ?match([{Tab2, 10, 53}], mnesia:dirty_index_read(Tab2, 53, val)), %% Check Index ?match([], mnesia:dirty_index_read(Tab2, 11, val)), ?match({ok, [1]}, mnesia:snmp_get_next_index(Tab1,[])), %% Check Snmp ?match({ok, {Tab1, 1, 43}}, mnesia:snmp_get_row(Tab1, [1])), ?match({ok, {Tab1, 11, 12}}, mnesia:snmp_get_row(Tab1, [11])), %% Check schema info ?match([Node2], mnesia:table_info(Tab2, where_to_write)); recreate_tables -> check_subscr(Tab1, 0), [rpc:call(Node, ?MODULE, check_tab, [Res1, ?LINE]) || Node <- Nodes], [rpc:call(Node, ?MODULE, check_tab, [Res2, ?LINE]) || Node <- Nodes], [rpc:call(Node, ?MODULE, check_tab, [Res3, ?LINE]) || Node <- Nodes], ?match([], mnesia:dirty_read({Tab1, 11})), ?match([], mnesia:dirty_read({Tab2, 11})), ?match([], mnesia:dirty_read({Tab3, 11})), %% Check Index ?match([{Tab2, 10, 53}], mnesia:dirty_index_read(Tab2, 53, val)), ?match([], mnesia:dirty_index_read(Tab2, 11, val)), %% Check Snmp ?match({ok, [1]}, mnesia:snmp_get_next_index(Tab1,[])), ?match({ok, {Tab1, 1, 43}}, mnesia:snmp_get_row(Tab1, [1])), ?match(undefined, mnesia:snmp_get_row(Tab1, [11])), %% Check schema info Ns = lists:sort([Node1, Node2]), ?match(Ns, lists:sort(mnesia:table_info(Tab2, where_to_write))) end, ?match(ok, file:delete(File1)), ?match(ok, file:delete(File2)), ?match([], Check() -- (Before ++ [{ok, latest_log}, {ok, previous_log}])), ?verify_mnesia(Nodes, []). check_subscr(Tab) -> check_subscr(Tab, 10). check_subscr(_Tab, 0) -> receive Msg -> ?error("Too many msgs ~p~n", [Msg]) after 500 -> ok end; check_subscr(Tab, N) -> V = N +42, receive {mnesia_table_event, {write, {Tab, N, V}, _}} -> check_subscr(Tab, N-1) after 500 -> ?error("Missing ~p~n", [{Tab, N, V}]) end. restore_clear_ram(suite) -> []; restore_clear_ram(Config) when is_list(Config) -> Nodes = ?acquire_nodes(3, [{diskless, true}|Config]), ?match({atomic, ok}, mnesia:create_table(a, [{ram_copies, Nodes}])), Write = fun(What) -> mnesia:write({a,1,What}), mnesia:write({a,2,What}), mnesia:write({a,3,What}) end, Bup = "restore_clear_ram.BUP", ?match({atomic, ok}, mnesia:transaction(Write, [initial])), ?match({ok, _, _}, mnesia:activate_checkpoint([{name,test}, {min, [schema, a]}, {ram_overrides_dump, true}])), ?match(ok, mnesia:backup_checkpoint(test, Bup)), ?match({atomic, ok}, mnesia:transaction(Write, [data])), ?match({atomic, [a]}, mnesia:restore(Bup, [{clear_tables,[a]},{default_op,skip_tables}])), restore_clear_ram_loop(100, Nodes, Bup), ok. restore_clear_ram_loop(N, Nodes = [N1,N2,N3], Bup) when N > 0 -> ?match([], mnesia_test_lib:stop_mnesia(Nodes)), ?match({_, []}, rpc:multicall([N1,N2], mnesia, start, [[{extra_db_nodes, Nodes}]])), Key = rpc:async_call(N3, mnesia, start, [[{extra_db_nodes, Nodes}]]), ?match({atomic, ok}, mnesia:create_table(a, [{ram_copies, Nodes}])), ?match({atomic, [a]}, mnesia:restore(Bup, [{clear_tables,[a]},{default_op,skip_tables}])), ?match(ok, rpc:yield(Key)), ?match(ok, rpc:call(N3, mnesia, wait_for_tables, [[a], 3000])), case rpc:multicall(Nodes, mnesia, table_info, [a,size]) of {[3,3,3], []} -> restore_clear_ram_loop(N-1, Nodes, Bup); Error -> ?match(3, Error) end; restore_clear_ram_loop(_,_,_) -> ok. traverse_backup(doc) -> ["Testing the traverse_backup interface, the resulting file is not tested though", "See install_fallback for result using the output file from traverse_backup", "A side effect is that the backup file contents are tested"]; traverse_backup(suite) -> []; traverse_backup(Config) when is_list(Config) -> [Node1, Node2] = Nodes = ?acquire_nodes(2, Config), Tab = backup_tab, Def = [{disc_copies, [Node1]}, {ram_copies, [Node2]}], ?match({atomic, ok}, mnesia:create_table(Tab, Def)), ?match(ok, mnesia:dirty_write({Tab, 1, test_nok})), ?match(ok, mnesia:dirty_write({Tab, 2, test_nok})), ?match(ok, mnesia:dirty_write({Tab, 3, test_nok})), ?match(ok, mnesia:dirty_write({Tab, 4, test_nok})), ?match(ok, mnesia:dirty_write({Tab, 5, test_nok})), File = "_treverse_backup.BUP", File2 = "traverse_backup2.BUP", File3 = "traverse_backup3.BUP", ?match(ok, mnesia:backup(File)), Fun = fun({backup_tab, N, _}, Acc) -> {[{backup_tab, N, test_ok}], Acc+1}; (Other, Acc) -> {[Other], Acc} end, ?match({ok, 5}, mnesia:traverse_backup(File, read_only, Fun, 0)), ?match(ok, file:delete(read_only)), ?match({ok, 5}, mnesia:traverse_backup(File, mnesia_backup, dummy, read_only, Fun, 0)), ?match({ok, 5}, mnesia:traverse_backup(File, File2, Fun, 0)), ?match({ok, 5}, mnesia:traverse_backup(File2, mnesia_backup, File3, mnesia_backup, Fun, 0)), BadFun = fun({bad_tab, _N, asd}, Acc) -> {{error, error}, Acc} end, ?match({error, _}, mnesia:traverse_backup(File, read_only, BadFun, 0)), ?match({error, _}, file:delete(read_only)), ?match(ok, file:delete(File)), ?match(ok, file:delete(File2)), ?match(ok, file:delete(File3)), ?verify_mnesia(Nodes, []). install_fallback(doc) -> ["This tests the install_fallback intf.", "It also verifies that the output from backup_checkpoint and traverse_backup", "is valid"]; install_fallback(suite) -> []; install_fallback(Config) when is_list(Config) -> [Node1, Node2] = Nodes = ?acquire_nodes(2, Config), Tab = fallbacks_test, Def = [{disc_copies, [Node1]}, {ram_copies, [Node2]}], ?match({atomic, ok}, mnesia:create_table(Tab, Def)), ?match(ok, mnesia:dirty_write({Tab, 1, test_nok})), ?match(ok, mnesia:dirty_write({Tab, 2, test_nok})), ?match(ok, mnesia:dirty_write({Tab, 3, test_nok})), ?match(ok, mnesia:dirty_write({Tab, 4, test_nok})), ?match(ok, mnesia:dirty_write({Tab, 5, test_nok})), Tab2 = fallbacks_test2, Def2 = [{disc_copies, [node()]}], ?match({atomic, ok}, mnesia:create_table(Tab2, Def2)), Tab3 = fallbacks_test3, ?match({atomic, ok}, mnesia:create_table(Tab3, Def2)), Fun2 = fun(Key) -> Rec = {Tab2, Key, test_ok}, mnesia:dirty_write(Rec), [Rec] end, TabSize3 = 1000, OldRecs2 = [Fun2(K) || K <- lists:seq(1, TabSize3)], Spec =[{name, cp_name}, {max, mnesia:system_info(tables)}], ?match({ok, _Name, Nodes}, mnesia:activate_checkpoint(Spec)), ?match(ok, mnesia:dirty_write({Tab, 6, test_nok})), [mnesia:dirty_write({Tab2, K, test_nok}) || K <- lists:seq(1, TabSize3 + 10)], File = "install_fallback.BUP", File2 = "install_fallback2.BUP", File3 = "install_fallback3.BUP", ?match(ok, mnesia:backup_checkpoint(cp_name, File)), Fun = fun({T, N, _}, Acc) when T == Tab -> case N rem 2 of 0 -> io:format("write ~p -> ~p~n", [N, T]), {[{T, N, test_ok}], Acc + 1}; 1 -> io:format("write ~p -> ~p~n", [N, Tab3]), {[{Tab3, N, test_ok}], Acc + 1} end; ({T, N}, Acc) when T == Tab -> case N rem 2 of 0 -> io:format("delete ~p -> ~p~n", [N, T]), {[{T, N}], Acc + 1}; 1 -> io:format("delete ~p -> ~p~n", [N, Tab3]), {[{Tab3, N}], Acc + 1} end; (Other, Acc) -> {[Other], Acc} end, ?match({ok, _}, mnesia:traverse_backup(File, File2, Fun, 0)), ?match(ok, mnesia:install_fallback(File2)), mnesia_test_lib:kill_mnesia([Node1, Node2]), timer:sleep(timer:seconds(1)), % Let it die! ok = mnesia:start([{ignore_fallback_at_startup, true}]), ok = mnesia:wait_for_tables([Tab, Tab2, Tab3], 10000), ?match([{Tab, 6, test_nok}], mnesia:dirty_read({Tab, 6})), mnesia_test_lib:kill_mnesia([Node1]), application:set_env(mnesia, ignore_fallback_at_startup, false), timer:sleep(timer:seconds(1)), % Let it die! ?match([], mnesia_test_lib:start_mnesia([Node1, Node2], [Tab, Tab2, Tab3])), % Verify ?match([], mnesia:dirty_read({Tab, 1})), ?match([{Tab3, 1, test_ok}], mnesia:dirty_read({Tab3, 1})), ?match([{Tab, 2, test_ok}], mnesia:dirty_read({Tab, 2})), ?match([], mnesia:dirty_read({Tab3, 2})), ?match([], mnesia:dirty_read({Tab, 3})), ?match([{Tab3, 3, test_ok}], mnesia:dirty_read({Tab3, 3})), ?match([{Tab, 4, test_ok}], mnesia:dirty_read({Tab, 4})), ?match([], mnesia:dirty_read({Tab3, 4})), ?match([], mnesia:dirty_read({Tab, 5})), ?match([{Tab3, 5, test_ok}], mnesia:dirty_read({Tab3, 5})), ?match([], mnesia:dirty_read({Tab, 6})), ?match([], mnesia:dirty_read({Tab3, 6})), ?match([], [mnesia:dirty_read({Tab2, K}) || K <- lists:seq(1, TabSize3)] -- OldRecs2), ?match(TabSize3, mnesia:table_info(Tab2, size)), % Check the interface file:delete(File3), ?match({error, _}, mnesia:install_fallback(File3)), ?match({error, _}, mnesia:install_fallback(File2, mnesia_badmod)), ?match({error, _}, mnesia:install_fallback(File2, {foo, foo})), ?match({error, _}, mnesia:install_fallback(File2, [{foo, foo}])), ?match({error, {badarg, {skip_tables, _}}}, mnesia:install_fallback(File2, [{default_op, skip_tables}, {default_op, keep_tables}, {keep_tables, [Tab, Tab2, Tab3]}, {skip_tables, [foo,{asd}]}])), ?match(ok, mnesia:install_fallback(File2, mnesia_backup)), ?match(ok, file:delete(File)), ?match(ok, file:delete(File2)), ?match({error, _}, file:delete(File3)), ?verify_mnesia(Nodes, []). uninstall_fallback(suite) -> []; uninstall_fallback(Config) when is_list(Config) -> [Node1, Node2] = Nodes = ?acquire_nodes(2, Config), Tab = uinst_fallbacks_test, File = "uinst_fallback.BUP", File2 = "uinst_fallback2.BUP", Def = [{disc_copies, [Node1]}, {ram_copies, [Node2]}], ?match({atomic, ok}, mnesia:create_table(Tab, Def)), ?match(ok, mnesia:dirty_write({Tab, 1, test_ok})), ?match(ok, mnesia:backup(File)), Fun = fun({T, N, _}, Acc) when T == Tab -> {[{T, N, test_nok}], Acc+1}; (Other, Acc) -> {[Other], Acc} end, ?match({ok, _}, mnesia:traverse_backup(File, File2, Fun, 0)), ?match({error, enoent}, mnesia:uninstall_fallback()), ?match(ok, mnesia:install_fallback(File2)), ?match(ok, file:delete(File)), ?match(ok, file:delete(File2)), ?match({error, _}, mnesia:uninstall_fallback([foobar])), ?match(ok, mnesia:uninstall_fallback()), mnesia_test_lib:kill_mnesia([Node1, Node2]), timer:sleep(timer:seconds(1)), % Let it die! ?match([], mnesia_test_lib:start_mnesia([Node1, Node2], [Tab])), ?match([{Tab, 1, test_ok}], mnesia:dirty_read({Tab, 1})), ?verify_mnesia(Nodes, []). local_fallback(suite) -> []; local_fallback(Config) when is_list(Config) -> [Node1, Node2] = Nodes = ?acquire_nodes(2, Config), Tab = local_fallback, File = "local_fallback.BUP", Def = [{disc_copies, Nodes}], Key = foo, Pre = {Tab, Key, pre}, Post = {Tab, Key, post}, ?match({atomic, ok}, mnesia:create_table(Tab, Def)), ?match(ok, mnesia:dirty_write(Pre)), ?match(ok, mnesia:backup(File)), ?match(ok, mnesia:dirty_write(Post)), Local = [{scope, local}], ?match({error, enoent}, mnesia:uninstall_fallback(Local)), ?match(ok, mnesia:install_fallback(File, Local)), ?match(true, mnesia:system_info(fallback_activated)), ?match(ok, mnesia:uninstall_fallback(Local)), ?match(false, mnesia:system_info(fallback_activated)), ?match(ok, mnesia:install_fallback(File, Local)), ?match(true, mnesia:system_info(fallback_activated)), ?match(false, rpc:call(Node2, mnesia, system_info , [fallback_activated])), ?match(ok, rpc:call(Node2, mnesia, install_fallback , [File, Local])), ?match([Post], mnesia:dirty_read({Tab, Key})), ?match([Post], rpc:call(Node2, mnesia, dirty_read, [{Tab, Key}])), ?match([], mnesia_test_lib:kill_mnesia(Nodes)), ?match([], mnesia_test_lib:start_mnesia(Nodes, [Tab])), ?match([Pre], mnesia:dirty_read({Tab, Key})), ?match([Pre], rpc:call(Node2, mnesia, dirty_read, [{Tab, Key}])), Dir = rpc:call(Node2, mnesia, system_info , [directory]), ?match(ok, mnesia:dirty_write(Post)), ?match([Post], mnesia:dirty_read({Tab, Key})), ?match([], mnesia_test_lib:kill_mnesia([Node2])), ?match(ok, mnesia:install_fallback(File, Local ++ [{mnesia_dir, Dir}])), ?match([], mnesia_test_lib:kill_mnesia([Node1])), ?match([], mnesia_test_lib:start_mnesia([Node2], [])), ?match(yes, rpc:call(Node2, mnesia, force_load_table, [Tab])), ?match([], mnesia_test_lib:start_mnesia(Nodes, [Tab])), ?match([Pre], mnesia:dirty_read({Tab, Key})), ?match(ok, file:delete(File)), ?verify_mnesia(Nodes, []). selective_backup_checkpoint(doc) -> ["Perform a selective backup of a checkpoint"]; selective_backup_checkpoint(suite) -> []; selective_backup_checkpoint(Config) when is_list(Config) -> [Node1, Node2] = Nodes = ?acquire_nodes(2, Config), Tab = sel_backup, OmitTab = sel_backup_omit, CpName = sel_cp, Def = [{disc_copies, [Node1, Node2]}], File = "selective_backup_checkpoint.BUP", ?match({atomic, ok}, mnesia:create_table(Tab, Def)), ?match({atomic, ok}, mnesia:create_table(OmitTab, Def)), ?match(ok, mnesia:dirty_write({Tab, 1, test_ok})), ?match(ok, mnesia:dirty_write({OmitTab, 1, test_ok})), CpSpec = [{name, CpName}, {max, mnesia:system_info(tables)}], ?match({ok, CpName, _Ns}, mnesia:activate_checkpoint(CpSpec)), BupSpec = [{tables, [Tab]}], ?match(ok, mnesia:backup_checkpoint(CpName, File, BupSpec)), ?match([schema, sel_backup], bup_tables(File, mnesia_backup)), ?match(ok, file:delete(File)), BupSpec2 = [{tables, [Tab, OmitTab]}], ?match(ok, mnesia:backup_checkpoint(CpName, File, BupSpec2)), ?match([schema, sel_backup, sel_backup_omit], bup_tables(File, mnesia_backup)), ?match(ok, file:delete(File)), ?verify_mnesia(Nodes, []). bup_tables(File, Mod) -> Fun = fun(Rec, Tabs) -> Tab = element(1, Rec), Tabs2 = [Tab | lists:delete(Tab, Tabs)], {[Rec], Tabs2} end, case mnesia:traverse_backup(File, Mod, dummy, read_only, Fun, []) of {ok, Tabs} -> lists:sort(Tabs); {error, Reason} -> exit(Reason) end. incremental_backup_checkpoint(doc) -> ["Perform a incremental backup of a checkpoint"]; incremental_backup_checkpoint(suite) -> []; incremental_backup_checkpoint(Config) when is_list(Config) -> [Node1] = Nodes = ?acquire_nodes(1, Config), Tab = incr_backup, Def = [{disc_copies, [Node1]}], ?match({atomic, ok}, mnesia:create_table(Tab, Def)), OldRecs = [{Tab, K, -K} || K <- lists:seq(1, 5)], ?match([ok|_], [mnesia:dirty_write(R) || R <- OldRecs]), OldCpName = old_cp, OldCpSpec = [{name, OldCpName}, {min, [Tab]}], ?match({ok, OldCpName, _Ns}, mnesia:activate_checkpoint(OldCpSpec)), BupSpec = [{tables, [Tab]}], OldFile = "old_full_backup.BUP", ?match(ok, mnesia:backup_checkpoint(OldCpName, OldFile, BupSpec)), ?match(OldRecs, bup_records(OldFile, mnesia_backup)), ?match(ok, mnesia:dirty_delete({Tab, 1})), ?match(ok, mnesia:dirty_write({Tab, 2, 2})), ?match(ok, mnesia:dirty_write({Tab, 3, -3})), NewCpName = new_cp, NewCpSpec = [{name, NewCpName}, {min, [Tab]}], ?match({ok, NewCpName, _Ns}, mnesia:activate_checkpoint(NewCpSpec)), ?match(ok, mnesia:dirty_write({Tab, 4, 4})), NewFile = "new_full_backup.BUP", ?match(ok, mnesia:backup_checkpoint(NewCpName, NewFile, BupSpec)), NewRecs = [{Tab, 2, 2}, {Tab, 3, -3}, {Tab, 4, 4}, {Tab, 4}, {Tab, 4, -4}, {Tab, 5, -5}], ?match(NewRecs, bup_records(NewFile, mnesia_backup)), DiffFile = "diff_backup.BUP", DiffBupSpec = [{tables, [Tab]}, {incremental, OldCpName}], ?match(ok, mnesia:backup_checkpoint(NewCpName, DiffFile, DiffBupSpec)), DiffRecs = [{Tab, 1}, {Tab, 2}, {Tab, 2, 2}, {Tab, 3}, {Tab, 3, -3}, {Tab, 4}, {Tab, 4, 4}, {Tab, 4}, {Tab, 4, -4}], ?match(DiffRecs, bup_records(DiffFile, mnesia_backup)), ?match(ok, mnesia:deactivate_checkpoint(OldCpName)), ?match(ok, mnesia:deactivate_checkpoint(NewCpName)), ?match(ok, file:delete(OldFile)), ?match(ok, file:delete(NewFile)), ?match(ok, file:delete(DiffFile)), ?verify_mnesia(Nodes, []). bup_records(File, Mod) -> Fun = fun(Rec, Recs) when element(1, Rec) == schema -> {[Rec], Recs}; (Rec, Recs) -> {[Rec], [Rec | Recs]} end, case mnesia:traverse_backup(File, Mod, dummy, read_only, Fun, []) of {ok, Recs} -> lists:keysort(1, lists:keysort(2, lists:reverse(Recs))); {error, Reason} -> exit(Reason) end. sops_with_checkpoint(doc) -> ["Test schema operations during a checkpoint"]; sops_with_checkpoint(suite) -> []; sops_with_checkpoint(Config) when is_list(Config) -> Ns = [N1,N2] = ?acquire_nodes(2, Config), ?match({ok, cp1, Ns}, mnesia:activate_checkpoint([{name, cp1},{max,mnesia:system_info(tables)}])), Tab = tab, ?match({atomic, ok}, mnesia:create_table(Tab, [{disc_copies,Ns}])), OldRecs = [{Tab, K, -K} || K <- lists:seq(1, 5)], [mnesia:dirty_write(R) || R <- OldRecs], ?match({ok, cp2, Ns}, mnesia:activate_checkpoint([{name, cp2},{max,mnesia:system_info(tables)}])), File1 = "cp1_delete_me.BUP", ?match(ok, mnesia:dirty_write({Tab,6,-6})), ?match(ok, mnesia:backup_checkpoint(cp1, File1)), ?match(ok, mnesia:dirty_write({Tab,7,-7})), File2 = "cp2_delete_me.BUP", ?match(ok, mnesia:backup_checkpoint(cp2, File2)), ?match(ok, mnesia:deactivate_checkpoint(cp1)), ?match(ok, mnesia:backup_checkpoint(cp2, File1)), ?match(ok, mnesia:dirty_write({Tab,8,-8})), ?match({atomic,ok}, mnesia:delete_table(Tab)), ?match({error,_}, mnesia:backup_checkpoint(cp2, File2)), ?match({'EXIT',_}, mnesia:dirty_write({Tab,9,-9})), ?match({atomic,_}, mnesia:restore(File1, [{default_op, recreate_tables}])), Test = fun(N) when N > 5 -> ?error("To many records in backup ~p ~n", [N]); (N) -> case mnesia:dirty_read(Tab,N) of [{Tab,N,B}] when -B =:= N -> ok; Other -> ?error("Not matching ~p ~p~n", [N,Other]) end end, [Test(N) || N <- mnesia:dirty_all_keys(Tab)], ?match({aborted,enoent}, mnesia:restore(File2, [{default_op, recreate_tables}])), %% Mnesia crashes when deleting a table during backup ?match([], mnesia_test_lib:stop_mnesia([N2])), Tab2 = ram, ?match({atomic, ok}, mnesia:create_table(Tab2, [{ram_copies,[N1]}])), ?match({ok, cp3, _}, mnesia:activate_checkpoint([{name, cp3}, {ram_overrides_dump,true}, {min,[Tab2]}])), Write = fun Loop (N) -> case N > 0 of true -> mnesia:dirty_write({Tab2, N+100, N+100}), Loop(N-1); false -> ok end end, ok = Write(100000), spawn_link(fun() -> ?match({atomic, ok},mnesia:delete_table(Tab2)) end), %% We don't check result here, depends on timing of above call mnesia:backup_checkpoint(cp3, File2), file:delete(File1), file:delete(File2), ?verify_mnesia([N1], [N2]).