356 lines
12 KiB
Elixir
356 lines
12 KiB
Elixir
## This Source Code Form is subject to the terms of the Mozilla Public
|
|
## License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
## file, You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
##
|
|
## Copyright (c) 2007-2020 VMware, Inc. or its affiliates. All rights reserved.
|
|
|
|
defmodule EnablePluginsCommandTest do
|
|
use ExUnit.Case, async: false
|
|
import TestHelper
|
|
|
|
alias RabbitMQ.CLI.Core.ExitCodes
|
|
|
|
@command RabbitMQ.CLI.Plugins.Commands.EnableCommand
|
|
@disable_command RabbitMQ.CLI.Plugins.Commands.DisableCommand
|
|
|
|
setup_all do
|
|
RabbitMQ.CLI.Core.Distribution.start()
|
|
node = get_rabbit_hostname()
|
|
|
|
{:ok, plugins_file} =
|
|
:rabbit_misc.rpc_call(node, :application, :get_env, [:rabbit, :enabled_plugins_file])
|
|
|
|
{:ok, plugins_dir} =
|
|
:rabbit_misc.rpc_call(node, :application, :get_env, [:rabbit, :plugins_dir])
|
|
|
|
rabbitmq_home = :rabbit_misc.rpc_call(node, :code, :lib_dir, [:rabbit])
|
|
|
|
IO.puts(
|
|
"plugins enable tests default env: enabled plugins = #{plugins_file}, plugins dir = #{plugins_dir}, RabbitMQ home directory = #{rabbitmq_home}"
|
|
)
|
|
|
|
{:ok, [enabled_plugins]} = :file.consult(plugins_file)
|
|
|
|
IO.puts(
|
|
"plugins enable tests will assume tnat #{Enum.join(enabled_plugins, ",")} is the list of enabled plugins to revert to"
|
|
)
|
|
|
|
opts = %{
|
|
enabled_plugins_file: plugins_file,
|
|
plugins_dir: plugins_dir,
|
|
rabbitmq_home: rabbitmq_home,
|
|
online: false,
|
|
offline: false,
|
|
all: false
|
|
}
|
|
|
|
on_exit(fn ->
|
|
set_enabled_plugins(enabled_plugins, :online, get_rabbit_hostname(), opts)
|
|
end)
|
|
|
|
{:ok, opts: opts}
|
|
end
|
|
|
|
setup context do
|
|
reset_enabled_plugins_to_preconfigured_defaults(context)
|
|
|
|
{
|
|
:ok,
|
|
opts:
|
|
Map.merge(context[:opts], %{
|
|
node: get_rabbit_hostname(),
|
|
timeout: 1000
|
|
})
|
|
}
|
|
end
|
|
|
|
def reset_enabled_plugins_to_preconfigured_defaults(context) do
|
|
set_enabled_plugins(
|
|
[:rabbitmq_stomp, :rabbitmq_federation],
|
|
:online,
|
|
get_rabbit_hostname(),
|
|
context[:opts]
|
|
)
|
|
end
|
|
|
|
test "validate: specifying both --online and --offline is reported as invalid", context do
|
|
assert match?(
|
|
{:validation_failure, {:bad_argument, _}},
|
|
@command.validate(["a"], Map.merge(context[:opts], %{online: true, offline: true}))
|
|
)
|
|
end
|
|
|
|
test "validate: not specifying any plugins to enable is reported as invalid", context do
|
|
assert match?(
|
|
{:validation_failure, :not_enough_args},
|
|
@command.validate([], Map.merge(context[:opts], %{online: true, offline: false}))
|
|
)
|
|
end
|
|
|
|
test "validate_execution_environment: specifying a non-existent enabled_plugins_file is fine",
|
|
context do
|
|
assert @command.validate_execution_environment(
|
|
["a"],
|
|
Map.merge(context[:opts], %{enabled_plugins_file: "none"})
|
|
) == :ok
|
|
end
|
|
|
|
test "validate_execution_environment: specifying a non-existent plugins_dir is reported as an error",
|
|
context do
|
|
assert @command.validate_execution_environment(
|
|
["a"],
|
|
Map.merge(context[:opts], %{plugins_dir: "none"})
|
|
) ==
|
|
{:validation_failure, :plugins_dir_does_not_exist}
|
|
end
|
|
|
|
test "run: if node is inaccessible, writes enabled plugins file and reports implicitly enabled plugin list",
|
|
context do
|
|
# Clears enabled plugins file
|
|
set_enabled_plugins([], :offline, :nonode, context[:opts])
|
|
|
|
assert {:stream, test_stream} =
|
|
@command.run(["rabbitmq_stomp"], Map.merge(context[:opts], %{node: :nonode}))
|
|
|
|
assert [
|
|
[:rabbitmq_stomp],
|
|
%{mode: :offline, enabled: [:rabbitmq_stomp], set: [:rabbitmq_stomp]}
|
|
] ==
|
|
Enum.to_list(test_stream)
|
|
|
|
check_plugins_enabled([:rabbitmq_stomp], context)
|
|
|
|
assert [:amqp_client, :rabbitmq_federation, :rabbitmq_stomp] ==
|
|
currently_active_plugins(context)
|
|
end
|
|
|
|
test "run: in offline mode, writes enabled plugins and reports implicitly enabled plugin list",
|
|
context do
|
|
# Clears enabled plugins file
|
|
set_enabled_plugins([], :offline, :nonode, context[:opts])
|
|
|
|
assert {:stream, test_stream} =
|
|
@command.run(
|
|
["rabbitmq_stomp"],
|
|
Map.merge(context[:opts], %{offline: true, online: false})
|
|
)
|
|
|
|
assert [
|
|
[:rabbitmq_stomp],
|
|
%{mode: :offline, enabled: [:rabbitmq_stomp], set: [:rabbitmq_stomp]}
|
|
] ==
|
|
Enum.to_list(test_stream)
|
|
|
|
check_plugins_enabled([:rabbitmq_stomp], context)
|
|
|
|
assert_equal_sets(
|
|
[:amqp_client, :rabbitmq_federation, :rabbitmq_stomp],
|
|
currently_active_plugins(context)
|
|
)
|
|
end
|
|
|
|
test "run: adds additional plugins to those already enabled", context do
|
|
# Clears enabled plugins file
|
|
set_enabled_plugins([], :offline, :nonode, context[:opts])
|
|
|
|
assert {:stream, test_stream0} =
|
|
@command.run(
|
|
["rabbitmq_stomp"],
|
|
Map.merge(context[:opts], %{offline: true, online: false})
|
|
)
|
|
|
|
assert [
|
|
[:rabbitmq_stomp],
|
|
%{mode: :offline, enabled: [:rabbitmq_stomp], set: [:rabbitmq_stomp]}
|
|
] ==
|
|
Enum.to_list(test_stream0)
|
|
|
|
check_plugins_enabled([:rabbitmq_stomp], context)
|
|
|
|
assert {:stream, test_stream1} =
|
|
@command.run(
|
|
["rabbitmq_federation"],
|
|
Map.merge(context[:opts], %{offline: true, online: false})
|
|
)
|
|
|
|
assert [
|
|
[:rabbitmq_federation, :rabbitmq_stomp],
|
|
%{
|
|
mode: :offline,
|
|
enabled: [:rabbitmq_federation],
|
|
set: [:rabbitmq_federation, :rabbitmq_stomp]
|
|
}
|
|
] ==
|
|
Enum.to_list(test_stream1)
|
|
|
|
check_plugins_enabled([:rabbitmq_stomp, :rabbitmq_federation], context)
|
|
end
|
|
|
|
test "run: updates plugin list and starts newly enabled plugins", context do
|
|
# Clears enabled plugins file and stop all plugins
|
|
set_enabled_plugins([], :online, context[:opts][:node], context[:opts])
|
|
|
|
assert {:stream, test_stream0} = @command.run(["rabbitmq_stomp"], context[:opts])
|
|
|
|
assert [
|
|
[:rabbitmq_stomp],
|
|
%{
|
|
mode: :online,
|
|
started: [:rabbitmq_stomp],
|
|
stopped: [],
|
|
enabled: [:rabbitmq_stomp],
|
|
set: [:rabbitmq_stomp]
|
|
}
|
|
] ==
|
|
Enum.to_list(test_stream0)
|
|
|
|
check_plugins_enabled([:rabbitmq_stomp], context)
|
|
assert_equal_sets([:amqp_client, :rabbitmq_stomp], currently_active_plugins(context))
|
|
|
|
{:stream, test_stream1} = @command.run(["rabbitmq_federation"], context[:opts])
|
|
|
|
assert [
|
|
[:rabbitmq_federation, :rabbitmq_stomp],
|
|
%{
|
|
mode: :online,
|
|
started: [:rabbitmq_federation],
|
|
stopped: [],
|
|
enabled: [:rabbitmq_federation],
|
|
set: [:rabbitmq_federation, :rabbitmq_stomp]
|
|
}
|
|
] ==
|
|
Enum.to_list(test_stream1)
|
|
|
|
check_plugins_enabled([:rabbitmq_stomp, :rabbitmq_federation], context)
|
|
|
|
assert_equal_sets(
|
|
[:amqp_client, :rabbitmq_federation, :rabbitmq_stomp],
|
|
currently_active_plugins(context)
|
|
)
|
|
|
|
reset_enabled_plugins_to_preconfigured_defaults(context)
|
|
end
|
|
|
|
test "run: can enable multiple plugins at once", context do
|
|
# Clears plugins file and stop all plugins
|
|
set_enabled_plugins([], :online, context[:opts][:node], context[:opts])
|
|
|
|
assert {:stream, test_stream} =
|
|
@command.run(["rabbitmq_stomp", "rabbitmq_federation"], context[:opts])
|
|
|
|
assert [
|
|
[:rabbitmq_federation, :rabbitmq_stomp],
|
|
%{
|
|
mode: :online,
|
|
started: [:rabbitmq_federation, :rabbitmq_stomp],
|
|
stopped: [],
|
|
enabled: [:rabbitmq_federation, :rabbitmq_stomp],
|
|
set: [:rabbitmq_federation, :rabbitmq_stomp]
|
|
}
|
|
] ==
|
|
Enum.to_list(test_stream)
|
|
|
|
check_plugins_enabled([:rabbitmq_stomp, :rabbitmq_federation], context)
|
|
|
|
assert_equal_sets(
|
|
[:amqp_client, :rabbitmq_federation, :rabbitmq_stomp],
|
|
currently_active_plugins(context)
|
|
)
|
|
|
|
reset_enabled_plugins_to_preconfigured_defaults(context)
|
|
end
|
|
|
|
test "run: does not enable an already implicitly enabled plugin", context do
|
|
# Clears enabled plugins file and stop all plugins
|
|
set_enabled_plugins([:rabbitmq_federation], :online, context[:opts][:node], context[:opts])
|
|
assert {:stream, test_stream} = @command.run(["amqp_client"], context[:opts])
|
|
|
|
assert [
|
|
[:rabbitmq_federation],
|
|
%{mode: :online, started: [], stopped: [], enabled: [], set: [:rabbitmq_federation]}
|
|
] ==
|
|
Enum.to_list(test_stream)
|
|
|
|
check_plugins_enabled([:rabbitmq_federation], context)
|
|
|
|
assert [:amqp_client, :rabbitmq_federation] ==
|
|
currently_active_plugins(context)
|
|
|
|
reset_enabled_plugins_to_preconfigured_defaults(context)
|
|
end
|
|
|
|
test "run: does not enable plugins with unmet version requirements", context do
|
|
set_enabled_plugins([], :online, context[:opts][:node], context[:opts])
|
|
|
|
plugins_directory = fixture_plugins_path("plugins_with_version_requirements")
|
|
opts = get_opts_with_plugins_directories(context, [plugins_directory])
|
|
switch_plugins_directories(context[:opts][:plugins_dir], opts[:plugins_dir])
|
|
|
|
{:stream, _} = @command.run(["mock_rabbitmq_plugin_for_3_9"], opts)
|
|
check_plugins_enabled([:mock_rabbitmq_plugin_for_3_9], context)
|
|
|
|
# Not changed
|
|
{:error, _version_error} = @command.run(["mock_rabbitmq_plugin_for_3_7"], opts)
|
|
check_plugins_enabled([:mock_rabbitmq_plugin_for_3_9], context)
|
|
|
|
reset_enabled_plugins_to_preconfigured_defaults(context)
|
|
end
|
|
|
|
test "run: does not enable plugins with unmet version requirements even when enabling all plugins",
|
|
context do
|
|
set_enabled_plugins([], :online, context[:opts][:node], context[:opts])
|
|
|
|
plugins_directory = fixture_plugins_path("plugins_with_version_requirements")
|
|
opts = get_opts_with_plugins_directories(context, [plugins_directory])
|
|
opts = Map.merge(opts, %{all: true})
|
|
switch_plugins_directories(context[:opts][:plugins_dir], opts[:plugins_dir])
|
|
|
|
{:error, _version_error} = @command.run([], opts)
|
|
|
|
check_plugins_enabled([], context)
|
|
|
|
reset_enabled_plugins_to_preconfigured_defaults(context)
|
|
end
|
|
|
|
test "output: formats enabled plugins mismatch errors", context do
|
|
err = {:enabled_plugins_mismatch, '/tmp/a/cli/path', '/tmp/a/server/path'}
|
|
|
|
assert {:error, ExitCodes.exit_dataerr(),
|
|
"Could not update enabled plugins file at /tmp/a/cli/path: target node #{context[:opts][:node]} uses a different path (/tmp/a/server/path)"} ==
|
|
@command.output({:error, err}, context[:opts])
|
|
|
|
reset_enabled_plugins_to_preconfigured_defaults(context)
|
|
end
|
|
|
|
test "output: formats enabled plugins write errors", context do
|
|
err1 = {:cannot_write_enabled_plugins_file, "/tmp/a/path", :eacces}
|
|
|
|
assert {:error, ExitCodes.exit_dataerr(),
|
|
"Could not update enabled plugins file at /tmp/a/path: the file does not exist or permission was denied (EACCES)"} ==
|
|
@command.output({:error, err1}, context[:opts])
|
|
|
|
err2 = {:cannot_write_enabled_plugins_file, "/tmp/a/path", :enoent}
|
|
|
|
assert {:error, ExitCodes.exit_dataerr(),
|
|
"Could not update enabled plugins file at /tmp/a/path: the file does not exist (ENOENT)"} ==
|
|
@command.output({:error, err2}, context[:opts])
|
|
|
|
reset_enabled_plugins_to_preconfigured_defaults(context)
|
|
end
|
|
|
|
test "output: enable command will also load its dependent plugins", context do
|
|
# Clears enabled plugins file and stop all plugins
|
|
set_enabled_plugins([], :online, context[:opts][:node], context[:opts])
|
|
|
|
# Enable rabbitmq_stream_management
|
|
@command.run(["rabbitmq_stream_management"], context[:opts])
|
|
|
|
# Check command add_super_stream is available due to dependency plugin rabbitmq_stream
|
|
assert RabbitMQ.CLI.Core.CommandModules.load_commands(:all, %{})["add_super_stream"] ==
|
|
RabbitMQ.CLI.Ctl.Commands.AddSuperStreamCommand
|
|
|
|
@disable_command.run(["rabbitmq_stream_management"], context[:opts])
|
|
reset_enabled_plugins_to_preconfigured_defaults(context)
|
|
end
|
|
end
|