% @doc `rebar3 hex publish' - publish packages and docs
%
%% Builds and publishing a new version of a package and docs.
%%
%% ```
%% $ rebar3 hex publish
%% '''
%%
%% The current authenticated user will be the package owner. Only package owners can publish the package,
%% new owners can be added with the `rebar3 hex owner' task.
%%
%% Packages and documentation sizes are limited to 8mb compressed, and 64mb uncompressed.
%% By default this provider will build both a package tarball and docs tarball. Before attempting to
%% publish your package see the section below on documentation.
%%
%%
Publishing Documentation
%%
%% By default rebar3_hex will not generate or publish docs. You must add a doc provider to your the hex config in either
%% your global `rebar.config' (i.e., `~/.config/rebar3/rebar.config') or locally in a projects `rebar.config'.
%%
%% We recommend using rebar3_ex_doc for publishing documentation along
%% with your package for a consistent format and style on hexpm.
%%
%% Example :
%%
%% ```
%% {ex_doc, [
%% {source_url, <<"https://github.com/namespace/your_app">>},
%% {extras, [<<"README.md">>, <<"LICENSE">>]},
%% {main, <<"readme">>}
%% ]}.
%%
%% {hex, [{doc, ex_doc}]}.
%% '''
%%
%% Alternatively you can use the edoc provider that ships with rebar3 :
%%
%% ```erlang
%% {hex, [{doc, edoc}]}.
%% '''
%%
%% The expected result of the task is the generated documentation located in the either `doc/' directory with an
%% `index.html' file or the directory you have configured the documentation provider to use.
%%
%% The documentation will be accessible at `https://hexdocs.pm/my_package/1.0.0', `https://hexdocs.pm/my_package' will always redirect to the latest published version.
%%
%% When a doc provider is configured documentation will be built and published automatically. To publish a package
%% without documentation run `mix rebar3 hex publish package' or to only publish documentation
%% run `rebar3 hex publish docs'.
%%
%% Reverting a package
%%
%% A new package can be reverted or updated within 24 hours of it's initial publish, a new version of an existing
%% package can be reverted or updated within one hour. Documentation have no limitations on when it can be updated.
%%
%% To update the package simply run the `rebar hex publish' task again.
%% To revert run `rebar3 hex publish --revert VERSION' or to only revert the documentation
%% run `rebar3 hex publish docs --revert VERSION'.
%%
%% If the last version is reverted, the package is removed.
%%
%% Command line options
%%
%%
%% - `-a', `--app' - Specify which app you want to publish without being prompted to choose. This
%% option is required when reverting a package, otherwise it's only useful in the context of an umbrella.
%% - `-r', `--repo' - Specify the repository work with. This option is required when
%% you have multiple repositories configured, including organizations. The argument must
%% be a fully qualified repository name (e.g, `hexpm', `hexpm:my_org', `my_own_hexpm').
%% Defaults to `hexpm'.
%%
%% - `-y', `--yes' - Publishes the package without any confirmation prompts
%% - `--replace' - Allows overwriting an existing package version if it exists. Private packages can always be
%% overwritten, public packages can only be overwritten within one hour after they were initially
%% published.
%% - `--revert VERSION' - Revert the given version. If the last version is reverted, the package is removed.
%% - `--dry-run' - Performs a dry run of the publish task (i.e., it won't actually publish).
%% - `--doc-dir' - The option may be used in case there is no doc provider available for your desired method of
%% generating documentation. Instead of finding and running a doc provider, rebar3_hex will simply gather
%% the files specified in the directory given to this option. Be sure your docs are up to date before publishing
%% with this option.
%%
%%
-module(rebar3_hex_publish).
-export([ init/1
, do/1
, format_error/1
]).
-export([
publish/4,
publish_package/4
, publish_docs/4
]).
-include("rebar3_hex.hrl").
-define(PROVIDER, publish).
-define(DEPS, [{default, lock}]).
%% ===================================================================
%% Public API
%% ===================================================================
-spec init(rebar_state:t()) -> {ok, rebar_state:t()}.
init(State) ->
Provider = providers:create([{name, ?PROVIDER},
{module, ?MODULE},
{namespace, hex},
{bare, true},
{deps, ?DEPS},
{example, "rebar3 hex publish"},
{short_desc, "Publish a new version of your package and update the package"},
{desc, support()},
{opts, [rebar3_hex:repo_opt(),
{yes, $y, "yes", {boolean, false}, help(yes)},
{app, $a, "app", {string, undefined}, help(app)},
{doc_dir, undefined, "doc-dir", {string, undefined}, help(yes)},
{dry_run, undefined, "dry-run", {boolean, false}, help(dry_run)},
{replace, undefined, "replace", {boolean, false}, help(replace)},
{revert, undefined, "revert", string, help(revert)}]}]),
State1 = rebar_state:add_provider(State, Provider),
{ok, State1}.
-spec do(rebar_state:t()) -> {ok, rebar_state:t()} | {error, term()}.
do(State) ->
case rebar3_hex:task_state(State) of
{ok, Task} ->
handle_task(Task);
{error, Reason} ->
?RAISE(Reason)
end.
%% TODO: Move me somewhere else
-spec format_error(any()) -> iolist().
format_error(ErrList) when is_list(ErrList) ->
F = fun(Err, Acc) ->
ErrStr = format_error(Err),
Acc ++ " " ++ ErrStr ++ "\n"
end,
More = "\n Please see https://hex.pm/docs/rebar3_publish for more info.\n",
lists:foldl(F, "Validator Errors:\n", ErrList) ++ More;
format_error(no_apps_found) ->
"publish can not continue, no apps were found.";
%% Option errors
format_error({app_not_found, AppName}) ->
io_lib:format("App ~s specified with --app switch not found in project", [AppName]);
format_error({publish_package, app_switch_required}) ->
"--app required when publishing with the package argument in a umbrella";
format_error({publish_docs, app_switch_required}) ->
"--app required when publishing with the docs argument in a umbrella";
format_error({revert, app_switch_required}) ->
"--app required when reverting in a umbrella with multiple apps";
format_error({required, repo}) ->
"publish requires a repo name argument to identify the repo to publish to";
format_error({not_valid_repo, RepoName}) ->
io_lib:format("No configuration for repository ~ts found.", [RepoName]);
%% Validation errors
format_error({validation_errors, Errs}) ->
lists:map(fun(E) -> format_error(E) end, Errs);
format_error({has_contributors, AppName}) ->
Err = "~ts.app.src : deprecated field contributors found",
io_lib:format(Err, [AppName]);
format_error({has_maintainers, AppName}) ->
Err = "~ts.app.src : deprecated field maintainers found",
io_lib:format(Err, [AppName]);
format_error({no_description, AppName}) ->
Err = "~ts.app.src : missing or empty description property",
io_lib:format(Err, [AppName]);
format_error({no_license, AppName}) ->
Err = "~ts.app.src : missing or empty licenses property",
io_lib:format(Err, [AppName]);
format_error({invalid_licenses, Invalids, AppName}) ->
InvalidLicenses = string:join([ "'" ++ L ++ "'" || L <- Invalids], ", "),
Url = "See https://spdx.org/licenses/ for a list of valid license identifiers",
Err = "~ts.app.src : invalid license types detected - ~ts~n~5c~ts~n",
io_lib:format(Err, [AppName, InvalidLicenses, 32, Url]);
format_error({invalid_semver, {AppName, Version}}) ->
Err = "~ts.app.src : non-semantic version number \"~ts\" found",
io_lib:format(Err, [AppName, Version]);
format_error({invalid_semver_arg, Vsn}) ->
io_lib:format("The version argument provided \"~s\" is not a valid semantic version.", [Vsn]);
format_error({has_unstable_deps, Deps}) ->
MainMsg = "The following pre-release dependencies were found : ",
DepList = [io_lib:format("~s - ~s ", [Pkg, Ver]) || {Pkg, Ver} <- Deps],
Msg = [
"In the future packages with pre-release dependencies will be considered unstable ",
"and will be prevented from being published. ",
"We recommend you upgrade your these dependencies as soon as possible"
],
io_lib:format("~s~n~n~s~n~n~s~n", [MainMsg, DepList, Msg]);
format_error({non_hex_deps, Excluded}) ->
Err = "Can not publish package because the following deps are not available"
++ " in hex: ~s",
io_lib:format(Err, [string:join(Excluded, ", ")]);
%% create package errors
format_error({create_package, {error, Err}}) when is_list(Err) ->
io_lib:format("Error creating package : ~ts", [Err]);
%% create docs errors
format_error({create_docs, {error, {doc_provider_not_found, PrvName}}}) ->
io_lib:format("The ~ts documentation provider could not be found", [PrvName]);
format_error({create_docs, {error, {doc_provider_failed, PrvName}}}) ->
io_lib:format("The ~ts documentation provider failed", [PrvName]);
format_error({create_docs, {error, missing_doc_index}}) ->
"No index.html file was found in expected docs directory.\n"
"If you provided --doc-dir option ensure that your docs were generated before running this task.\n"
"Otherwise, check that your preferred doc provider is properly generating docs outside the scope of this task.\n"
"Once resolved, run rebar3 hex publish docs to publish only the docs for this version of your package.\n";
%% publish package errors
format_error({publish_package, Name, Version, {error, #{<<"errors">> := Errors, <<"message">> := Message}}}) ->
ErrorString = errors_to_string(Errors),
io_lib:format("Failed to publish package ~ts - ~ts : ~ts~n\t~ts", [Name, Version, Message, ErrorString]);
format_error({publish_package, Name, Version, {error, #{<<"message">> := Message}}}) ->
io_lib:format("Failed to publish package ~ts - ~ts : ~ts", [Name, Version, Message]);
%% publish docs errors
format_error({publish_docs, Name, Version, {error, #{<<"errors">> := Errors, <<"message">> := Message}}}) ->
ErrorString = errors_to_string(Errors),
io_lib:format("Failed to publish docs for ~ts - ~ts : ~ts~n\t~ts", [Name, Version, Message, ErrorString]);
format_error({publish_docs, Name, Version, {error, #{<<"message">> := Message}}}) ->
Reason = case Message of
<<"Page not found">> ->
io_lib:format("the ~ts package at version ~ts could not be found.", [Name, Version]);
Other ->
Other
end,
io_lib:format("Failed to publish docs for ~ts - ~ts : ~ts", [Name, Version, Reason]);
%% revert package errors
format_error({revert_package, Name, Version, {error, #{<<"errors">> := Errors, <<"message">> := Message}}}) ->
ErrorString = errors_to_string(Errors),
io_lib:format("Failed to revert package ~ts - ~ts : ~ts~n\t~ts", [Name, Version, Message, ErrorString]);
format_error({revert_package, Name, Version, {error, #{<<"message">> := Message}}}) ->
io_lib:format("Failed to revert package ~ts - ~ts : ~ts", [Name, Version, Message]);
%% revert docs errors
format_error({revert_docs, Name, Version, {error, #{<<"errors">> := Errors, <<"message">> := Message}}}) ->
ErrorString = errors_to_string(Errors),
io_lib:format("Failed to revert docs for ~ts - ~ts : ~ts~n\t~ts", [Name, Version, Message, ErrorString]);
format_error({revert_docs, Name, Version, {error, #{<<"message">> := Message}}}) ->
io_lib:format("Failed to revert docs for ~ts - ~ts : ~ts", [Name, Version, Message]);
%% TODO: Check if this is dead code
%% Server errors
format_error(undefined_server_error) ->
"Unknown server error";
format_error({status, Status}) ->
rebar3_hex_client:pretty_print_status(Status);
format_error({status, Status, undefined_server_error}) ->
"Unknown server error: " ++ rebar3_hex_client:pretty_print_status(Status);
format_error({status, Status, Error}) ->
Message = maps:get(<<"message">>, Error, ""),
Errors = maps:get(<<"errors">>, Error, ""),
ErrorString = errors_to_string(Errors),
Data = [rebar3_hex_client:pretty_print_status(Status), Message, ErrorString],
io_lib:format("Status Code: ~s~nHex Error: ~s~n\t~s", Data);
format_error(bad_command) ->
"Invalid arguments, expected one of:\n\n"
"rebar3 hex publish\n"
"rebar3 hex publish package\n"
"rebar3 hex publish docs\n";
format_error(Reason) ->
rebar3_hex_error:format_error(Reason).
%% ===================================================================
%% Private
%% ===================================================================
%% ===================================================================
%% Publish package only operations
%% ===================================================================
handle_task(#{apps := []}) ->
?RAISE(no_apps_found);
handle_task(#{args := #{task := package, app := undefined}, multi_app := true}) ->
?RAISE({publish_package, app_switch_required});
handle_task(#{args := #{task := package, revert := Vsn}, apps := [App]} = Task) ->
#{repo := Repo, state := State} = Task,
AppName = rebar_app_info:name(App),
revert_package(State, Repo, AppName, Vsn);
handle_task(#{args := #{task := package}, apps := [App]} = Task) ->
maybe_warn_about_single_app_args(Task),
#{args := Args, repo := Repo, state := State} = Task,
HexConfig = rebar3_hex_config:get_hex_config(?MODULE, Repo, write),
rebar_api:info("package argument given, will not publish docs", []),
publish_package(State, HexConfig, App, Args);
handle_task(#{args := #{task := package, app := AppName}, apps := Apps} = Task) ->
#{args := Args, repo := Repo, state := State} = Task,
case rebar3_hex_app:find(Apps, AppName) of
{error, app_not_found} ->
?RAISE({app_not_found, AppName});
{ok, App} ->
HexConfig = rebar3_hex_config:get_hex_config(?MODULE, Repo, write),
rebar_api:info("package argument given, will not publish docs", []),
publish_package(State, HexConfig, App, Args)
end,
{ok, State};
%% ===================================================================
%% Publish docs only operations
%% ===================================================================
handle_task(#{args := #{task := docs, app := undefined}, multi_app := true}) ->
?RAISE({publish_docs, app_switch_required});
handle_task(#{args := #{task := docs, revert := Vsn}, apps := [App]} = Task) ->
#{repo := Repo, state := State} = Task,
AppName = rebar_app_info:name(App),
revert_docs(State, Repo, AppName, Vsn);
handle_task(#{args := #{task := docs}, apps := [App]} = Task) ->
#{args := Args, repo := Repo, state := State} = Task,
rebar_api:info("docs argument given, will not publish package", []),
publish_docs(State, Repo, App, Args),
{ok, State};
handle_task(#{args := #{task := docs, app := AppName}, apps := Apps} = Task) ->
case rebar3_hex_app:find(Apps, AppName) of
{error, app_not_found} ->
?RAISE({app_not_found, AppName});
{ok, App} ->
#{args := Args, repo := Repo, state := State} = Task,
rebar_api:info("docs argument given, will not publish package", []),
publish_docs(State, Repo, App, Args),
{ok, State}
end;
%% ===================================================================
%% Revert operations
%% ===================================================================
handle_task(#{args := #{revert := _, app := undefined}, multi_app := true}) ->
?RAISE({revert, app_switch_required});
handle_task(#{args := #{revert := Vsn}, apps := [App]} = Task) ->
#{repo := Repo, state := State} = Task,
AppName = rebar_app_info:name(App),
revert_package(State, Repo, AppName, Vsn);
handle_task(#{args := #{revert := Vsn, app := AppName}, apps := Apps} = Task) ->
#{repo := Repo, state := State} = Task,
case rebar3_hex_app:find(Apps, AppName) of
{error, app_not_found} ->
?RAISE({app_not_found, AppName});
{ok, _App} ->
revert_package(State, Repo, AppName, Vsn)
end;
%% ===================================================================
%% Publish package and docs (the default path)
%% ===================================================================
handle_task(#{args := #{task := undefined, app := undefined}, repo := Repo, state := State, apps := Apps} = Task) ->
#{args := Args} = Task,
maybe_warn_about_single_app_args(Task),
Selected = rebar3_hex_io:select_apps(Apps),
lists:foreach(fun(App) -> publish(State, Repo, App, Args) end, Selected),
{ok, State};
handle_task(#{args := #{task := undefined, app := AppName}, apps := Apps, multi_app := true} = Task) ->
case rebar3_hex_app:find(Apps, AppName) of
{error, app_not_found} ->
?RAISE({app_not_found, AppName});
{ok, App} ->
#{args := Args, repo := Repo, state := State} = Task,
publish(State, Repo, App, Args)
end;
handle_task(_) ->
?RAISE(bad_command).
-dialyzer({nowarn_function, publish/4}).
publish(State, Repo, App, Args) ->
maybe_warn_about_doc_config(State, Repo),
HexConfig = rebar3_hex_config:get_hex_config(?MODULE, Repo, write),
case publish_package(State, HexConfig, App, Args) of
abort ->
{ok, State};
_ ->
publish_docs(State, HexConfig, App, Args)
end.
%%% ===================================================================
%%% package publishing and related functions
%%% ===================================================================
-dialyzer({nowarn_function, publish_package/4}).
publish_package(State, Repo, App, Args) ->
assert_valid_app(State, App),
Package = create_package(State, App),
maybe_print_checkouts_warnings(App, Package),
print_package_info(Package, Repo),
maybe_say_coc(Repo),
MaybeCheckoutWarnings = maybe_checkout_warnings(Package),
case maybe_prompt(Args, "Proceed" ++ MaybeCheckoutWarnings ++ "?") of
proceed ->
HexOpts = hex_opts(Args),
#{tarball := Tarball} = Package,
case Args of
#{dry_run := true} ->
rebar_api:info("--dry-run enabled : will not publish package.", []),
{ok, State};
_ ->
case rebar3_hex_client:publish(Repo, Tarball, HexOpts) of
{ok, _Res} ->
#{name := Name, version := Version} = Package,
rebar_api:info("Published ~ts ~ts", [Name, Version]),
{ok, State};
Error ->
#{name := Name, version := Version} = Package,
?RAISE({publish_package, Name, Version, Error})
end
end;
abort ->
rebar3_hex_io:say("Goodbye..."),
abort
end.
create_package(State, App) ->
case rebar3_hex_build:create_package(State, App) of
{ok, Package} ->
Package;
Err ->
?RAISE({create_package, Err})
end.
print_package_info(Package, #{name := RepoName} = _Repo) ->
#{metadata := Meta, files := Files, deps := Deps, name := Name, version := Version} = Package,
rebar3_hex_io:say("Publishing ~ts ~ts to ~ts", [Name, Version, RepoName]),
rebar3_hex_io:say(" Description: ~ts", [rebar_utils:to_list(maps:get(<<"description">>, Meta, <<"">>))]),
rebar3_hex_io:say(" Dependencies:~n ~ts", [format_deps(Deps)]),
rebar3_hex_io:say(" Included files:~n ~ts", [string:join([F || {F, _} <- Files], "\n ")]),
rebar3_hex_io:say(" Licenses: ~ts", [format_licenses(maps:get(<<"licenses">>, Meta, []))]),
rebar3_hex_io:say(" Links:~n ~ts", [format_links(maps:get(<<"links">>, Meta, []))]),
rebar3_hex_io:say(" Build tools: ~ts", [format_build_tools(maps:get(<<"build_tools">>, Meta))]).
format_build_tools(BuildTools) ->
string:join([io_lib:format("~ts", [Tool]) || Tool <- BuildTools], ", ").
format_deps(Deps) ->
Res = [rebar_utils:to_list(<>) || {N, #{<<"requirement">> := V}} <- Deps],
string:join(Res, "\n ").
format_licenses(Licenses) ->
string:join([rebar_utils:to_list(L) || L <- Licenses], ", ").
format_links(Links) ->
Links1 = maps:to_list(Links),
LinksList = [lists:flatten([rebar_utils:to_list(Name), ": ", rebar_utils:to_list(Url)]) || {Name, Url} <- Links1],
string:join(LinksList, "\n ").
%% if publishing to the public repo or to a private organization link to the code of conduct
maybe_say_coc(#{parent := <<"hexpm">>}) ->
rebar3_hex_io:say("Before publishing, please read Hex CoC: https://hex.pm/policies/codeofconduct", []);
maybe_say_coc(#{name := <<"hexpm">>}) ->
rebar3_hex_io:say("Be aware, you are publishing to the public Hexpm repository.", []),
rebar3_hex_io:say("Before publishing, please read Hex CoC: https://hex.pm/policies/codeofconduct", []);
maybe_say_coc(_) ->
ok.
maybe_checkout_warnings(#{has_checkouts := false}) -> "";
maybe_checkout_warnings(#{has_checkouts := true}) -> " (with warnings)".
maybe_print_checkouts_warnings(_App, #{has_checkouts := false}) -> ok;
maybe_print_checkouts_warnings(App, #{has_checkouts := true}) ->
AppDir = rebar_app_info:dir(App),
Checkouts = filename:join(AppDir, "_checkouts"),
rebar_log:log(warn, "~p directory found; this might interfere with publishing", [Checkouts]).
maybe_prompt(#{yes := true}, _Message) ->
proceed;
maybe_prompt(_Args, Message) ->
case rebar3_hex_io:ask(Message, boolean, "Y") of
true ->
proceed;
_ ->
abort
end.
maybe_warn_about_doc_config(State, Repo) ->
case rebar3_hex_build:doc_opts(State, Repo) of
undefined ->
Warning = "No doc provider configuration was found, therefore docs can not be published.~n~n"
"You may resolve this by adding a doc provider property to your projects hex config.~n~n"
"To configure rebar3_ex_doc as your doc provider (recommended but requires >= OTP 24) ~n~n"
"Add the following configuration to your rebar.config : ~n~n"
"{project_plugins, [rebar3_ex_doc]}.~n~n"
"{hex, [{doc, ex_doc}]}.~n~n"
"~s~n"
"~10.s~s~n"
"~10.s~s~n"
"~10.s~s~n~n"
"Alternatively if using OTP < 24 or you prefer edoc, add the following to your rebar.config :~n~n"
"{hex, [{doc, edoc}]}.~n~n"
"If you always intend to publish packages without docs and wish to silence this warning "
"you should use : ~n~nrebar3 hex publish package~n~n"
"Note : In version 7.1 this warning will be treated as an error~n",
rebar_api:warn(Warning, [
"{ex_doc, [",
" ",
"{source_url, <<\"https://github.com/namespace/your_app\">>},",
" ",
"{extras, [<<\"README.md\">>, <<\"LICENSE.md\">>]},",
" ",
"{main, <<\"readme\">>}]}."
]
);
_ ->
ok
end.
hex_opts(Opts) ->
lists:filter(fun({replace, _}) -> true;
({_,_}) -> false
end,
maps:to_list(Opts)).
%%% ===================================================================
%%% doc publishing and related functions
%%% ===================================================================
publish_docs(State, Repo, App, Args) ->
case create_docs(State, Repo, App, Args) of
#{tarball := Tar, name := Name, version := Vsn} ->
case Args of
#{dry_run := true} ->
rebar_api:info("--dry-run enabled : will not publish docs.", []),
{ok, State};
_ ->
Config = rebar3_hex_config:get_hex_config(?MODULE, Repo, write),
case rebar3_hex_client:publish_docs(Config, Name, Vsn, Tar) of
{ok, _} ->
rebar_api:info("Published docs for ~ts ~ts", [Name, Vsn]),
{ok, State};
Reason ->
?RAISE({publish_docs, Name, Vsn, Reason})
end
end;
ok ->
{ok, State}
end.
create_docs(State, Repo, App, Args) ->
case rebar3_hex_build:create_docs(State, Repo, App, Args) of
{ok, Docs} ->
Docs;
{error, no_doc_config} ->
rebar_api:warn("No documentation provider was configured, docs will not be generated.", []);
Err ->
?RAISE({create_docs, Err})
end.
%%% ===================================================================
%%% package and doc reversion functions
%%% ===================================================================
revert_package(State, Repo, AppName, Vsn) ->
BinAppName = rebar_utils:to_binary(AppName),
BinVsn = rebar_utils:to_binary(Vsn),
assert_valid_version_arg(BinVsn),
HexConfig = rebar3_hex_config:get_hex_config(?MODULE, Repo, write),
case rebar3_hex_client:delete_release(HexConfig, BinAppName, BinVsn) of
{ok, _} ->
rebar_api:info("Successfully deleted package ~ts ~ts", [AppName, Vsn]),
Prompt = io_lib:format("Also delete tag v~ts?", [Vsn]),
case rebar3_hex_io:ask(Prompt, boolean, "N") of
true ->
rebar_utils:sh(io_lib:format("git tag -d v~ts", [Vsn]), []);
_ ->
{ok, State}
end;
Reason ->
?RAISE({revert_package, BinAppName, BinVsn, Reason})
end.
revert_docs(State, Repo, AppName, Vsn) ->
BinAppName = rebar_utils:to_binary(AppName),
BinVsn = rebar_utils:to_binary(Vsn),
assert_valid_version_arg(BinVsn),
Config = rebar3_hex_config:get_hex_config(?MODULE, Repo, write),
case rebar3_hex_client:delete_docs(Config, BinAppName, BinVsn) of
{ok, _} ->
rebar_api:info("Successfully deleted docs for ~ts ~ts", [AppName, Vsn]),
{ok, State};
Reason ->
?RAISE({revert_docs, BinAppName, BinVsn, Reason})
end.
%% ===================================================================
%% General purpose helpers
%% ===================================================================
assert_valid_app(State, App) ->
Name = rebar_app_info:name(App),
Version = rebar_app_info:original_vsn(App),
ResolvedVersion = rebar_utils:vcs_vsn(App, Version, State),
{application, _, AppDetails} = rebar3_hex_file:update_app_src(App, ResolvedVersion),
Deps = rebar_state:get(State, {locks, default}, []),
AppData = #{name => Name, version => ResolvedVersion, details => AppDetails, deps => Deps},
case rebar3_hex_app:validate(AppData) of
ok ->
{ok, State};
{error, #{warnings := Warnings, errors := Errors}} ->
lists:foreach(fun(W) -> rebar_log:log(warn, format_error(W), []) end, Warnings),
case Errors of
[] ->
{ok, State};
Errs ->
?RAISE({validation_errors, Errs})
end
end.
assert_valid_version_arg(Vsn) ->
case verl:parse(Vsn) of
{ok, _} ->
ok;
_ ->
?RAISE({invalid_semver_arg, Vsn})
end.
errors_to_string(Value) when is_binary(Value) ->
Value;
errors_to_string(Map) when is_map(Map) ->
errors_to_string(maps:to_list(Map));
errors_to_string({<<"inserted_at">>, E}) ->
lists:flatten(io_lib:format("Inserted At: ~s~n", [E]));
errors_to_string({<<"requirements">>, Rs}) ->
lists:flatten(["Requirements could not be computed\n",
[io_lib:format("~s\n~20.20c\n~s\n",[P,$-, R]) || {P, R} <- maps:to_list(Rs)]]);
errors_to_string({Key, Value}) ->
io_lib:format("~s: ~s", [Key, errors_to_string(Value)]);
errors_to_string(Errors) when is_list(Errors) ->
lists:flatten([io_lib:format("~s", [errors_to_string(Values)]) || Values <- Errors]).
help(app) ->
"Specifies the app to use with the publish command, currently only utilized for publish and revert operations"
"Note that the app switch and value only have to be provided if you are publishing within an umbrella.";
help(dry_run) ->
"Generates docs (if configured) but does not publish the docs. Useful for inspecting docs before publishing.";
help(revert) ->
"Revert given version, if the last version is reverted the package is removed";
help(replace) ->
"Allows overwriting an existing package version if it exists. Private "
"packages can always be overwritten, publicpackages can only be "
"overwritten within one hour after they were initially published.";
help(yes) ->
"Publishes the package without any confirmation prompts".
maybe_warn_about_single_app_args(#{args := #{app := AppName}, apps := [_]}) when AppName =/= undefined ->
rebar_api:error("--app switch has no effect in single app projects", []);
maybe_warn_about_single_app_args(_) -> ok.
support() ->
"Publishes a new version of a package with options to revert and replace existing packages~n~n"
"Supported command combinations:~n~n"
" rebar3 hex publish~n~n"
" rebar3 hex publish package~n~n"
" rebar3 hex publish --yes~n~n"
" rebar3 hex publish package~n~n"
" rebar3 hex publish docs~n~n"
" rebar3 hex publish --dry-run~n~n"
" rebar3 hex publish --repo ~n~n"
" rebar3 hex publish --repo --yes~n~n"
" rebar3 hex publish --revert ~n~n"
" rebar3 hex publish --revert --yes~n~n"
" rebar3 hex publish --revert --app ~n~n"
" rebar3 hex publish --revert --app --yes~n~n"
" rebar3 hex publish --replace~n~n"
" rebar3 hex publish --replace --yes~n~n"
"Argument descriptions:~n~n"
" - a valid repository, only required when multiple repositories are configured~n~n"
" - a valid version string, currently only utilized with --revert switch~n~n".