Files
caliverse_server/UGQApiServer/Controllers/QuestEditorController.cs
2025-05-01 07:20:41 +09:00

625 lines
26 KiB
C#

using System.Globalization;
using Asp.Versioning;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc;
using MongoDB.Bson.Serialization.Attributes;
using Swashbuckle.AspNetCore.Annotations;
using Swashbuckle.AspNetCore.Filters;
using UGQApiServer.Models;
using UGQDatabase.Models;
using UGQDataAccess.Service;
using UGQApiServer.Converter;
using UGQApiServer.Storage;
using UGQDataAccess.Repository.Models;
using Microsoft.AspNetCore.Authorization;
using UGQDataAccess;
using UGQApiServer.Validation;
using UGQApiServer.UGQData;
using UGQApiServer.Auth;
using JwtRoleAuthentication.Services;
using System.Security.Claims;
using Newtonsoft.Json.Linq;
using ServerCommon;
using ServerCommon.UGQ;
using ServerCommon.UGQ.Models;
using Microsoft.Extensions.Caching.Distributed;
using Microsoft.IdentityModel.Tokens;
using System.Collections.Generic;
using UGQApiServer.Controllers.Common;
using UGQDataAccess.Repository;
using ServerCore; using ServerBase;
using StackExchange.Redis;
using UGQDataAccess.Logs;
namespace UGQApiServer.Controllers
{
[Authorize]
[ApiController]
[ApiVersion("1.0")]
[Route("api/v{version:apiVersion}/[controller]")]
[UGQWebApi]
[AllowWhenSingleLogin]
public class QuestEditorController : ControllerBase
{
AccountService _accountService;
QuestEditorService _questEditorService;
IStorageService _storageService;
UGQMetaData _ugqMetaData;
TokenService _tokenService;
WebPortalTokenAuth _webPortalTokenAuth;
DynamoDbClient _dynamoDbClient;
readonly IConnectionMultiplexer _redis;
QuestEditorApi _questEditorApi;
public QuestEditorController(
AccountService accountService,
QuestEditorService questEditorService, IStorageService storageService,
UGQMetaData uqgMetadata, TokenService tokenService,
WebPortalTokenAuth webPortalTokenAuth,
DynamoDbClient dynamoDbClient,
QuestEditorApi questEditorApi,
IConnectionMultiplexer redis)
{
_accountService = accountService;
_questEditorService = questEditorService;
_storageService = storageService;
_ugqMetaData = uqgMetadata;
_tokenService = tokenService;
_webPortalTokenAuth = webPortalTokenAuth;
_dynamoDbClient = dynamoDbClient;
_questEditorApi = questEditorApi;
_redis = redis;
}
string? userGuidFromToken()
{
return User.Claims.FirstOrDefault(c => c.Type == "Id")?.Value;
}
/// <summary>
/// 퀘스트 랩 시작 페이지
/// </summary>
/// <param name="pageNumber">pageNumber</param>
/// <param name="pageSize">pageSize (최대 100)</param>
/// <param name="state">검색할 상태</param>
/// <param name="searchText">검색할 텍스트</param>
[HttpGet]
[Route("summaries")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQSummaryList))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> getSummaries([FromQuery] int pageNumber, [FromQuery] int pageSize, [FromQuery] QuestContentState state, [FromQuery] string? searchText)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
pageSize = Math.Clamp(pageSize, 1, UGQConstants.MAX_INPUT_PAGE_SIZE);
// TODO: state만 얻어오게 수정
List<QuestContentEntity> all = await _questEditorService.getAll(userGuid);
var result = await _questEditorService.getSummaries(pageNumber, pageSize, userGuid, state, searchText);
var query = all.GroupBy(
p => p.State,
p => p.State,
(state, states) => new
{
Key = state,
Count = states.Count(),
});
var stateGroups = query.Select(x => new UGQStateGroup
{
State = x.Key,
Count = x.Count,
}).ToList();
LangEnum langEnum = Culture.ToLangEnum(CultureInfo.CurrentCulture.Name);
return Results.Ok(new UGQSummaryResponse
{
TotalCount = all.Count,
StateGroups = stateGroups,
PageNumber = result.PageNumber,
PageSize = result.PageSize,
TotalPages = result.TotalPages,
Summaries = result.Items.Select(x => x.ToDTO(langEnum)).ToList(),
});
}
/// <summary>
/// 유저 정보 얻기
/// </summary>
[HttpGet]
[Route("account")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQAccountDetail))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> getAccount()
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
var accountEntity = await _accountService.getAccount(userGuid);
if(accountEntity == null)
return Results.BadRequest(ApiResponseFactory.error(ServerErrorCode.UgqNullEntity));
// TODO: count만 얻어오게 수정
List<QuestContentEntity> all = await _questEditorService.getAll(userGuid);
return Results.Ok(new UGQAccountDetail
{
UserGuid = accountEntity.UserGuid,
Nickname = accountEntity.Nickname,
AccountId = accountEntity.AccountId ?? "",
SlotCount = (MetaHelper.GameConfigMeta.UGQDefaultSlot + accountEntity.AdditionalSlotCount),
GradeType = accountEntity.GradeType,
CreatorPoint = accountEntity.CreatorPoint,
TotalQuests = all.Count,
});
}
/// <summary>
/// 유저 재화 정보 얻기
/// </summary>
[HttpGet]
[Route("account-currency")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQAccount))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> getAccountCurrency()
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
if (Enum.TryParse(MetaHelper.GameConfigMeta.UGQAddSlotType, true, out CurrencyType currencyType) == false)
return Results.BadRequest(ApiResponseFactory.error(ServerErrorCode.UgqServerException));
double slotPrice = MetaHelper.GameConfigMeta.UGQAddSlotAmount;
(var result, double currentAmount) = await CurrencyControlHelper.getMoneyByUserGuid(userGuid, currencyType);
if (result.isFail())
return Results.BadRequest(ApiResponseFactory.error(ServerErrorCode.UgqCurrencyError));
return Results.Ok(new UGQAccountCurrency
{
CurrencyType = currencyType,
Amount = currentAmount,
});
}
/// <summary>
/// 유저의 메타버스 로그인 상태 얻기
/// </summary>
[HttpGet]
[Route("metaverse-online-status")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQAccount))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> getMetaverseOnlineStatus()
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
bool online = false;
var game_login_cache = await _redis.GetDatabase().StringGetAsync($"login:{userGuid}");
if (game_login_cache.HasValue == true)
{
online = true;
}
return Results.Ok(new {
Online = online
});
}
/// <summary>
/// 퀘스트 슬롯을 추가하는데 필요한 재화와 소모량 얻기
/// </summary>
[HttpGet]
[Route("slot-price")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQSlotPrice))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> getSlotPrice()
{
await Task.CompletedTask;
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
// MetaHelper.GameConfigMeta.UGQDefaultSlot
if (Enum.TryParse(MetaHelper.GameConfigMeta.UGQAddSlotType, true, out CurrencyType currencyType) == false)
return Results.BadRequest(ApiResponseFactory.error(ServerErrorCode.UgqServerException));
double slotPrice = MetaHelper.GameConfigMeta.UGQAddSlotAmount;
return Results.Ok(new UGQSlotPrice
{
CurrencyType = currencyType,
Price = slotPrice
});
}
/// <summary>
/// 퀘스트 슬롯 추가
/// </summary>
[HttpPost]
[Route("slot")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQAccount))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> addSlot()
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
if (Enum.TryParse(MetaHelper.GameConfigMeta.UGQAddSlotType, true, out CurrencyType currencyType) == false)
return Results.BadRequest(ApiResponseFactory.error(ServerErrorCode.UgqServerException));
double slotPrice = MetaHelper.GameConfigMeta.UGQAddSlotAmount;
var accountEntity = await _accountService.getAccount(userGuid);
if (accountEntity == null)
return Results.BadRequest(ApiResponseFactory.error(ServerErrorCode.UgqNullEntity));
int finalCount = MetaHelper.GameConfigMeta.UGQDefaultSlot + accountEntity.AdditionalSlotCount + 1;
if (finalCount > MetaHelper.GameConfigMeta.UGQMaximumSlot)
return Results.BadRequest(ApiResponseFactory.error(ServerErrorCode.UgqSlotLimit));
var result = new Result();
var db_connector = CurrencyControlHelper.getDbConnector();
(result, var is_login) = await ServerCommon.EntityHelper.isUserLoggedIn(db_connector, userGuid);
if (result.isFail())
return Results.BadRequest(ApiResponseFactory.error(ServerErrorCode.UgqServerException));
if (true == is_login)
return Results.BadRequest(ApiResponseFactory.error(ServerErrorCode.UgqServerException));
// 일단 슬롯 추가할 수 있음
// 게임 DB 재화 체크, 소모
(result, double currentAmount) = await CurrencyControlHelper.getMoneyByUserGuid(userGuid, currencyType);
if (result.isFail())
return Results.BadRequest(ApiResponseFactory.error(ServerErrorCode.UgqCurrencyError));
if (currentAmount < slotPrice)
return Results.BadRequest(ApiResponseFactory.error(ServerErrorCode.UgqNotEnoughCurrency));
(result, double updatedAmount) = await CurrencyControlHelper.changeMoneyByUserGuid(userGuid, currencyType, slotPrice);
if (result.isFail())
return Results.BadRequest(ApiResponseFactory.error(ServerErrorCode.UgqCurrencyError));
(ServerErrorCode errorCode, var updatedAccountEntity) = await _accountService.addSlotCount(userGuid, 1, accountEntity.SlotCountVersion);
if (errorCode != ServerErrorCode.Success)
return Results.BadRequest(ApiResponseFactory.error(errorCode));
if (updatedAccountEntity == null)
{
// 슬롯 추가 실패
// 슬롯수 획인, 재화 소모 처리후 SlotCountVersion이 변경되었으면 슬롯 수 변경이 저장되지 않고 실패함.
// 재화는 돌려준다.
(result, double rollbackAmount) = await CurrencyControlHelper.earnMoneyByUserGuid(userGuid, currencyType, slotPrice);
if (result.isFail())
{
Log.getLogger().error($"currency rollback error. userGuid: {userGuid}, currencyType: {currencyType}, slotPrice: {slotPrice}");
return Results.BadRequest(ApiResponseFactory.error(ServerErrorCode.UgqCurrencyError));
}
return Results.BadRequest(ApiResponseFactory.error(ServerErrorCode.UgqNullEntity));
}
List<ILogInvoker> business_logs = [
new UgqApiAddSlotBusinessLog(updatedAccountEntity.AdditionalSlotCount, ""),
];
var log_action = new LogActionEx(LogActionType.UgqApiAddSlot);
UgqApiBusinessLogger.collectLogs(log_action, userGuid, business_logs);
return Results.Ok(updatedAccountEntity.ToDTO());
}
/// <summary>
/// 퀘스트 생성
/// </summary>
/// <remarks>
/// 빈 슬롯에 퀘스트 추가하는 경우에 호출
/// </remarks>
[HttpPost]
[Route("quest")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQContent))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> addQuest([FromBody] UGQSaveQuestModel saveQuestModel)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
return await _questEditorApi.addQuest(userGuid, saveQuestModel);
}
/// <summary>
/// 퀘스트 내용 가져오기
/// </summary>
/// <param name="questContentId">UGQSummary의 QuestContentId 값으로 api 호출</param>
[HttpGet]
[Route("quest/{questContentId}")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQContent))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> getQuest(string questContentId)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
return await _questEditorApi.getQuest(userGuid, questContentId);
}
/// <summary>
/// 편집 가능한 경우 퀘스트 내용 가져오기
/// </summary>
/// <remarks>
/// GET api와는 다르게 편집 불가능한 state인 경우에는 실패 리턴
/// </remarks>
/// <param name="questContentId">UGQSummary의 QuestContentId 값으로 api 호출</param>
[HttpPost]
[Route("quest/{questContentId}/edit")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQContent))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> editQuest(string questContentId)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
return await _questEditorApi.editQuest(userGuid, questContentId);
}
/// <summary>
/// 퀘스트 내용 업데이트
/// </summary>
/// <remarks>
/// Task의 DialogId는 업데이트하지 않음.
/// 편집 불가능 state인 경우 에러 리턴
/// </remarks>
/// <param name="questContentId">GET 또는 edit 에서 얻은 questContentId 사용</param>
/// <param name="saveQuestModel">웹에서 입력한 값</param>
[HttpPut]
[Route("quest/{questContentId}")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQContent))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> saveQuest(string questContentId, [FromBody] UGQSaveQuestModel saveQuestModel)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
return await _questEditorApi.saveQuest(userGuid, questContentId, saveQuestModel);
}
/// <summary>
/// 퀘스트 삭제
/// </summary>
/// <param name="questContentId">GET또는 edit 에서 얻은 questContentId 사용</param>
[HttpDelete]
[Route("quest/{questContentId}")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK)]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> deleteQuest(string questContentId)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
return await _questEditorApi.deleteQuest(userGuid, questContentId);
}
/// <summary>
/// 퀘스트 이미지 업로드
/// </summary>
/// <remarks>
/// 세부사항은 정리 예정
/// </remarks>
/// <param name="questContentId">GET 또는 edit 에서 얻은 questContentId 사용</param>
/// <param name="titleImageId">기본 이미지 인덱스</param>
/// <param name="bannerImageId">기본 이미지 인덱스</param>
/// <param name="title">업로드 파일</param>
/// <param name="banner">업로드 파일</param>
[HttpPost]
[Route("quest-image/{questContentId}")]
[Produces("application/json")]
public async Task<IResult> uploadQuestImage(string questContentId, [FromQuery] int? titleImageId, [FromQuery] int? bannerImageId, IFormFile? title, IFormFile? banner)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
return await _questEditorApi.uploadQuestImage(userGuid, questContentId, titleImageId ?? 0, bannerImageId ?? 0, title, banner);
}
/// <summary>
/// 퀘스트 타이틀 이미지 업로드
/// </summary>
/// <remarks>
/// 세부사항은 정리 예정
/// </remarks>
/// <param name="questContentId">GET 또는 edit 에서 얻은 questContentId 사용</param>
/// <param name="titleImageId">기본 이미지 인덱스</param>
/// <param name="title">업로드 파일</param>
[HttpPost]
[Route("quest-title-image/{questContentId}")]
[Produces("application/json")]
public async Task<IResult> uploadQuestTitleImage(string questContentId, [FromQuery] int? titleImageId, IFormFile? title)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
return await _questEditorApi.uploadQuestImage(userGuid, questContentId, titleImageId ?? 0, 0, title, null);
}
/// <summary>
/// 퀘스트 배너 이미지 업로드
/// </summary>
/// <remarks>
/// 세부사항은 정리 예정
/// </remarks>
/// <param name="questContentId">GET 또는 edit 에서 얻은 questContentId 사용</param>
/// <param name="bannerImageId">기본 이미지 인덱스</param>
/// <param name="banner">업로드 파일</param>
[HttpPost]
[Route("quest-banner-image/{questContentId}")]
[Produces("application/json")]
public async Task<IResult> uploadQuestBannerImage(string questContentId, [FromQuery] int? bannerImageId, IFormFile? banner)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
return await _questEditorApi.uploadQuestImage(userGuid, questContentId, 0, bannerImageId ?? 0, null, banner);
}
/// <summary>
/// 퀘스트 다이얼로그 가져오기
/// </summary>
/// <param name="questContentId">GET 또는 edit 에서 얻은 questContentId 사용</param>
/// <param name="dialogId">UGQContent.Tasks에 있는 DialogId 사용</param>
[HttpGet]
[Route("quest-dialog/{questContentId}/{dialogId}")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQDialog))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> getQuestDialog(string questContentId, string dialogId)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
return await _questEditorApi.getQuestDialog(userGuid, questContentId, dialogId);
}
/// <summary>
/// 퀘스트 다이얼로그 새로 생성
/// </summary>
/// <remarks>
/// Dialog 를 추가하고, UGQContent.Tasks의 DialogId를 저장
/// </remarks>
/// <param name="questContentId">GET 또는 edit 에서 얻은 questContentId 사용</param>
/// <param name="taskIndex">몇번째 Task에 추가되는 다이얼로그인지. 0부터 시작</param>
/// <param name="questDialog">questDialog</param>
[HttpPost]
[Route("quest-dialog/{questContentId}")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQAddQuestDialogResponse))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> addQuestDialog(string questContentId, [FromQuery] int taskIndex, [FromBody] UGQSaveDialogModel questDialog)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
return await _questEditorApi.addQuestDialog(userGuid, questContentId, taskIndex, questDialog);
}
/// <summary>
/// 퀘스트 다이얼로그 업데이트
/// </summary>
/// <param name="questContentId">GET 또는 edit 에서 얻은 questContentId 사용</param>
/// <param name="dialogId">UGQContent.Tasks에 있는 DialogId 사용</param>
/// /// <param name="questDialog">questDialog</param>
[HttpPut]
[Route("quest-dialog/{questContentId}/{dialogId}")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQDialog))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> saveQuestDialog(string questContentId, string dialogId, [FromBody] UGQSaveDialogModel questDialog)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
return await _questEditorApi.saveQuestDialog(userGuid, questContentId, dialogId, questDialog);
}
/// <summary>
/// 퀘스트 state 변경
/// </summary>
/// <remarks>
/// Test 로 변경 실패 시 ValidationErrorResponse로 응답
/// </remarks>
/// <param name="questContentId">GET또는 edit 에서 얻은 questContentId 사용</param>
/// <param name="state">QuestContentState</param>
[HttpPatch]
[Route("quest-state/{questContentId}")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQContent))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ValidationErrorResponse))]
public async Task<IResult> changeQuestState(string questContentId, [FromQuery] QuestContentState state)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
return await _questEditorApi.changeQuestState(userGuid, questContentId, state);
}
/// <summary>
/// CreatorPoint 내역 보기
/// </summary>
/// <param name="pageNumber">페이지</param>
/// <param name="pageSize">페이지 사이즈 (최대 100)</param>
/// <param name="kind">kind</param>
/// <param name="startDate">검색할 시작 시간</param>
/// <param name="endDate">검색할 종료 시간</param>
[HttpGet]
[Route("creator-point-history")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQCreatorPointHistoryResult))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> getCreatorPointHistory([FromQuery] int pageNumber, [FromQuery] int pageSize, CreatorPointHistoryKind kind, DateTimeOffset startDate, DateTimeOffset endDate)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
return await _questEditorApi.getCreatorPointHistory(userGuid, pageNumber, pageSize, kind, startDate, endDate);
}
/// <summary>
/// 퀘스트별 수익 보기
/// </summary>
/// <param name="pageNumber">페이지</param>
/// <param name="pageSize">페이지 사이즈 (최대 100)</param>
[HttpGet]
[Route("quest-profit-stats")]
[Produces("application/json")]
[SwaggerResponse(StatusCodes.Status200OK, Type = typeof(UGQQuestProfitStatsResult))]
[SwaggerResponse(StatusCodes.Status400BadRequest, Type = typeof(ApiErrorResponse))]
public async Task<IResult> getQuestProfitStats([FromQuery] int pageNumber, [FromQuery] int pageSize)
{
var userGuid = userGuidFromToken();
if (userGuid == null)
return Results.Unauthorized();
return await _questEditorApi.getQuestProfitStats(userGuid, pageNumber, pageSize);
}
}
}