Добавлен эндпойнт /v1/admin/reviews #20
This commit is contained in:
@@ -5,7 +5,7 @@
|
|||||||
update/2, delete/1, hide/2, unhide/2]).
|
update/2, delete/1, hide/2, unhide/2]).
|
||||||
-export([get_average_rating/2, has_user_reviewed/3]).
|
-export([get_average_rating/2, has_user_reviewed/3]).
|
||||||
-export([generate_id/0]).
|
-export([generate_id/0]).
|
||||||
-export([count_reviews/0]).
|
-export([count_reviews/0, list_all/0]).
|
||||||
|
|
||||||
%% Создание отзыва
|
%% Создание отзыва
|
||||||
create(UserId, TargetType, TargetId, Rating, Comment) ->
|
create(UserId, TargetType, TargetId, Rating, Comment) ->
|
||||||
@@ -115,6 +115,8 @@ has_user_reviewed(UserId, TargetType, TargetId) ->
|
|||||||
|
|
||||||
count_reviews() -> mnesia:table_info(review, size).
|
count_reviews() -> mnesia:table_info(review, size).
|
||||||
|
|
||||||
|
list_all() -> mnesia:dirty_match_object(#review{_ = '_'}).
|
||||||
|
|
||||||
%% Внутренние функции
|
%% Внутренние функции
|
||||||
generate_id() ->
|
generate_id() ->
|
||||||
base64:encode(crypto:strong_rand_bytes(16), #{mode => urlsafe, padding => false}).
|
base64:encode(crypto:strong_rand_bytes(16), #{mode => urlsafe, padding => false}).
|
||||||
|
|||||||
@@ -111,7 +111,8 @@ start_admin_http() ->
|
|||||||
{"/v1/admin/reports", admin_handler_reports, []},
|
{"/v1/admin/reports", admin_handler_reports, []},
|
||||||
{"/v1/admin/reports/:id", admin_handler_report_by_id, []},
|
{"/v1/admin/reports/:id", admin_handler_report_by_id, []},
|
||||||
% ================== ОТЗЫВЫ ==================
|
% ================== ОТЗЫВЫ ==================
|
||||||
{"/v1/admin/reviews/:id", admin_handler_reviews, []},
|
{"/v1/admin/reviews", admin_handler_reviews, []},
|
||||||
|
{"/v1/admin/reviews/:id", admin_handler_reviews_by_id, []},
|
||||||
% ================== БАН-СЛОВА ==================
|
% ================== БАН-СЛОВА ==================
|
||||||
{"/v1/admin/banned-words", admin_handler_banned_words, []},
|
{"/v1/admin/banned-words", admin_handler_banned_words, []},
|
||||||
{"/v1/admin/banned-words/:word", admin_handler_banned_words, []},
|
{"/v1/admin/banned-words/:word", admin_handler_banned_words, []},
|
||||||
|
|||||||
@@ -1,64 +1,72 @@
|
|||||||
-module(admin_handler_reviews).
|
-module(admin_handler_reviews).
|
||||||
-behaviour(cowboy_handler).
|
-behaviour(cowboy_handler).
|
||||||
-export([init/2]).
|
|
||||||
|
|
||||||
-include("records.hrl").
|
-include("records.hrl").
|
||||||
|
|
||||||
|
-export([init/2]).
|
||||||
|
|
||||||
init(Req, _Opts) ->
|
init(Req, _Opts) ->
|
||||||
case cowboy_req:method(Req) of
|
case cowboy_req:method(Req) of
|
||||||
<<"GET">> -> get_review(Req);
|
<<"GET">> -> list_reviews(Req);
|
||||||
<<"PUT">> -> update_review(Req);
|
<<"PATCH">> -> bulk_update_reviews(Req);
|
||||||
_ -> send_error(Req, 405, <<"Method not allowed">>)
|
_ -> send_error(Req, 405, <<"Method not allowed">>)
|
||||||
end.
|
end.
|
||||||
|
|
||||||
get_review(Req) ->
|
list_reviews(Req) ->
|
||||||
case handler_auth:authenticate(Req) of
|
case auth_admin(Req) of
|
||||||
{ok, AdminId, Req1} ->
|
{ok, _AdminId, Req1} ->
|
||||||
case admin_utils:is_admin(AdminId) of
|
Filters = parse_filters(Req1),
|
||||||
true ->
|
Reviews = logic_review:list_admin_reviews(Filters),
|
||||||
ReviewId = cowboy_req:binding(id, Req1),
|
Json = [review_to_json(R) || R <- Reviews],
|
||||||
case core_review:get_by_id(ReviewId) of
|
send_json(Req1, 200, Json);
|
||||||
{ok, Review} ->
|
{error, Code, Msg, Req1} ->
|
||||||
send_json(Req1, 200, review_to_json(Review));
|
send_error(Req1, Code, Msg)
|
||||||
{error, not_found} ->
|
|
||||||
send_error(Req1, 404, <<"Review not found">>)
|
|
||||||
end;
|
|
||||||
false ->
|
|
||||||
send_error(Req1, 403, <<"Admin access required">>)
|
|
||||||
end;
|
|
||||||
{error, Code, Message, Req1} ->
|
|
||||||
send_error(Req1, Code, Message)
|
|
||||||
end.
|
end.
|
||||||
|
|
||||||
update_review(Req) ->
|
bulk_update_reviews(Req) ->
|
||||||
|
case auth_admin(Req) of
|
||||||
|
{ok, _AdminId, Req1} ->
|
||||||
|
try
|
||||||
|
{ok, Body, Req2} = cowboy_req:read_body(Req1),
|
||||||
|
Operations = jsx:decode(Body, [return_maps]),
|
||||||
|
case logic_review:bulk_update_status(Operations) of
|
||||||
|
{ok, Count} ->
|
||||||
|
send_json(Req2, 200, #{updated_count => Count});
|
||||||
|
{error, Reason} ->
|
||||||
|
send_error(Req2, 400, Reason)
|
||||||
|
end
|
||||||
|
catch
|
||||||
|
_:_ -> send_error(Req1, 400, <<"Invalid JSON body">>)
|
||||||
|
end;
|
||||||
|
{error, Code, Msg, Req1} ->
|
||||||
|
send_error(Req1, Code, Msg)
|
||||||
|
end.
|
||||||
|
|
||||||
|
auth_admin(Req) ->
|
||||||
case handler_auth:authenticate(Req) of
|
case handler_auth:authenticate(Req) of
|
||||||
{ok, AdminId, Req1} ->
|
{ok, AdminId, Req1} ->
|
||||||
case admin_utils:is_admin(AdminId) of
|
case admin_utils:is_admin(AdminId) of
|
||||||
true ->
|
true -> {ok, AdminId, Req1};
|
||||||
ReviewId = cowboy_req:binding(id, Req1),
|
false -> {error, 403, <<"Admin access required">>, Req1}
|
||||||
{ok, Body, Req2} = cowboy_req:read_body(Req1),
|
|
||||||
try jsx:decode(Body, [return_maps]) of
|
|
||||||
#{<<"status">> := NewStatus} ->
|
|
||||||
case core_review:update_status(ReviewId, NewStatus) of
|
|
||||||
{ok, Review} ->
|
|
||||||
send_json(Req2, 200, review_to_json(Review));
|
|
||||||
{error, not_found} ->
|
|
||||||
send_error(Req2, 404, <<"Review not found">>);
|
|
||||||
{error, _} ->
|
|
||||||
send_error(Req2, 500, <<"Internal server error">>)
|
|
||||||
end;
|
end;
|
||||||
_ ->
|
{error, Code, Msg, Req1} ->
|
||||||
send_error(Req2, 400, <<"Missing status field">>)
|
{error, Code, Msg, Req1}
|
||||||
catch
|
|
||||||
_:_ -> send_error(Req2, 400, <<"Invalid JSON">>)
|
|
||||||
end;
|
|
||||||
false ->
|
|
||||||
send_error(Req1, 403, <<"Admin access required">>)
|
|
||||||
end;
|
|
||||||
{error, Code, Message, Req1} ->
|
|
||||||
send_error(Req1, Code, Message)
|
|
||||||
end.
|
end.
|
||||||
|
|
||||||
|
%% Извлечение параметров фильтрации из query string.
|
||||||
|
%% Например: ?target_type=event&target_id=...&user_id=...
|
||||||
|
parse_filters(Req) ->
|
||||||
|
Qs = cowboy_req:parse_qs(Req),
|
||||||
|
lists:filtermap(
|
||||||
|
fun
|
||||||
|
({<<"target_type">>, Val}) -> {true, {target_type, Val}};
|
||||||
|
({<<"target_id">>, Val}) -> {true, {target_id, Val}};
|
||||||
|
({<<"user_id">>, Val}) -> {true, {user_id, Val}};
|
||||||
|
(_) -> false
|
||||||
|
end,
|
||||||
|
Qs
|
||||||
|
).
|
||||||
|
|
||||||
review_to_json(R) ->
|
review_to_json(R) ->
|
||||||
#{
|
#{
|
||||||
id => R#review.id,
|
id => R#review.id,
|
||||||
|
|||||||
93
src/handlers/admin/admin_handler_reviews_by_id.erl
Normal file
93
src/handlers/admin/admin_handler_reviews_by_id.erl
Normal file
@@ -0,0 +1,93 @@
|
|||||||
|
-module(admin_handler_reviews_by_id).
|
||||||
|
-behaviour(cowboy_handler).
|
||||||
|
-export([init/2]).
|
||||||
|
|
||||||
|
-include("records.hrl").
|
||||||
|
|
||||||
|
init(Req, _Opts) ->
|
||||||
|
case cowboy_req:method(Req) of
|
||||||
|
<<"GET">> -> get_review(Req);
|
||||||
|
<<"PUT">> -> update_review(Req);
|
||||||
|
_ -> send_error(Req, 405, <<"Method not allowed">>)
|
||||||
|
end.
|
||||||
|
|
||||||
|
get_review(Req) ->
|
||||||
|
case handler_auth:authenticate(Req) of
|
||||||
|
{ok, AdminId, Req1} ->
|
||||||
|
case admin_utils:is_admin(AdminId) of
|
||||||
|
true ->
|
||||||
|
ReviewId = cowboy_req:binding(id, Req1),
|
||||||
|
case core_review:get_by_id(ReviewId) of
|
||||||
|
{ok, Review} ->
|
||||||
|
send_json(Req1, 200, review_to_json(Review));
|
||||||
|
{error, not_found} ->
|
||||||
|
send_error(Req1, 404, <<"Review not found">>)
|
||||||
|
end;
|
||||||
|
false ->
|
||||||
|
send_error(Req1, 403, <<"Admin access required">>)
|
||||||
|
end;
|
||||||
|
{error, Code, Message, Req1} ->
|
||||||
|
send_error(Req1, Code, Message)
|
||||||
|
end.
|
||||||
|
|
||||||
|
update_review(Req) ->
|
||||||
|
case handler_auth:authenticate(Req) of
|
||||||
|
{ok, AdminId, Req1} ->
|
||||||
|
case admin_utils:is_admin(AdminId) of
|
||||||
|
true ->
|
||||||
|
ReviewId = cowboy_req:binding(id, Req1),
|
||||||
|
{ok, Body, Req2} = cowboy_req:read_body(Req1),
|
||||||
|
try jsx:decode(Body, [return_maps]) of
|
||||||
|
#{<<"status">> := NewStatus} ->
|
||||||
|
case core_review:update_status(ReviewId, NewStatus) of
|
||||||
|
{ok, Review} ->
|
||||||
|
send_json(Req2, 200, review_to_json(Review));
|
||||||
|
{error, not_found} ->
|
||||||
|
send_error(Req2, 404, <<"Review not found">>);
|
||||||
|
{error, _} ->
|
||||||
|
send_error(Req2, 500, <<"Internal server error">>)
|
||||||
|
end;
|
||||||
|
_ ->
|
||||||
|
send_error(Req2, 400, <<"Missing status field">>)
|
||||||
|
catch
|
||||||
|
_:_ -> send_error(Req2, 400, <<"Invalid JSON">>)
|
||||||
|
end;
|
||||||
|
false ->
|
||||||
|
send_error(Req1, 403, <<"Admin access required">>)
|
||||||
|
end;
|
||||||
|
{error, Code, Message, Req1} ->
|
||||||
|
send_error(Req1, Code, Message)
|
||||||
|
end.
|
||||||
|
|
||||||
|
review_to_json(R) ->
|
||||||
|
#{
|
||||||
|
id => R#review.id,
|
||||||
|
user_id => R#review.user_id,
|
||||||
|
target_type => R#review.target_type,
|
||||||
|
target_id => R#review.target_id,
|
||||||
|
rating => R#review.rating,
|
||||||
|
comment => R#review.comment,
|
||||||
|
status => R#review.status,
|
||||||
|
created_at => datetime_to_iso8601(R#review.created_at),
|
||||||
|
updated_at => datetime_to_iso8601(R#review.updated_at)
|
||||||
|
}.
|
||||||
|
|
||||||
|
datetime_to_iso8601({{Year, Month, Day}, {Hour, Minute, Second}}) ->
|
||||||
|
iolist_to_binary(io_lib:format("~4..0B-~2..0B-~2..0BT~2..0B:~2..0B:~2..0BZ",
|
||||||
|
[Year, Month, Day, Hour, Minute, Second]));
|
||||||
|
datetime_to_iso8601(undefined) -> undefined.
|
||||||
|
|
||||||
|
send_json(Req, Status, Data) ->
|
||||||
|
Headers = #{
|
||||||
|
<<"content-type">> => <<"application/json">>,
|
||||||
|
<<"access-control-allow-origin">> => <<"*">>,
|
||||||
|
<<"access-control-expose-headers">> => <<"Content-Range">>
|
||||||
|
},
|
||||||
|
Body = jsx:encode(Data),
|
||||||
|
cowboy_req:reply(Status, Headers, Body, Req),
|
||||||
|
{ok, Body, []}.
|
||||||
|
|
||||||
|
send_error(Req, Status, Message) ->
|
||||||
|
Body = jsx:encode(#{error => Message}),
|
||||||
|
cowboy_req:reply(Status, #{<<"content-type">> => <<"application/json">>}, Body, Req),
|
||||||
|
{ok, Body, []}.
|
||||||
@@ -2,8 +2,9 @@
|
|||||||
-include("records.hrl").
|
-include("records.hrl").
|
||||||
|
|
||||||
-export([create_review/5, get_review/2, list_reviews/3, list_user_reviews/1,
|
-export([create_review/5, get_review/2, list_reviews/3, list_user_reviews/1,
|
||||||
update_review/3, delete_review/2, hide_review/2, unhide_review/2]).
|
update_review/3, delete_review/2, hide_review/2, hide_review/3, unhide_review/2, unhide_review/3]).
|
||||||
-export([can_review/3, update_target_rating/2, can_moderate_review/2]).
|
-export([can_review/3, update_target_rating/2, can_moderate_review/2]).
|
||||||
|
-export([list_admin_reviews/1, bulk_update_status/1]).
|
||||||
|
|
||||||
%% Создание отзыва
|
%% Создание отзыва
|
||||||
create_review(UserId, TargetType, TargetId, Rating, Comment) ->
|
create_review(UserId, TargetType, TargetId, Rating, Comment) ->
|
||||||
@@ -114,9 +115,12 @@ delete_review(UserId, ReviewId) ->
|
|||||||
end.
|
end.
|
||||||
|
|
||||||
hide_review(UserId, ReviewId) ->
|
hide_review(UserId, ReviewId) ->
|
||||||
|
hide_review(UserId, ReviewId, <<>>).
|
||||||
|
|
||||||
|
hide_review(UserId, ReviewId, Reason) ->
|
||||||
case can_moderate_review(UserId, ReviewId) of
|
case can_moderate_review(UserId, ReviewId) of
|
||||||
true ->
|
true ->
|
||||||
case core_review:hide(ReviewId) of
|
case core_review:hide(ReviewId, Reason) of
|
||||||
{ok, Review} ->
|
{ok, Review} ->
|
||||||
update_target_rating(Review#review.target_type, Review#review.target_id),
|
update_target_rating(Review#review.target_type, Review#review.target_id),
|
||||||
{ok, Review};
|
{ok, Review};
|
||||||
@@ -128,9 +132,11 @@ hide_review(UserId, ReviewId) ->
|
|||||||
end.
|
end.
|
||||||
|
|
||||||
unhide_review(UserId, ReviewId) ->
|
unhide_review(UserId, ReviewId) ->
|
||||||
|
unhide_review(UserId, ReviewId, <<>>).
|
||||||
|
unhide_review(UserId, ReviewId, Reason) ->
|
||||||
case can_moderate_review(UserId, ReviewId) of
|
case can_moderate_review(UserId, ReviewId) of
|
||||||
true ->
|
true ->
|
||||||
case core_review:unhide(ReviewId) of
|
case core_review:unhide(ReviewId, Reason) of
|
||||||
{ok, Review} ->
|
{ok, Review} ->
|
||||||
update_target_rating(Review#review.target_type, Review#review.target_id),
|
update_target_rating(Review#review.target_type, Review#review.target_id),
|
||||||
{ok, Review};
|
{ok, Review};
|
||||||
@@ -187,6 +193,64 @@ can_moderate_review(UserId, ReviewId) ->
|
|||||||
end
|
end
|
||||||
end.
|
end.
|
||||||
|
|
||||||
|
%%%-------------------------------------------------------------------
|
||||||
|
%%% @doc Получить список всех отзывов (административный режим).
|
||||||
|
%%% Фильтры: [{target_type, Type}, {target_id, Id}, {user_id, UserId}]
|
||||||
|
%%% Все фильтры опциональны.
|
||||||
|
%%% @end
|
||||||
|
%%%-------------------------------------------------------------------
|
||||||
|
list_admin_reviews(Filters) ->
|
||||||
|
AllReviews = core_review:list_all(),
|
||||||
|
apply_filters(AllReviews, Filters).
|
||||||
|
|
||||||
|
%% Вспомогательная функция: фильтрация списка по proplist
|
||||||
|
apply_filters(Reviews, []) ->
|
||||||
|
Reviews;
|
||||||
|
apply_filters(Reviews, [{target_type, Type} | Rest]) ->
|
||||||
|
apply_filters(
|
||||||
|
[R || R <- Reviews, R#review.target_type =:= Type],
|
||||||
|
Rest
|
||||||
|
);
|
||||||
|
apply_filters(Reviews, [{target_id, Id} | Rest]) ->
|
||||||
|
apply_filters(
|
||||||
|
[R || R <- Reviews, R#review.target_id =:= Id],
|
||||||
|
Rest
|
||||||
|
);
|
||||||
|
apply_filters(Reviews, [{user_id, UserId} | Rest]) ->
|
||||||
|
apply_filters(
|
||||||
|
[R || R <- Reviews, R#review.user_id =:= UserId],
|
||||||
|
Rest
|
||||||
|
);
|
||||||
|
apply_filters(Reviews, [_ | Rest]) ->
|
||||||
|
apply_filters(Reviews, Rest). % Игнорируем неизвестные фильтры
|
||||||
|
|
||||||
|
%%%-------------------------------------------------------------------
|
||||||
|
%%% @doc Массово обновить статусы отзывов.
|
||||||
|
%%% Operations: [#{id => ReviewId, status => visible | hidden}, ...]
|
||||||
|
%%% Все изменения выполняются в одной Mnesia-транзакции.
|
||||||
|
%%% @end
|
||||||
|
%%%-------------------------------------------------------------------
|
||||||
|
bulk_update_status(Operations) when is_list(Operations) ->
|
||||||
|
Fun = fun() ->
|
||||||
|
lists:foreach(fun do_update_status/1, Operations)
|
||||||
|
end,
|
||||||
|
case mnesia:transaction(Fun) of
|
||||||
|
{atomic, ok} ->
|
||||||
|
{ok, length(Operations)};
|
||||||
|
{aborted, Reason} ->
|
||||||
|
{error, Reason}
|
||||||
|
end.
|
||||||
|
|
||||||
|
do_update_status(#{<<"id">> := Id, <<"status">> := NewStatus}) ->
|
||||||
|
case core_review:get_by_id(Id) of
|
||||||
|
{ok, Review} ->
|
||||||
|
IdReview = Review#review{id = Id},
|
||||||
|
UpdatedReview = Review#review{status = NewStatus},
|
||||||
|
core_review:update(IdReview, UpdatedReview);
|
||||||
|
not_found ->
|
||||||
|
mnesia:abort(<<"review_not_found">>)
|
||||||
|
end.
|
||||||
|
|
||||||
%% Внутренние функции
|
%% Внутренние функции
|
||||||
target_exists(event, EventId) ->
|
target_exists(event, EventId) ->
|
||||||
case core_event:get_by_id(EventId) of
|
case core_event:get_by_id(EventId) of
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
-module(api_admin_tests).
|
-module(api_admin_tests).
|
||||||
-export([test/0]).
|
-export([test/0]).
|
||||||
|
|
||||||
%% Учётные данные по умолчанию (используются, если словарь процесса пуст)
|
%% Учётные данные по умолчанию
|
||||||
-define(FALLBACK_ADMIN_SUPER_EMAIL, <<"superadmin@eventhub.local">>).
|
-define(FALLBACK_ADMIN_SUPER_EMAIL, <<"superadmin@eventhub.local">>).
|
||||||
-define(FALLBACK_ADMIN_SUPER_PASSWORD, <<"123456">>).
|
-define(FALLBACK_ADMIN_SUPER_PASSWORD, <<"123456">>).
|
||||||
-define(FALLBACK_ADMIN_MODER_EMAIL, <<"moderator@eventhub.local">>).
|
-define(FALLBACK_ADMIN_MODER_EMAIL, <<"moderator@eventhub.local">>).
|
||||||
@@ -14,7 +14,7 @@ test() ->
|
|||||||
AdminURL = api_test_runner:get_admin_url(),
|
AdminURL = api_test_runner:get_admin_url(),
|
||||||
UserURL = api_test_runner:get_base_url(),
|
UserURL = api_test_runner:get_base_url(),
|
||||||
|
|
||||||
% Получаем токен суперадмина (уже проинициализирован в api_test_runner)
|
% Получаем токен суперадмина
|
||||||
AdminToken = api_test_runner:get_admin_token(),
|
AdminToken = api_test_runner:get_admin_token(),
|
||||||
|
|
||||||
%% TEST 1: Admin healthcheck (public)
|
%% TEST 1: Admin healthcheck (public)
|
||||||
@@ -24,45 +24,39 @@ test() ->
|
|||||||
|
|
||||||
%% TEST 2: Admin login (дополнительная проверка)
|
%% TEST 2: Admin login (дополнительная проверка)
|
||||||
ct:pal(" TEST 2: Admin login (attempt)... "),
|
ct:pal(" TEST 2: Admin login (attempt)... "),
|
||||||
% Теперь используем суперадмина, который гарантированно создан
|
LoginBody = jsx:encode(#{
|
||||||
LoginBody = jsx:encode(#{<<"email">> => ?FALLBACK_ADMIN_SUPER_EMAIL, <<"password">> => ?FALLBACK_ADMIN_SUPER_PASSWORD}),
|
<<"email">> => ?FALLBACK_ADMIN_SUPER_EMAIL,
|
||||||
|
<<"password">> => ?FALLBACK_ADMIN_SUPER_PASSWORD
|
||||||
|
}),
|
||||||
case httpc:request(post, {AdminURL ++ "/v1/admin/login", [], "application/json", LoginBody}, [], []) of
|
case httpc:request(post, {AdminURL ++ "/v1/admin/login", [], "application/json", LoginBody}, [], []) of
|
||||||
{ok, {{_, 200, _}, _, _}} ->
|
{ok, {{_, 200, _}, _, _}} -> ct:pal("OK (logged in)~n");
|
||||||
ct:pal("OK (logged in)~n");
|
_ -> ct:pal("SKIPPED (credentials not found, using runner token)~n")
|
||||||
_ ->
|
|
||||||
ct:pal("SKIPPED (credentials not found, using runner token)~n")
|
|
||||||
end,
|
end,
|
||||||
|
|
||||||
%% TEST 3: Admin stats (superadmin)
|
%% TEST 3: Admin stats (superadmin)
|
||||||
ct:pal(" TEST 3: Admin stats (superadmin)... "),
|
ct:pal(" TEST 3: Admin stats (superadmin)... "),
|
||||||
{ok, {{_, 200, _}, _, StatsResp1}} = httpc:request(get,
|
{ok, {{_, 200, _}, _, StatsResp1}} = httpc:request(get, {AdminURL ++ "/v1/admin/stats", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/stats", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
|
||||||
Stats1 = jsx:decode(list_to_binary(StatsResp1), [return_maps]),
|
Stats1 = jsx:decode(list_to_binary(StatsResp1), [return_maps]),
|
||||||
ct:pal(" OK (keys: ~p)~n", [maps:keys(Stats1)]),
|
ct:pal(" OK (keys: ~p)~n", [maps:keys(Stats1)]),
|
||||||
|
|
||||||
%% TEST 4: List users
|
%% TEST 4: List users
|
||||||
ct:pal(" TEST 4: List users... "),
|
ct:pal(" TEST 4: List users... "),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(get,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(get, {AdminURL ++ "/v1/admin/users", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/users", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 5: Get user by ID
|
%% TEST 5: Get user by ID
|
||||||
ct:pal(" TEST 5: Get user by ID... "),
|
ct:pal(" TEST 5: Get user by ID... "),
|
||||||
UserId = api_test_runner:get_user_id(),
|
UserId = api_test_runner:get_user_id(),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(get,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(get, {AdminURL ++ "/v1/admin/users/" ++ binary_to_list(UserId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/users/" ++ binary_to_list(UserId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 6: List reports
|
%% TEST 6: List reports
|
||||||
ct:pal(" TEST 6: List reports... "),
|
ct:pal(" TEST 6: List reports... "),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(get,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(get, {AdminURL ++ "/v1/admin/reports", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/reports", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% ── TEST 7: Full moderation flow (create event, report, resolve) ──
|
%% ── TEST 7: Full moderation flow (create event, report, resolve) ──
|
||||||
ct:pal(" TEST 7: Moderation flow... "),
|
ct:pal(" TEST 7: Moderation flow... "),
|
||||||
|
|
||||||
% Создаём календарь и событие от имени пользователя
|
|
||||||
UserToken = api_test_runner:get_user_token(),
|
UserToken = api_test_runner:get_user_token(),
|
||||||
CalId = api_test_runner:create_calendar(UserToken, #{title => <<"ModerationTest">>}),
|
CalId = api_test_runner:create_calendar(UserToken, #{title => <<"ModerationTest">>}),
|
||||||
EventId = api_test_runner:create_event(UserToken, CalId, #{
|
EventId = api_test_runner:create_event(UserToken, CalId, #{
|
||||||
@@ -70,133 +64,200 @@ test() ->
|
|||||||
start_time => api_SUITE:future_date(),
|
start_time => api_SUITE:future_date(),
|
||||||
duration => 60
|
duration => 60
|
||||||
}),
|
}),
|
||||||
|
|
||||||
% Подаём жалобу на это событие
|
% Подаём жалобу на это событие
|
||||||
CreateBody = jsx:encode(#{
|
CreateBody = jsx:encode(#{
|
||||||
<<"target_type">> => <<"event">>,
|
<<"target_type">> => <<"event">>,
|
||||||
<<"target_id">> => EventId,
|
<<"target_id">> => EventId,
|
||||||
<<"reason">> => <<"Inappropriate content">>
|
<<"reason">> => <<"Inappropriate content">>
|
||||||
}),
|
}),
|
||||||
{ok, {{_, 201, _}, _, CreateResp}} = httpc:request(post,
|
{ok, {{_, 201, _}, _, CreateResp}} = httpc:request(post, {UserURL ++ "/v1/reports", [{"Authorization", "Bearer " ++ binary_to_list(UserToken)}], "application/json", CreateBody}, [], []),
|
||||||
{UserURL ++ "/v1/reports",
|
|
||||||
[{"Authorization", "Bearer " ++ binary_to_list(UserToken)}],
|
|
||||||
"application/json", CreateBody}, [], []),
|
|
||||||
#{<<"id">> := ReportId} = jsx:decode(list_to_binary(CreateResp), [return_maps]),
|
#{<<"id">> := ReportId} = jsx:decode(list_to_binary(CreateResp), [return_maps]),
|
||||||
|
|
||||||
% Администратор изменяет статус жалобы
|
% Администратор изменяет статус жалобы
|
||||||
EditBody = jsx:encode(#{
|
EditBody = jsx:encode(#{
|
||||||
<<"status">> => <<"reviewed">>,
|
<<"status">> => <<"reviewed">>,
|
||||||
<<"reason">> => <<"Issue resolved">>
|
<<"reason">> => <<"Issue resolved">>
|
||||||
}),
|
}),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(put,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(put, {AdminURL ++ "/v1/admin/reports/" ++ binary_to_list(ReportId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", EditBody}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/reports/" ++ binary_to_list(ReportId),
|
|
||||||
[{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}],
|
|
||||||
"application/json", EditBody}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 8: List banned words
|
%% TEST 8: List banned words
|
||||||
ct:pal(" TEST 8: List banned words... "),
|
ct:pal(" TEST 8: List banned words... "),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(get,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(get, {AdminURL ++ "/v1/admin/banned-words", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/banned-words", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 9: Add banned word
|
%% TEST 9: Add banned word
|
||||||
ct:pal(" TEST 9: Add banned word... "),
|
ct:pal(" TEST 9: Add banned word... "),
|
||||||
BannedWordBody = jsx:encode(#{<<"word">> => <<"test_banned_word">>}),
|
BannedWordBody = jsx:encode(#{<<"word">> => <<"test_banned_word">>}),
|
||||||
{ok, {{_, 201, _}, _, _}} = httpc:request(post,
|
{ok, {{_, 201, _}, _, _}} = httpc:request(post, {AdminURL ++ "/v1/admin/banned-words", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", BannedWordBody}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/banned-words", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", BannedWordBody}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 10: Delete banned word
|
%% TEST 10: Delete banned word
|
||||||
ct:pal(" TEST 10: Delete banned word... "),
|
ct:pal(" TEST 10: Delete banned word... "),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(delete,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(delete, {AdminURL ++ "/v1/admin/banned-words/test_banned_word", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/banned-words/test_banned_word", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 11: List tickets
|
%% TEST 11: List tickets
|
||||||
ct:pal(" TEST 11: List tickets... "),
|
ct:pal(" TEST 11: List tickets... "),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(get,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(get, {AdminURL ++ "/v1/admin/tickets", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/tickets", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 12: Create ticket
|
%% TEST 12: Create ticket
|
||||||
ct:pal(" TEST 12: Create ticket... "),
|
ct:pal(" TEST 12: Create ticket... "),
|
||||||
TicketBody = jsx:encode(#{<<"error_message">> => <<"Test error">>, <<"stacktrace">> => <<"trace">>}),
|
TicketBody = jsx:encode(#{
|
||||||
{ok, {{_, 201, _}, _, TicketResp}} = httpc:request(post,
|
<<"error_message">> => <<"Test error">>,
|
||||||
{AdminURL ++ "/v1/admin/tickets", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", TicketBody}, [], []),
|
<<"stacktrace">> => <<"trace">>
|
||||||
|
}),
|
||||||
|
{ok, {{_, 201, _}, _, TicketResp}} = httpc:request(post, {AdminURL ++ "/v1/admin/tickets", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", TicketBody}, [], []),
|
||||||
#{<<"id">> := TicketId} = jsx:decode(list_to_binary(TicketResp), [return_maps]),
|
#{<<"id">> := TicketId} = jsx:decode(list_to_binary(TicketResp), [return_maps]),
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 13: Get ticket by ID
|
%% TEST 13: Get ticket by ID
|
||||||
ct:pal(" TEST 13: Get ticket by ID... "),
|
ct:pal(" TEST 13: Get ticket by ID... "),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(get,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(get, {AdminURL ++ "/v1/admin/tickets/" ++ binary_to_list(TicketId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/tickets/" ++ binary_to_list(TicketId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 14: Update ticket
|
%% TEST 14: Update ticket
|
||||||
ct:pal(" TEST 14: Update ticket... "),
|
ct:pal(" TEST 14: Update ticket... "),
|
||||||
UpdateTicketBody = jsx:encode(#{<<"status">> => <<"closed">>}),
|
UpdateTicketBody = jsx:encode(#{<<"status">> => <<"closed">>}),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(put,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(put, {AdminURL ++ "/v1/admin/tickets/" ++ binary_to_list(TicketId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", UpdateTicketBody}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/tickets/" ++ binary_to_list(TicketId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", UpdateTicketBody}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 15: Delete ticket
|
%% TEST 15: Delete ticket
|
||||||
ct:pal(" TEST 15: Delete ticket... "),
|
ct:pal(" TEST 15: Delete ticket... "),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(delete,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(delete, {AdminURL ++ "/v1/admin/tickets/" ++ binary_to_list(TicketId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/tickets/" ++ binary_to_list(TicketId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 16: Ticket stats
|
%% TEST 16: Ticket stats
|
||||||
ct:pal(" TEST 16: Ticket stats... "),
|
ct:pal(" TEST 16: Ticket stats... "),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(get,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(get, {AdminURL ++ "/v1/admin/tickets/stats", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/tickets/stats", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 17: List subscriptions
|
%% TEST 17: List subscriptions
|
||||||
ct:pal(" TEST 17: List subscriptions... "),
|
ct:pal(" TEST 17: List subscriptions... "),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(get,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(get, {AdminURL ++ "/v1/admin/subscriptions", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/subscriptions", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 18: Create subscription
|
%% TEST 18: Create subscription
|
||||||
ct:pal(" TEST 18: Create subscription... "),
|
ct:pal(" TEST 18: Create subscription... "),
|
||||||
SubBody = jsx:encode(#{<<"user_id">> => UserId, <<"plan">> => <<"monthly">>}),
|
SubBody = jsx:encode(#{
|
||||||
{ok, {{_, 201, _}, _, SubResp}} = httpc:request(post,
|
<<"user_id">> => UserId,
|
||||||
{AdminURL ++ "/v1/admin/subscriptions", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", SubBody}, [], []),
|
<<"plan">> => <<"monthly">>
|
||||||
|
}),
|
||||||
|
{ok, {{_, 201, _}, _, SubResp}} = httpc:request(post, {AdminURL ++ "/v1/admin/subscriptions", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", SubBody}, [], []),
|
||||||
#{<<"id">> := SubId} = jsx:decode(list_to_binary(SubResp), [return_maps]),
|
#{<<"id">> := SubId} = jsx:decode(list_to_binary(SubResp), [return_maps]),
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 19: Get subscription by ID
|
%% TEST 19: Get subscription by ID
|
||||||
ct:pal(" TEST 19: Get subscription by ID... "),
|
ct:pal(" TEST 19: Get subscription by ID... "),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(get,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(get, {AdminURL ++ "/v1/admin/subscriptions/" ++ binary_to_list(SubId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/subscriptions/" ++ binary_to_list(SubId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 20: Update subscription
|
%% TEST 20: Update subscription
|
||||||
ct:pal(" TEST 20: Update subscription... "),
|
ct:pal(" TEST 20: Update subscription... "),
|
||||||
UpdateSubBody = jsx:encode(#{<<"status">> => <<"cancelled">>}),
|
UpdateSubBody = jsx:encode(#{<<"status">> => <<"cancelled">>}),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(put,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(put, {AdminURL ++ "/v1/admin/subscriptions/" ++ binary_to_list(SubId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", UpdateSubBody}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/subscriptions/" ++ binary_to_list(SubId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", UpdateSubBody}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 21: Delete subscription
|
%% TEST 21: Delete subscription
|
||||||
ct:pal(" TEST 21: Delete subscription... "),
|
ct:pal(" TEST 21: Delete subscription... "),
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(delete,
|
{ok, {{_, 200, _}, _, _}} = httpc:request(delete, {AdminURL ++ "/v1/admin/subscriptions/" ++ binary_to_list(SubId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
{AdminURL ++ "/v1/admin/subscriptions/" ++ binary_to_list(SubId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 22: Moderation - block user
|
%% TEST 22: Moderation - block user
|
||||||
ct:pal(" TEST 22: Moderation - block user... "),
|
ct:pal(" TEST 22: Moderation - block user... "),
|
||||||
ModBody = jsx:encode(#{<<"action">> => <<"block">>, <<"reason">> => <<"test">>}),
|
ModBody = jsx:encode(#{
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(put,
|
<<"action">> => <<"block">>,
|
||||||
{AdminURL ++ "/v1/admin/user/" ++ binary_to_list(UserId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", ModBody}, [], []),
|
<<"reason">> => <<"test">>
|
||||||
|
}),
|
||||||
|
{ok, {{_, 200, _}, _, _}} = httpc:request(put, {AdminURL ++ "/v1/admin/user/" ++ binary_to_list(UserId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", ModBody}, [], []),
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
%% TEST 23: Moderation - unblock user
|
%% TEST 23: Moderation - unblock user
|
||||||
ct:pal(" TEST 23: Moderation - unblock user... "),
|
ct:pal(" TEST 23: Moderation - unblock user... "),
|
||||||
UnblockBody = jsx:encode(#{<<"action">> => <<"unblock">>, <<"reason">> => <<"restore">>}),
|
UnblockBody = jsx:encode(#{
|
||||||
{ok, {{_, 200, _}, _, _}} = httpc:request(put,
|
<<"action">> => <<"unblock">>,
|
||||||
{AdminURL ++ "/v1/admin/user/" ++ binary_to_list(UserId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", UnblockBody}, [], []),
|
<<"reason">> => <<"restore">>
|
||||||
|
}),
|
||||||
|
{ok, {{_, 200, _}, _, _}} = httpc:request(put, {AdminURL ++ "/v1/admin/user/" ++ binary_to_list(UserId), [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", UnblockBody}, [], []),
|
||||||
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
|
%% ========================================================
|
||||||
|
%% Admin Reviews list tests
|
||||||
|
%% ========================================================
|
||||||
|
|
||||||
|
%% Подготовка тестовых данных для отзывов
|
||||||
|
ct:pal(" Preparing test data for reviews... "),
|
||||||
|
UserToken = api_test_runner:get_user_token(),
|
||||||
|
% Создаем календарь и событие (отдельные переменные, чтобы не перекрыть TEST 7)
|
||||||
|
RevCalId = api_test_runner:create_calendar(UserToken, #{title => <<"ReviewsTest">>}),
|
||||||
|
RevEventId = api_test_runner:create_event(UserToken, RevCalId, #{
|
||||||
|
title => <<"Event for review testing">>,
|
||||||
|
start_time => api_SUITE:future_date(),
|
||||||
|
duration => 60
|
||||||
|
}),
|
||||||
|
ct:pal("OK (calendar: ~s, event: ~s)~n", [RevCalId, RevEventId]),
|
||||||
|
|
||||||
|
ParticipantEmail = api_test_runner:unique_email(<<"rev_1">>),
|
||||||
|
ParticipantEmail2 = api_test_runner:unique_email(<<"rev_2">>),
|
||||||
|
ParticipantToken = api_test_runner:register_and_login(ParticipantEmail, <<"part123">>),
|
||||||
|
ParticipantToken2 = api_test_runner:register_and_login(ParticipantEmail2, <<"part123">>),
|
||||||
|
|
||||||
|
% Создаём и подтверждаем бронирование
|
||||||
|
BookingId = api_test_runner:extract_json(
|
||||||
|
api_test_runner:http_post("/v1/events/" ++ binary_to_list(RevEventId) ++ "/bookings", #{}, ParticipantToken), <<"id">>),
|
||||||
|
api_test_runner:http_put("/v1/bookings/" ++ binary_to_list(BookingId), #{action => <<"confirm">>}, UserToken),
|
||||||
|
|
||||||
|
Booking2Id = api_test_runner:extract_json(
|
||||||
|
api_test_runner:http_post("/v1/events/" ++ binary_to_list(RevEventId) ++ "/bookings", #{}, ParticipantToken2), <<"id">>),
|
||||||
|
api_test_runner:http_put("/v1/bookings/" ++ binary_to_list(Booking2Id), #{action => <<"confirm">>}, UserToken),
|
||||||
|
|
||||||
|
ReviewId = api_test_runner:extract_json(
|
||||||
|
api_test_runner:http_post("/v1/reviews",
|
||||||
|
#{target_type => <<"event">>, target_id => RevEventId, rating => 5, comment => <<"Great!">>},
|
||||||
|
ParticipantToken), <<"id">>),
|
||||||
|
ct:pal(" Review2Id: ~p~n", [ReviewId]),
|
||||||
|
Review2Id = api_test_runner:extract_json(
|
||||||
|
api_test_runner:http_post("/v1/reviews",
|
||||||
|
#{target_type => <<"event">>, target_id => RevEventId, rating => 5, comment => <<"Great!">>},
|
||||||
|
ParticipantToken2), <<"id">>),
|
||||||
|
ct:pal(" Review2Id: ~p~n", [Review2Id]),
|
||||||
|
|
||||||
|
%% TEST 24: List all reviews (GET /v1/admin/reviews)
|
||||||
|
ct:pal(" TEST 24: List all reviews... "),
|
||||||
|
{ok, {{_, 200, _}, _, ListReviewsResp}} = httpc:request(get, {AdminURL ++ "/v1/admin/reviews", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
|
ReviewsList = jsx:decode(list_to_binary(ListReviewsResp), [return_maps]),
|
||||||
|
true = is_list(ReviewsList),
|
||||||
|
ct:pal(" OK (count: ~p)~n", [length(ReviewsList)]),
|
||||||
|
|
||||||
|
%% TEST 25: List reviews with filters (GET /v1/admin/reviews?target_type=event&target_id=...)
|
||||||
|
ct:pal(" TEST 25: List reviews with filters... "),
|
||||||
|
FilterURL = AdminURL ++ "/v1/admin/reviews?target_type=event&target_id=" ++ binary_to_list(RevEventId),
|
||||||
|
{ok, {{_, 200, _}, _, FilteredResp}} = httpc:request(get, {FilterURL, [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}]}, [], []),
|
||||||
|
FilteredList = jsx:decode(list_to_binary(FilteredResp), [return_maps]),
|
||||||
|
ct:pal(" OK (filtered count: ~p)~n", [length(FilteredList)]),
|
||||||
|
|
||||||
|
%% TEST 26: Bulk update review statuses (PATCH /v1/admin/reviews)
|
||||||
|
ct:pal(" TEST 26: Bulk update review statuses... "),
|
||||||
|
case ReviewsList of
|
||||||
|
[FirstReview, SecondReview | _] ->
|
||||||
|
FirstId = maps:get(<<"id">>, FirstReview),
|
||||||
|
SecondId = maps:get(<<"id">>, SecondReview),
|
||||||
|
PatchBody = jsx:encode([
|
||||||
|
#{<<"id">> => FirstId, <<"status">> => <<"visible">>},
|
||||||
|
#{<<"id">> => SecondId, <<"status">> => <<"hidden">>}
|
||||||
|
]),
|
||||||
|
ct:pal(" OK (PatchBody: ~p)~n", [PatchBody]),
|
||||||
|
{ok, {{_, 200, _}, _, PatchResp}} = httpc:request(patch, {AdminURL ++ "/v1/admin/reviews", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", PatchBody}, [], []),
|
||||||
|
#{<<"updated_count">> := UpdatedCount} = jsx:decode(list_to_binary(PatchResp), [return_maps]),
|
||||||
|
true = (UpdatedCount =:= 2),
|
||||||
|
ct:pal(" OK (updated: ~p)~n", [UpdatedCount]);
|
||||||
|
_ ->
|
||||||
|
ct:pal("SKIPPED (not enough reviews for bulk update)~n")
|
||||||
|
end,
|
||||||
|
|
||||||
|
%% TEST 27: Method not allowed (POST /v1/admin/reviews → 405)
|
||||||
|
ct:pal(" TEST 27: POST method not allowed... "),
|
||||||
|
{ok, {{_, 405, _}, _, _}} = httpc:request(post, {AdminURL ++ "/v1/admin/reviews", [{"Authorization", "Bearer " ++ binary_to_list(AdminToken)}], "application/json", <<"{}">>}, [], []),
|
||||||
ct:pal("OK~n"),
|
ct:pal("OK~n"),
|
||||||
|
|
||||||
ct:pal("~n✅ Admin API tests passed!~n"),
|
ct:pal("~n✅ Admin API tests passed!~n"),
|
||||||
|
|||||||
@@ -243,7 +243,9 @@ register_and_login(Email, Password) ->
|
|||||||
end.
|
end.
|
||||||
|
|
||||||
create_calendar(Token, Params) ->
|
create_calendar(Token, Params) ->
|
||||||
Id = extract_json(http_post("/v1/calendars", Params, Token), <<"id">>),
|
Response = http_post("/v1/calendars", Params, Token),
|
||||||
|
ct:pal(" create_calendar Response: ~p~n", [Response]),
|
||||||
|
Id = extract_json(Response, <<"id">>),
|
||||||
Id.
|
Id.
|
||||||
|
|
||||||
create_event(Token, CalId, Params) ->
|
create_event(Token, CalId, Params) ->
|
||||||
|
|||||||
Reference in New Issue
Block a user