Initial version of rabbitmqctl import_definitions

This commit is contained in:
Michael Klishin 2019-11-16 05:18:52 +03:00
parent 27212997c2
commit fa0db7ad22
2 changed files with 139 additions and 1 deletions

View File

@ -83,7 +83,7 @@ defmodule RabbitMQ.CLI.Ctl.Commands.ExportDefinitionsCommand do
{:error, :eisdir} ->
{:error, ExitCodes.exit_dataerr(), "Path #{path} is a directory"}
{:error, err} ->
{:error, ExitCodes.exit_dataerr(), "Could not write to file: #{err}"}
{:error, ExitCodes.exit_dataerr(), "Could not write to file #{path}: #{err}"}
end
end
end

View File

@ -0,0 +1,138 @@
## The contents of this file are subject to the Mozilla Public License
## Version 1.1 (the "License"); you may not use this file except in
## compliance with the License. You may obtain a copy of the License
## at https://www.mozilla.org/MPL/
##
## Software distributed under the License is distributed on an "AS IS"
## basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See
## the License for the specific language governing rights and
## limitations under the License.
##
## The Original Code is RabbitMQ.
##
## The Initial Developer of the Original Code is GoPivotal, Inc.
## Copyright (c) 2007-2019 Pivotal Software, Inc. All rights reserved.
defmodule RabbitMQ.CLI.Ctl.Commands.ImportDefinitionsCommand do
alias RabbitMQ.CLI.Core.{DocGuide, ExitCodes, Helpers}
@behaviour RabbitMQ.CLI.CommandBehaviour
def merge_defaults(["-"] = args, opts) do
{args, Map.merge(%{format: "json", silent: true}, Helpers.case_insensitive_format(opts))}
end
def merge_defaults(args, opts) do
{args, Map.merge(%{format: "json"}, Helpers.case_insensitive_format(opts))}
end
def switches(), do: [timeout: :integer, format: :string]
def aliases(), do: [t: :timeout]
def validate(_, %{format: format})
when format != "json" and format != "JSON" and format != "erlang" do
{:validation_failure, {:bad_argument, "Format should be either json or erlang"}}
end
def validate(args, _) when length(args) > 1 do
{:validation_failure, :too_many_args}
end
# output to stdout
def validate(["-"], _) do
:ok
end
def validate([path], _) do
dir = Path.dirname(path)
case File.exists?(dir, [raw: true]) do
true -> :ok
false -> {:validation_failure, {:bad_argument, "Directory #{dir} does not exist"}}
end
end
def validate(_, _), do: :ok
use RabbitMQ.CLI.Core.RequiresRabbitAppRunning
def run([], %{node: node_name, format: format, timeout: timeout} = opts) do
# case Input.consume_stdio do
# :eof -> {:error, :not_enough_args}
# bin ->
# {:ok, map} = deserialise(bin, format)
# :rabbit_misc.rpc_call(node_name, :rabbit_definition, :import_parsed, [map], timeout)
# end
end
def run([path], %{node: node_name, timeout: timeout, format: format}) do
abs_path = Path.absname(path)
case File.read(abs_path) do
# no output
{:ok, bin} ->
{:ok, map} = deserialise(bin, format)
:rabbit_misc.rpc_call(node_name, :rabbit_definitions, :import_parsed, [map], timeout)
{:error, :enoent} ->
{:error, ExitCodes.exit_dataerr(), "Parent directory or file #{path} does not exist"}
{:error, :enotdir} ->
{:error, ExitCodes.exit_dataerr(), "Parent directory of file #{path} is not a directory"}
{:error, :eacces} ->
{:error, ExitCodes.exit_dataerr(), "No permissions to read from file #{path} or its parent directory"}
{:error, :eisdir} ->
{:error, ExitCodes.exit_dataerr(), "Path #{path} is a directory"}
{:error, err} ->
{:error, ExitCodes.exit_dataerr(), "Could not read from file #{path}: #{err}"}
end
end
def output(:ok, %{node: node_name, formatter: "json"}) do
{:ok, %{"result" => "ok", "node" => node_name}}
end
def output(:ok, %{silent: silent}) when silent == false do
{:ok, "Successfully started definition import. " <>
"This process is asynchronous and can take some time."}
end
def output(:ok, %{quiet: quiet}) when quiet == false do
{:ok, "Successfully started definition import. " <>
"This process is asynchronous and can take some time."}
end
use RabbitMQ.CLI.DefaultOutput
def printer(), do: RabbitMQ.CLI.Printers.StdIORaw
def usage, do: "import_definitions <file_path | \"-\"> [--format <json | erlang>]"
def usage_additional() do
[
["[file]", "Local file path to import from. If omitted will be read from standard input."],
["--format", "input format to use: json or erlang"]
]
end
def usage_doc_guides() do
[
DocGuide.definitions()
]
end
def help_section(), do: :definitions
def description(), do: "Imports definitions in JSON or compressed Erlang Term Format."
def banner([], %{format: fmt}) do
"Importing definitions in #{human_fiendly_format(fmt)} from standard input ..."
end
def banner([path], %{format: fmt}) do
"Importing definitions in #{human_fiendly_format(fmt)} from a file at \"#{path}\" ..."
end
#
# Implementation
#
defp deserialise(bin, "json") do
JSON.decode(bin)
end
defp deserialise(bin, "erlang") do
{:ok, :erlang.binary_to_term(bin)}
end
defp human_fiendly_format("JSON"), do: "JSON"
defp human_fiendly_format("json"), do: "JSON"
defp human_fiendly_format("erlang"), do: "Erlang term format"
end