Merge branch 'main-merge-base-a'
This commit is contained in:
@@ -5,13 +5,16 @@ package player
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
"juwan-backend/app/player/rpc/playerservice"
|
"strconv"
|
||||||
|
"time"
|
||||||
|
|
||||||
"juwan-backend/app/player/api/internal/svc"
|
"juwan-backend/app/player/api/internal/svc"
|
||||||
"juwan-backend/app/player/api/internal/types"
|
"juwan-backend/app/player/api/internal/types"
|
||||||
|
"juwan-backend/app/player/rpc/playerservice"
|
||||||
|
"juwan-backend/app/users/rpc/usercenter"
|
||||||
|
|
||||||
"github.com/jinzhu/copier"
|
|
||||||
"github.com/zeromicro/go-zero/core/logx"
|
"github.com/zeromicro/go-zero/core/logx"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -31,18 +34,83 @@ func NewGetPlayerLogic(ctx context.Context, svcCtx *svc.ServiceContext) *GetPlay
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (l *GetPlayerLogic) GetPlayer(req *types.GetPlayerReq) (resp *types.PlayerProfile, err error) {
|
func (l *GetPlayerLogic) GetPlayer(req *types.GetPlayerReq) (resp *types.PlayerProfile, err error) {
|
||||||
player, err := l.svcCtx.PlayerRpc.GetPlayersById(l.ctx, &playerservice.GetPlayersByIdReq{
|
playerResp, err := l.svcCtx.PlayerRpc.GetPlayersById(l.ctx, &playerservice.GetPlayersByIdReq{
|
||||||
Id: req.Id,
|
Id: req.Id,
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logx.Errorf("GetPlayerLogic.GetPlayers err: %v", err)
|
logx.Errorf("GetPlayerLogic.GetPlayers err: %v", err)
|
||||||
return nil, errors.New("failed to get player details")
|
return nil, errors.New("failed to get player details")
|
||||||
}
|
}
|
||||||
resp = &types.PlayerProfile{}
|
player := playerResp.GetPlayers()
|
||||||
err = copier.Copy(resp, &player)
|
if player == nil {
|
||||||
if err != nil {
|
return nil, errors.New("player not found")
|
||||||
logx.Errorf("copier.Copy err: %v", err)
|
|
||||||
return nil, errors.New("copier.Copy err")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
resp = &types.PlayerProfile{
|
||||||
|
Id: player.Id,
|
||||||
|
Rating: player.Rating,
|
||||||
|
TotalOrders: player.TotalOrders,
|
||||||
|
Status: player.Status,
|
||||||
|
Gender: player.Gender,
|
||||||
|
Services: []types.PlayerService{},
|
||||||
|
Tags: append([]string{}, player.Tags...),
|
||||||
|
}
|
||||||
|
|
||||||
|
games := make([]string, 0, len(player.Games))
|
||||||
|
for _, gameID := range player.Games {
|
||||||
|
games = append(games, strconv.FormatInt(gameID, 10))
|
||||||
|
}
|
||||||
|
resp.Games = games
|
||||||
|
|
||||||
|
if player.ShopId != 0 {
|
||||||
|
resp.ShopId = strconv.FormatInt(player.ShopId, 10)
|
||||||
|
}
|
||||||
|
|
||||||
|
usersResp, err := l.svcCtx.UserRpc.GetUsersByIds(l.ctx, &usercenter.GetUsersByIdsReq{Ids: []int64{player.UserId}})
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if len(usersResp.Users) > 0 {
|
||||||
|
u := usersResp.Users[0]
|
||||||
|
verificationStatus := map[string]string{}
|
||||||
|
if u.VerificationStatus != "" {
|
||||||
|
_ = json.Unmarshal([]byte(u.VerificationStatus), &verificationStatus)
|
||||||
|
}
|
||||||
|
resp.User = types.UserProfile{
|
||||||
|
Id: strconv.FormatInt(u.Id, 10),
|
||||||
|
Username: u.Username,
|
||||||
|
Nickname: u.Nickname,
|
||||||
|
Avatar: u.Avatar,
|
||||||
|
Role: u.CurrentRole,
|
||||||
|
VerifiedRoles: append([]string{}, u.VerifiedRoles...),
|
||||||
|
VerificationStatus: verificationStatus,
|
||||||
|
Phone: u.Phone,
|
||||||
|
Bio: u.Bio,
|
||||||
|
CreatedAt: time.Unix(u.CreatedAt, 0).Format(time.DateTime),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
svcResp, svcErr := l.svcCtx.PlayerRpc.SearchPlayerServices(l.ctx, &playerservice.SearchPlayerServicesReq{
|
||||||
|
PlayerId: player.Id,
|
||||||
|
Limit: 100,
|
||||||
|
})
|
||||||
|
if svcErr != nil {
|
||||||
|
logx.Errorf("GetPlayer SearchPlayerServices player=%d err: %v", player.Id, svcErr)
|
||||||
|
} else {
|
||||||
|
for _, s := range svcResp.PlayerServices {
|
||||||
|
resp.Services = append(resp.Services, types.PlayerService{
|
||||||
|
Id: s.Id,
|
||||||
|
PlayerId: s.PlayerId,
|
||||||
|
GameId: s.GameId,
|
||||||
|
Title: s.Title,
|
||||||
|
Description: s.Description,
|
||||||
|
Price: s.Price,
|
||||||
|
Unit: s.Unit,
|
||||||
|
RankRange: s.RankRange,
|
||||||
|
Availability: s.Availability,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -36,6 +36,7 @@ func (l *GetPlayersByIdLogic) GetPlayersById(in *pb.GetPlayersByIdReq) (*pb.GetP
|
|||||||
Id: player.ID,
|
Id: player.ID,
|
||||||
UserId: player.UserID,
|
UserId: player.UserID,
|
||||||
Status: player.Status,
|
Status: player.Status,
|
||||||
|
Gender: player.Gender,
|
||||||
Rating: player.Rating.InexactFloat64(),
|
Rating: player.Rating.InexactFloat64(),
|
||||||
TotalOrders: int64(player.TotalOrders),
|
TotalOrders: int64(player.TotalOrders),
|
||||||
CompletedOrders: int64(player.CompletedOrders),
|
CompletedOrders: int64(player.CompletedOrders),
|
||||||
|
|||||||
+12
-1
@@ -3,8 +3,9 @@ static_resources:
|
|||||||
- name: ingress_http
|
- name: ingress_http
|
||||||
address:
|
address:
|
||||||
socket_address:
|
socket_address:
|
||||||
address: 0.0.0.0
|
address: "::"
|
||||||
port_value: 8080
|
port_value: 8080
|
||||||
|
ipv4_compat: true
|
||||||
filter_chains:
|
filter_chains:
|
||||||
- filters:
|
- filters:
|
||||||
- name: envoy.filters.network.http_connection_manager
|
- name: envoy.filters.network.http_connection_manager
|
||||||
@@ -651,6 +652,11 @@ static_resources:
|
|||||||
headers:
|
headers:
|
||||||
- name: ":method"
|
- name: ":method"
|
||||||
exact_match: GET
|
exact_match: GET
|
||||||
|
requires:
|
||||||
|
requires_any:
|
||||||
|
requirements:
|
||||||
|
- provider_name: juwan_user_jwt
|
||||||
|
- allow_missing: {}
|
||||||
- match:
|
- match:
|
||||||
safe_regex:
|
safe_regex:
|
||||||
google_re2: {}
|
google_re2: {}
|
||||||
@@ -663,6 +669,11 @@ static_resources:
|
|||||||
headers:
|
headers:
|
||||||
- name: ":method"
|
- name: ":method"
|
||||||
exact_match: GET
|
exact_match: GET
|
||||||
|
requires:
|
||||||
|
requires_any:
|
||||||
|
requirements:
|
||||||
|
- provider_name: juwan_user_jwt
|
||||||
|
- allow_missing: {}
|
||||||
- match:
|
- match:
|
||||||
prefix: /api/v1/reviews
|
prefix: /api/v1/reviews
|
||||||
headers:
|
headers:
|
||||||
|
|||||||
+208
-3
@@ -50,6 +50,28 @@ def same_id(left, right):
|
|||||||
return str(left) == str(right)
|
return str(left) == str(right)
|
||||||
|
|
||||||
|
|
||||||
|
def body_text(body):
|
||||||
|
if not isinstance(body, dict):
|
||||||
|
return str(body)
|
||||||
|
if "_raw" in body:
|
||||||
|
return str(body["_raw"])
|
||||||
|
if "message" in body:
|
||||||
|
return str(body["message"])
|
||||||
|
if "_error" in body:
|
||||||
|
return str(body["_error"])
|
||||||
|
return json.dumps(body, ensure_ascii=False)
|
||||||
|
|
||||||
|
|
||||||
|
def has_error_text(body, *parts):
|
||||||
|
text = body_text(body)
|
||||||
|
return all(part in text for part in parts)
|
||||||
|
|
||||||
|
|
||||||
|
def report_rejected(name, status_code, body, expect_status=(400, 403, 500)):
|
||||||
|
report(name, status_code, body, expect_status=expect_status)
|
||||||
|
report_check(f"{name} returns error body", bool(body_text(body)), body)
|
||||||
|
|
||||||
|
|
||||||
def pick_items(body):
|
def pick_items(body):
|
||||||
if isinstance(body.get("items"), list):
|
if isinstance(body.get("items"), list):
|
||||||
return body["items"]
|
return body["items"]
|
||||||
@@ -572,6 +594,20 @@ def phase3_admin_and_verification(
|
|||||||
body,
|
body,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
code, body, _ = s_reject.post(
|
||||||
|
f"{GATEWAY}/api/v1/users/me/switch-role",
|
||||||
|
json_body={"role": "owner"},
|
||||||
|
headers=csrf_reject,
|
||||||
|
)
|
||||||
|
report_rejected("POST /users/me/switch-role (rejected owner)", code, body)
|
||||||
|
code, body, _ = s_reject.get(f"{GATEWAY}/api/v1/users/me")
|
||||||
|
report("GET /users/me (after rejected owner switch)", code, body)
|
||||||
|
report_check(
|
||||||
|
"rejected owner switch does not change current role",
|
||||||
|
body.get("role") == "consumer",
|
||||||
|
body,
|
||||||
|
)
|
||||||
|
|
||||||
code, body, _ = s_user.get(f"{GATEWAY}/api/v1/users/me")
|
code, body, _ = s_user.get(f"{GATEWAY}/api/v1/users/me")
|
||||||
report("GET /users/me (after approval)", code, body)
|
report("GET /users/me (after approval)", code, body)
|
||||||
verified_roles = body.get("verifiedRoles") or []
|
verified_roles = body.get("verifiedRoles") or []
|
||||||
@@ -791,7 +827,12 @@ def phase6_player(s: Session, game_id):
|
|||||||
report(f"GET /players/{player_id}", code, body)
|
report(f"GET /players/{player_id}", code, body)
|
||||||
report_check(
|
report_check(
|
||||||
f"GET /players/{player_id} returns initialized online player",
|
f"GET /players/{player_id} returns initialized online player",
|
||||||
code == 200 and same_id(body.get("id"), player_id) and body.get("status") == "online",
|
code == 200
|
||||||
|
and same_id(body.get("id"), player_id)
|
||||||
|
and body.get("status") == "online"
|
||||||
|
and body.get("gender") is True
|
||||||
|
and as_int(user_from(body).get("id")) > 0
|
||||||
|
and isinstance(body.get("services"), list),
|
||||||
body,
|
body,
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -860,6 +901,19 @@ def phase6_player(s: Session, game_id):
|
|||||||
body,
|
body,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
code, body, _ = s.get(f"{GATEWAY}/api/v1/players/{player_id}")
|
||||||
|
report(f"GET /players/{player_id} (after service update)", code, body)
|
||||||
|
service = find_item_by_id(body.get("services") or [], service_id)
|
||||||
|
report_check(
|
||||||
|
f"GET /players/{player_id} includes updated service",
|
||||||
|
code == 200
|
||||||
|
and body.get("gender") is True
|
||||||
|
and bool(service)
|
||||||
|
and service.get("title") == "Updated Service"
|
||||||
|
and as_decimal(service.get("price")) == Decimal("60"),
|
||||||
|
body,
|
||||||
|
)
|
||||||
|
|
||||||
if player_id:
|
if player_id:
|
||||||
code, body, _ = s.get(f"{GATEWAY}/api/v1/players/{player_id}/services")
|
code, body, _ = s.get(f"{GATEWAY}/api/v1/players/{player_id}/services")
|
||||||
report(f"GET /players/{player_id}/services", code, body)
|
report(f"GET /players/{player_id}/services", code, body)
|
||||||
@@ -1076,6 +1130,40 @@ def phase7_shop(
|
|||||||
body,
|
body,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
code, body, _ = s_invited_player.post(
|
||||||
|
f"{GATEWAY}/api/v1/shops/{shop_id}/invitations",
|
||||||
|
json_body={"playerId": invited_player_id},
|
||||||
|
headers=s_invited_player.csrf_headers(),
|
||||||
|
)
|
||||||
|
report_rejected(
|
||||||
|
f"POST /shops/{shop_id}/invitations (non-owner)",
|
||||||
|
code,
|
||||||
|
body,
|
||||||
|
)
|
||||||
|
|
||||||
|
if invitation_id:
|
||||||
|
code, body, _ = s_owner.post(
|
||||||
|
f"{GATEWAY}/api/v1/shops/invitations/{invitation_id}/accept",
|
||||||
|
json_body={},
|
||||||
|
headers=csrf,
|
||||||
|
)
|
||||||
|
report_rejected(
|
||||||
|
f"POST /shops/invitations/{invitation_id}/accept (wrong player)",
|
||||||
|
code,
|
||||||
|
body,
|
||||||
|
)
|
||||||
|
code, body, _ = s_owner.get(
|
||||||
|
f"{GATEWAY}/api/v1/shops/{shop_id}/invitations",
|
||||||
|
)
|
||||||
|
report(f"GET /shops/{shop_id}/invitations (after wrong accept)", code, body)
|
||||||
|
pending_invitation = find_item_by_id(pick_items(body), invitation_id)
|
||||||
|
report_check(
|
||||||
|
"wrong invitation accept keeps status pending",
|
||||||
|
bool(pending_invitation)
|
||||||
|
and pending_invitation.get("status") == "pending",
|
||||||
|
body,
|
||||||
|
)
|
||||||
|
|
||||||
s_invited_player.post(
|
s_invited_player.post(
|
||||||
f"{GATEWAY}/api/v1/users/me/switch-role",
|
f"{GATEWAY}/api/v1/users/me/switch-role",
|
||||||
json_body={"role": "player"},
|
json_body={"role": "player"},
|
||||||
@@ -1213,7 +1301,7 @@ def phase8_order(s_consumer: Session, s_actor: Session, player_id, service_id, s
|
|||||||
and same_id(order_obj.get("playerId"), player_id)
|
and same_id(order_obj.get("playerId"), player_id)
|
||||||
and same_id(order_obj.get("shopId"), shop_id)
|
and same_id(order_obj.get("shopId"), shop_id)
|
||||||
and order_obj.get("status") == "pending_payment"
|
and order_obj.get("status") == "pending_payment"
|
||||||
and as_decimal(order_obj.get("totalPrice")) == Decimal("50")
|
and as_decimal(order_obj.get("totalPrice")) == Decimal("60")
|
||||||
and order_obj.get("note") == "test order",
|
and order_obj.get("note") == "test order",
|
||||||
body,
|
body,
|
||||||
)
|
)
|
||||||
@@ -1248,10 +1336,28 @@ def phase8_order(s_consumer: Session, s_actor: Session, player_id, service_id, s
|
|||||||
"created order detail matches participants",
|
"created order detail matches participants",
|
||||||
same_id(body.get("consumerId"), consumer_user_id)
|
same_id(body.get("consumerId"), consumer_user_id)
|
||||||
and same_id(body.get("playerId"), player_id)
|
and same_id(body.get("playerId"), player_id)
|
||||||
and same_id(body.get("shopId"), shop_id),
|
and same_id(body.get("shopId"), shop_id)
|
||||||
|
and as_decimal(body.get("totalPrice")) == Decimal("60")
|
||||||
|
and body.get("note") == "test order",
|
||||||
body,
|
body,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
code, body, _ = s_actor.post(
|
||||||
|
f"{GATEWAY}/api/v1/orders/{order_id}/accept",
|
||||||
|
json_body={},
|
||||||
|
headers=s_actor.csrf_headers(),
|
||||||
|
)
|
||||||
|
report_rejected(f"POST /orders/{order_id}/accept (before pay)", code, body)
|
||||||
|
check_order_status(s_consumer, order_id, "pending_payment", "after early accept")
|
||||||
|
|
||||||
|
code, body, _ = s_consumer.post(
|
||||||
|
f"{GATEWAY}/api/v1/orders/{order_id}/confirm-close",
|
||||||
|
json_body={},
|
||||||
|
headers=csrf,
|
||||||
|
)
|
||||||
|
report_rejected(f"POST /orders/{order_id}/confirm-close (before close request)", code, body)
|
||||||
|
check_order_status(s_consumer, order_id, "pending_payment", "after early confirm-close")
|
||||||
|
|
||||||
code, body, _ = s_consumer.post(
|
code, body, _ = s_consumer.post(
|
||||||
f"{GATEWAY}/api/v1/orders/{order_id}/pay",
|
f"{GATEWAY}/api/v1/orders/{order_id}/pay",
|
||||||
json_body={},
|
json_body={},
|
||||||
@@ -1260,6 +1366,22 @@ def phase8_order(s_consumer: Session, s_actor: Session, player_id, service_id, s
|
|||||||
report(f"POST /orders/{order_id}/pay", code, body)
|
report(f"POST /orders/{order_id}/pay", code, body)
|
||||||
check_order_status(s_consumer, order_id, "pending_accept", "after pay")
|
check_order_status(s_consumer, order_id, "pending_accept", "after pay")
|
||||||
|
|
||||||
|
code, body, _ = s_actor.post(
|
||||||
|
f"{GATEWAY}/api/v1/orders/{order_id}/request-close",
|
||||||
|
json_body={},
|
||||||
|
headers=s_actor.csrf_headers(),
|
||||||
|
)
|
||||||
|
report_rejected(f"POST /orders/{order_id}/request-close (before accept)", code, body)
|
||||||
|
check_order_status(s_consumer, order_id, "pending_accept", "after early request-close")
|
||||||
|
|
||||||
|
code, body, _ = s_consumer.post(
|
||||||
|
f"{GATEWAY}/api/v1/orders/{order_id}/pay",
|
||||||
|
json_body={},
|
||||||
|
headers=csrf,
|
||||||
|
)
|
||||||
|
report_rejected(f"POST /orders/{order_id}/pay (second time)", code, body)
|
||||||
|
check_order_status(s_consumer, order_id, "pending_accept", "after duplicate pay")
|
||||||
|
|
||||||
code, body, _ = s_actor.post(
|
code, body, _ = s_actor.post(
|
||||||
f"{GATEWAY}/api/v1/orders/{order_id}/accept",
|
f"{GATEWAY}/api/v1/orders/{order_id}/accept",
|
||||||
json_body={},
|
json_body={},
|
||||||
@@ -1269,6 +1391,14 @@ def phase8_order(s_consumer: Session, s_actor: Session, player_id, service_id, s
|
|||||||
body = check_order_status(s_consumer, order_id, "in_progress", "after accept")
|
body = check_order_status(s_consumer, order_id, "in_progress", "after accept")
|
||||||
report_check("accepted order has acceptedAt", bool(body.get("acceptedAt")), body)
|
report_check("accepted order has acceptedAt", bool(body.get("acceptedAt")), body)
|
||||||
|
|
||||||
|
code, body, _ = s_consumer.post(
|
||||||
|
f"{GATEWAY}/api/v1/orders/{order_id}/confirm-close",
|
||||||
|
json_body={},
|
||||||
|
headers=csrf,
|
||||||
|
)
|
||||||
|
report_rejected(f"POST /orders/{order_id}/confirm-close (before request-close)", code, body)
|
||||||
|
check_order_status(s_consumer, order_id, "in_progress", "after early confirm-close")
|
||||||
|
|
||||||
code, body, _ = s_actor.post(
|
code, body, _ = s_actor.post(
|
||||||
f"{GATEWAY}/api/v1/orders/{order_id}/request-close",
|
f"{GATEWAY}/api/v1/orders/{order_id}/request-close",
|
||||||
json_body={},
|
json_body={},
|
||||||
@@ -1302,6 +1432,15 @@ def phase8_order(s_consumer: Session, s_actor: Session, player_id, service_id, s
|
|||||||
body,
|
body,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if order2_id:
|
||||||
|
code, body, _ = s_consumer.post(
|
||||||
|
f"{GATEWAY}/api/v1/orders/{order2_id}/review",
|
||||||
|
json_body={"rating": 5, "content": "too early"},
|
||||||
|
headers=csrf,
|
||||||
|
)
|
||||||
|
report_rejected(f"POST /orders/{order2_id}/review (before completion)", code, body)
|
||||||
|
check_order_status(s_consumer, order2_id, "pending_payment", "after early review")
|
||||||
|
|
||||||
if order_id:
|
if order_id:
|
||||||
code, body, _ = s_consumer.post(
|
code, body, _ = s_consumer.post(
|
||||||
f"{GATEWAY}/api/v1/orders/{order_id}/reorder",
|
f"{GATEWAY}/api/v1/orders/{order_id}/reorder",
|
||||||
@@ -1336,6 +1475,14 @@ def phase8_order(s_consumer: Session, s_actor: Session, player_id, service_id, s
|
|||||||
report(f"POST /orders/{order2_id}/cancel", code, body)
|
report(f"POST /orders/{order2_id}/cancel", code, body)
|
||||||
check_order_status(s_consumer, order2_id, "cancelled", "second order after cancel")
|
check_order_status(s_consumer, order2_id, "cancelled", "second order after cancel")
|
||||||
|
|
||||||
|
code, body, _ = s_consumer.post(
|
||||||
|
f"{GATEWAY}/api/v1/orders/{order2_id}/pay",
|
||||||
|
json_body={},
|
||||||
|
headers=csrf,
|
||||||
|
)
|
||||||
|
report_rejected(f"POST /orders/{order2_id}/pay (after cancel)", code, body)
|
||||||
|
check_order_status(s_consumer, order2_id, "cancelled", "after cancelled pay")
|
||||||
|
|
||||||
code, body, _ = s_consumer.post(
|
code, body, _ = s_consumer.post(
|
||||||
f"{GATEWAY}/api/v1/orders/paid",
|
f"{GATEWAY}/api/v1/orders/paid",
|
||||||
json_body={
|
json_body={
|
||||||
@@ -1673,6 +1820,26 @@ def phase8e_search_and_favorites(s: Session, user_id, player_id, shop_id):
|
|||||||
skip("Favorites mutation flow", "Missing user or player id")
|
skip("Favorites mutation flow", "Missing user or player id")
|
||||||
return
|
return
|
||||||
|
|
||||||
|
code, body, _ = s.post(
|
||||||
|
f"{GATEWAY}/api/v1/favorites",
|
||||||
|
json_body={"targetType": "player", "targetId": "not-a-snowflake"},
|
||||||
|
headers=s.csrf_headers(),
|
||||||
|
)
|
||||||
|
report_rejected("POST /favorites (invalid targetId)", code, body)
|
||||||
|
code, body, _ = s.get(f"{GATEWAY}/api/v1/favorites")
|
||||||
|
report("GET /favorites (after invalid targetId)", code, body)
|
||||||
|
report_check(
|
||||||
|
"invalid favorite target is absent",
|
||||||
|
not bool(
|
||||||
|
find_item(
|
||||||
|
pick_items(body),
|
||||||
|
lambda item: item.get("targetType") == "player"
|
||||||
|
and same_id(item.get("targetId"), "not-a-snowflake"),
|
||||||
|
)
|
||||||
|
),
|
||||||
|
body,
|
||||||
|
)
|
||||||
|
|
||||||
code, body, _ = s.post(
|
code, body, _ = s.post(
|
||||||
f"{GATEWAY}/api/v1/favorites",
|
f"{GATEWAY}/api/v1/favorites",
|
||||||
json_body={"targetType": "player", "targetId": str(player_id)},
|
json_body={"targetType": "player", "targetId": str(player_id)},
|
||||||
@@ -1777,6 +1944,35 @@ def phase9_wallet(s: Session):
|
|||||||
body,
|
body,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
code, body, _ = s.post(
|
||||||
|
f"{GATEWAY}/api/v1/wallet/withdraw",
|
||||||
|
json_body={"amount": "0", "method": "alipay"},
|
||||||
|
headers=csrf,
|
||||||
|
)
|
||||||
|
report_rejected("POST /wallet/withdraw (zero amount)", code, body)
|
||||||
|
code, body, _ = s.get(f"{GATEWAY}/api/v1/wallet/balance")
|
||||||
|
report("GET /wallet/balance (after zero withdraw)", code, body)
|
||||||
|
report_check(
|
||||||
|
"zero withdraw keeps balance unchanged",
|
||||||
|
as_decimal(body.get("balance")) == initial_balance,
|
||||||
|
body,
|
||||||
|
)
|
||||||
|
|
||||||
|
excessive_amount = initial_balance + Decimal("999999")
|
||||||
|
code, body, _ = s.post(
|
||||||
|
f"{GATEWAY}/api/v1/wallet/withdraw",
|
||||||
|
json_body={"amount": str(excessive_amount), "method": "alipay"},
|
||||||
|
headers=csrf,
|
||||||
|
)
|
||||||
|
report_rejected("POST /wallet/withdraw (insufficient balance)", code, body)
|
||||||
|
code, body, _ = s.get(f"{GATEWAY}/api/v1/wallet/balance")
|
||||||
|
report("GET /wallet/balance (after insufficient withdraw)", code, body)
|
||||||
|
report_check(
|
||||||
|
"insufficient withdraw keeps balance unchanged",
|
||||||
|
as_decimal(body.get("balance")) == initial_balance,
|
||||||
|
body,
|
||||||
|
)
|
||||||
|
|
||||||
code, body, _ = s.post(
|
code, body, _ = s.post(
|
||||||
f"{GATEWAY}/api/v1/wallet/topup",
|
f"{GATEWAY}/api/v1/wallet/topup",
|
||||||
json_body={"amount": "100.00", "method": "alipay"},
|
json_body={"amount": "100.00", "method": "alipay"},
|
||||||
@@ -1993,6 +2189,15 @@ def phase10_community(s: Session, user_id):
|
|||||||
def phase11_objectstory(s: Session):
|
def phase11_objectstory(s: Session):
|
||||||
print("\n=== Phase 11: Objectstory (File) ===")
|
print("\n=== Phase 11: Objectstory (File) ===")
|
||||||
|
|
||||||
|
code, body, _ = s.post_multipart(
|
||||||
|
f"{GATEWAY}/api/v1/upload",
|
||||||
|
fields={"type": "post"},
|
||||||
|
files={"file": (f"empty-{rand_str(4)}.txt", b"", "text/plain")},
|
||||||
|
headers=s.csrf_headers(),
|
||||||
|
)
|
||||||
|
report_rejected("POST /upload (empty file)", code, body)
|
||||||
|
report_check("empty upload does not return url", not body.get("url"), body)
|
||||||
|
|
||||||
code, body, _ = s.post_multipart(
|
code, body, _ = s.post_multipart(
|
||||||
f"{GATEWAY}/api/v1/upload",
|
f"{GATEWAY}/api/v1/upload",
|
||||||
fields={"type": "post"},
|
fields={"type": "post"},
|
||||||
|
|||||||
@@ -449,6 +449,11 @@ data:
|
|||||||
headers:
|
headers:
|
||||||
- name: ":method"
|
- name: ":method"
|
||||||
exact_match: "GET"
|
exact_match: "GET"
|
||||||
|
requires:
|
||||||
|
requires_any:
|
||||||
|
requirements:
|
||||||
|
- provider_name: juwan_user_jwt
|
||||||
|
- allow_missing: {}
|
||||||
- match:
|
- match:
|
||||||
prefix: "/api/v1/reviews"
|
prefix: "/api/v1/reviews"
|
||||||
headers:
|
headers:
|
||||||
@@ -487,6 +492,11 @@ data:
|
|||||||
headers:
|
headers:
|
||||||
- name: ":method"
|
- name: ":method"
|
||||||
exact_match: "GET"
|
exact_match: "GET"
|
||||||
|
requires:
|
||||||
|
requires_any:
|
||||||
|
requirements:
|
||||||
|
- provider_name: juwan_user_jwt
|
||||||
|
- allow_missing: {}
|
||||||
- match:
|
- match:
|
||||||
safe_regex:
|
safe_regex:
|
||||||
google_re2: {}
|
google_re2: {}
|
||||||
|
|||||||
Reference in New Issue
Block a user