[
    { "instance_id": "ptt-official-app__ptt-backend-123", "patch": "diff --git a\/internal\/delivery\/http\/route_users.go b\/internal\/delivery\/http\/route_users.go\nindex f3295c9..bd54ca0 100644\n--- a\/internal\/delivery\/http\/route_users.go\n+++ b\/internal\/delivery\/http\/route_users.go\n@@ -15,6 +15,8 @@ func (delivery *httpDelivery) getUsers(w http.ResponseWriter, r *http.Request) {\n \t\tdelivery.getUserInformation(w, r, userId)\n \tcase \"favorites\":\n \t\tdelivery.getUserFavorites(w, r, userId)\n+\tcase \"articles\":\n+\t\tdelivery.getUserArticles(w, r, userId)\n \tdefault:\n \t\tdelivery.logger.Noticef(\"user id: %v not exist but be queried, info: %v err: %v\", userId, item, err)\n \t\tw.WriteHeader(http.StatusNotFound)\n@@ -74,6 +76,39 @@ func (delivery *httpDelivery) getUserFavorites(w http.ResponseWriter, r *http.Re\n \tdataItems, err := delivery.usecase.GetUserFavorites(context.Background(), userId)\n \tif err != nil {\n \t\tdelivery.logger.Errorf(\"failed to get user favorites: %s\\n\", err)\n+\t\tw.WriteHeader(http.StatusBadRequest)\n+\t\treturn\n+\t}\n+\n+\tresponseMap := map[string]interface{}{\n+\t\t\"data\": map[string]interface{}{\n+\t\t\t\"items\": dataItems,\n+\t\t},\n+\t}\n+\n+\tresponseByte, _ := json.MarshalIndent(responseMap, \"\", \"  \")\n+\n+\tw.Write(responseByte)\n+}\n+\n+func (delivery *httpDelivery) getUserArticles(w http.ResponseWriter, r *http.Request, userID string) {\n+\ttoken := delivery.getTokenFromRequest(r)\n+\terr := delivery.usecase.CheckPermission(token,\n+\t\t[]usecase.Permission{usecase.PermissionReadUserInformation},\n+\t\tmap[string]string{\n+\t\t\t\"user_id\": userID,\n+\t\t})\n+\n+\tif err != nil {\n+\t\t\/\/ TODO: record unauthorized access\n+\t\tw.WriteHeader(http.StatusUnauthorized)\n+\t\treturn\n+\t}\n+\n+\t\/\/ return need fix\n+\tdataItems, err := delivery.usecase.GetUserArticles(context.Background(), userID)\n+\tif err != nil {\n+\t\tdelivery.logger.Errorf(\"failed to get user's articles: %s\\n\", err)\n \t}\n \n \tresponseMap := map[string]interface{}{\ndiff --git a\/internal\/repository\/repository.go b\/internal\/repository\/repository.go\nindex 33b2839..b902f1c 100644\n--- a\/internal\/repository\/repository.go\n+++ b\/internal\/repository\/repository.go\n@@ -39,6 +39,8 @@ type Repository interface {\n \tGetUsers(ctx context.Context) ([]bbs.UserRecord, error)\n \t\/\/ GetUserFavoriteRecords returns favorite records of a user\n \tGetUserFavoriteRecords(ctx context.Context, userID string) ([]bbs.FavoriteRecord, error)\n+\t\/\/ GetUserArticles returns user's articles\n+\tGetUserArticles(ctx context.Context, boardID string) ([]bbs.ArticleRecord, error)\n \n \t\/\/ article.go\n \t\/\/ GetPopularArticles returns all popular articles\ndiff --git a\/internal\/repository\/user.go b\/internal\/repository\/user.go\nindex bdb7b97..234720e 100644\n--- a\/internal\/repository\/user.go\n+++ b\/internal\/repository\/user.go\n@@ -55,13 +55,17 @@ func (repo *repository) GetUserFavoriteRecords(ctx context.Context, userID strin\n \treturn repo.db.ReadUserFavoriteRecords(userID)\n }\n \n+func (repo *repository) GetUserArticles(_ context.Context, boardID string) ([]bbs.ArticleRecord, error) {\n+\treturn repo.db.ReadBoardArticleRecordsFile(boardID)\n+}\n+\n func loadUserRecords(db *bbs.DB) ([]bbs.UserRecord, error) {\n \tuserRecords, err := db.ReadUserRecords()\n \tif err != nil {\n \t\tlogger.Errorf(\"get user rec error: %v\", err)\n \t\treturn nil, fmt.Errorf(\"failed to read user records: %w\", err)\n \t}\n-\tresults := make([] bbs.UserRecord, 0, len(userRecords))\n+\tresults := make([]bbs.UserRecord, 0, len(userRecords))\n \tfor _, rec := range userRecords {\n \t\tresults = append(results, &bbsUserRecord{rec})\n \t}\ndiff --git a\/internal\/usecase\/usecase.go b\/internal\/usecase\/usecase.go\nindex 7f97679..d67f6eb 100644\n--- a\/internal\/usecase\/usecase.go\n+++ b\/internal\/usecase\/usecase.go\n@@ -18,6 +18,8 @@ type Usecase interface {\n \tGetUserFavorites(ctx context.Context, userID string) ([]interface{}, error) \/\/ FIXME: use concrete type rather than []interface{}\n \t\/\/ GetUserInformation returns user info of a user\n \tGetUserInformation(ctx context.Context, userID string) (map[string]interface{}, error) \/\/ FIXME: use concrete type rather than map[string]interface{}\n+\t\/\/ GetUserArticles returns user's articles\n+\tGetUserArticles(ctx context.Context, userID string) ([]interface{}, error) \/\/ FIXME: use concrete type rather than []interface{}\n \n \t\/\/ board.go\n \t\/\/ GetBoardByID returns board record of board id\ndiff --git a\/internal\/usecase\/user.go b\/internal\/usecase\/user.go\nindex 970d5e5..3ef5fb8 100644\n--- a\/internal\/usecase\/user.go\n+++ b\/internal\/usecase\/user.go\n@@ -48,19 +48,53 @@ func (usecase *usecase) GetUserInformation(ctx context.Context, userID string) (\n \t\t\"number_of_login_days\": fmt.Sprintf(\"%d\", user.NumLoginDays()),\n \t\t\"number_of_posts\":      fmt.Sprintf(\"%d\", user.NumPosts()),\n \t\t\"number_of_badposts\":   fmt.Sprintf(\"%d\", user.NumBadPosts()),\n-\t\t\"money\":           fmt.Sprintf(\"%d\", user.Money()),\n-\t\t\"money_description\": getMoneyDiscription(user.Money()),\n-\t\t\"last_login_time\": user.LastLogin().Format(time.RFC3339),\n-\t\t\"last_login_ipv4\": user.LastHost(),\n-\t\t\"last_login_ip\":   user.LastHost(),\n-\t\t\"last_login_country\": user.LastCountry(),\n-\t\t\"mailbox_description\": user.MailboxDescription(),\n-\t\t\"chess_status\": user.ChessStatus(),\n-\t\t\"plan\":         user.Plan(),\n+\t\t\"money\":                fmt.Sprintf(\"%d\", user.Money()),\n+\t\t\"money_description\":    getMoneyDiscription(user.Money()),\n+\t\t\"last_login_time\":      user.LastLogin().Format(time.RFC3339),\n+\t\t\"last_login_ipv4\":      user.LastHost(),\n+\t\t\"last_login_ip\":        user.LastHost(),\n+\t\t\"last_login_country\":   user.LastCountry(),\n+\t\t\"mailbox_description\":  user.MailboxDescription(),\n+\t\t\"chess_status\":         user.ChessStatus(),\n+\t\t\"plan\":                 user.Plan(),\n \t}\n \treturn result, nil\n }\n \n+func (usecase *usecase) GetUserArticles(ctx context.Context, userID string) ([]interface{}, error) {\n+\tdataItems := []interface{}{}\n+\n+\t\/\/ Because there is no user\u2019s historical article data stored, first get all public boards, and then get user articles\n+\tboards := usecase.GetBoards(ctx, userID)\n+\n+\tfor _, board := range boards {\n+\t\tarticleRecords, err := usecase.repo.GetUserArticles(ctx, board.BoardId())\n+\t\tif err != nil {\n+\t\t\treturn nil, err\n+\t\t}\n+\n+\t\tfor index := range articleRecords {\n+\t\t\tif articleRecords[index].Owner() == userID {\n+\t\t\t\tdataItems = append(dataItems, map[string]interface{}{\n+\t\t\t\t\t\"board_id\":        \"\", \/\/ FIXME: use concrete value rather than \"\"\n+\t\t\t\t\t\"filename\":        articleRecords[index].Filename(),\n+\t\t\t\t\t\"modified_time\":   articleRecords[index].Modified(),\n+\t\t\t\t\t\"recommend_count\": articleRecords[index].Recommend(),\n+\t\t\t\t\t\"comment_count\":   0,  \/\/ FIXME: use concrete value rather than 0\n+\t\t\t\t\t\"post_date\":       \"\", \/\/ FIXME: use concrete value rather than \"\"\n+\t\t\t\t\t\"title\":           articleRecords[index].Title(),\n+\t\t\t\t\t\"money\":           articleRecords[index].Money(),\n+\t\t\t\t\t\"owner\":           articleRecords[index].Owner(),\n+\t\t\t\t\t\"aid\":             \"\", \/\/ FIXME: use concrete value rather than \"\"\n+\t\t\t\t\t\"url\":             \"\", \/\/ FIXME: use concrete value rather than \"\"\n+\t\t\t\t})\n+\t\t\t}\n+\t\t}\n+\t}\n+\n+\treturn dataItems, nil\n+}\n+\n func (usecase *usecase) parseFavoriteFolderItem(recs []bbs.FavoriteRecord) []interface{} {\n \tdataItems := []interface{}{}\n \tfor _, item := range recs {\n"}
]
  