[
    {
      "base_commit":"ff1b2cc4fe82a4ccf629fdbd21a144bbfb1f9139",
      "created_at":"2021-03-08T05:13:12Z",
      "hints_text":"",
      "instance_id":"ptt-official-app__ptt-backend-123",
      "issue_numbers":[
        59
      ],
      "meta":{
        "commit_name":"head_commit",
        "failed_lite_validators":[
          "has_hyperlinks",
          "has_many_modified_files",
          "has_many_hunks"
        ],
        "has_test_patch":true,
        "is_lite":false,
        "num_modified_files":5,
        "num_modified_lines":87,
        "pr_author":"whitefloor",
        "pr_labels":[
  
        ]
      },
      "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",
      "pr_description":"\u5be6\u4f5c\u500b\u4eba\u770b\u677f\u6b77\u53f2\u6587\u7ae0\n<!-- \u539f\u5247\u4e0a\u4e0d\u63a5\u53d7\u6c92\u6709 Issue ID \u7684 PR\u3002 -->\r\n<!-- We don't accept PRs which has no Issue ID. -->\r\n\r\n## \ud83d\udc4f \u89e3\u6c7a\u6389\u7684 issue \/ Resolved Issues\r\n- close #59 \r\n\r\n## \ud83d\udcdd \u76f8\u95dc\u7684 issue \/ Related Issues\r\n- #59 \r\n\r\n## \u26cf \u8b8a\u66f4\u5167\u5bb9 \/ Details of Changes\r\n<!-- \u7c21\u8981\u9673\u5217\u60a8\u7684\u4fee\u6539 -->\r\n<!-- List down your changes concisely -->\r\n- delivery\r\n1. \u65b0\u589euser router: articles\r\n2. \u65b0\u589edelivery method:getUserArticles\r\n3. \u65b0\u589edelivery test:GetUserArticles(mock method)\/TestGetUserArticles\r\n\r\n- usecase\r\n1. \u65b0\u589eusecase method:GetUserArticles\r\n2. \u65b0\u589eusecase test:TestGetUserArticles\r\n\r\n- repository\r\n1. \u65b0\u589erepository method:GetUserArticles\r\n2. \u65b0\u589erepository test:GetUserArticles(mock method)",
      "problem_statement":"[\u4e3b\u7dda] [PTT] \u5be6\u4f5c\u500b\u4eba\u770b\u677f\u6b77\u53f2\u6587\u7ae0\n### \u5be6\u4f5c\u7d30\u7bc0 \/ Details of Implement\r\n\u6839\u64da\u76ee\u524d\u67b6\u69cb\uff0c\u61c9\u8a72\u5206\u6210 `delivery\/http` `usecase` `repository` \u4e09\u500b\u5b50 ISSUE \u4e0b\u53bb\u5be6\u4f5c\u3002\r\n* http \u90e8\u5206\u8acb\u4f9d\u7167\u6587\u4ef6\u5be6\u4f5c\u4e26\u4e14\u6709\u521d\u6b65\u7684 testcode\u3002\r\n* usecase \u90e8\u5206\u61c9\u8a72\u662f\u76f4\u63a5\u5c07 repository \u7684\u7d50\u679c\u8f38\u51fa\u5373\u53ef\u3002\r\n* repository \u90e8\u5206\u6b77\u53f2\u6587\u7ae0\u529f\u80fd\u61c9\u8a72\u7531 go-bbs \u5132\u5b58\u9084\u662f Ptt-backend \u6383\u63cf\u5f8c\u66ab\u5b58\u9700\u8981\u8a0e\u8ad6\uff0c\u66ab\u5b9a\u7531 go-bbs \u5132\u5b58\u3002\r\n\r\n### \u671f\u7a0b \/ Schedule\r\n- \u8a0e\u8ad6\u6642\u9593\uff1a \u4e00\u9031, \u5230 2\/20\r\n- \u5be6\u4f5c\u6642\u9593\uff1a \u4e00\u9031, \u5230 2\/27\r\n- \u78ba\u8a8d\u6642\u9593\uff1a \u4e00\u9031, \u5230 3\/5\r\n\r\n### \u76f8\u95dc\u6587\u4ef6 \/ Documents\r\n[PTT \u5f8c\u7aef\u7cfb\u7d71\u5354\u5b9a](https:\/\/docs.google.com\/document\/d\/18DsZOyrlr5BIl2kKxZH7P2QxFLG02xL2SO0PzVHVY3k)",
      "pull_number":123,
      "repo":"Ptt-official-app\/Ptt-backend",
      "test_patch":"diff --git a\/internal\/delivery\/http\/route_users_mock_test.go b\/internal\/delivery\/http\/route_users_mock_test.go\nindex 66cdfd6..8259b0b 100644\n--- a\/internal\/delivery\/http\/route_users_mock_test.go\n+++ b\/internal\/delivery\/http\/route_users_mock_test.go\n@@ -22,6 +22,10 @@ func (usecase *MockUsecase) GetUserInformation(ctx context.Context, userID strin\n \treturn result, nil\n }\n \n+func (usecase *MockUsecase) GetUserArticles(ctx context.Context, userID string) ([]interface{}, error) {\n+\treturn nil, nil\n+}\n+\n type MockUserRecord struct {\n \tuserId string\n }\ndiff --git a\/internal\/delivery\/http\/route_users_test.go b\/internal\/delivery\/http\/route_users_test.go\nindex d35dd11..59079ea 100644\n--- a\/internal\/delivery\/http\/route_users_test.go\n+++ b\/internal\/delivery\/http\/route_users_test.go\n@@ -88,3 +88,37 @@ func TestParseUserPath(t *testing.T) {\n \t}\n \n }\n+\n+func TestGetUserArticles(t *testing.T) {\n+\n+\tuserID := \"id\"\n+\tmockUsecase := NewMockUsecase()\n+\tmockDelivery := NewHTTPDelivery(mockUsecase)\n+\n+\treq, err := http.NewRequest(\"GET\", \"\/v1\/users\/user\/articles\", nil)\n+\tif err != nil {\n+\t\tt.Fatal(err)\n+\t}\n+\n+\ttoken := mockUsecase.CreateAccessTokenWithUsername(userID)\n+\tt.Logf(\"testing token: %v\", token)\n+\treq.Header.Add(\"Authorization\", \"bearer \"+token)\n+\n+\trr := httptest.NewRecorder()\n+\tr := http.NewServeMux()\n+\tr.HandleFunc(\"\/v1\/users\/\", mockDelivery.routeUsers)\n+\tr.ServeHTTP(rr, req)\n+\n+\tif status := rr.Code; status != http.StatusOK {\n+\t\tt.Errorf(\"handler returned wrong status code: got %v want %v\",\n+\t\t\tstatus, http.StatusOK)\n+\t}\n+\n+\tresponsedMap := map[string]interface{}{}\n+\tjson.Unmarshal(rr.Body.Bytes(), &responsedMap)\n+\tt.Logf(\"got response %v\", rr.Body.String())\n+\tif responsedMap[\"data\"] == nil {\n+\t\tt.Errorf(\"handler returned unexpected body, got %v want not nil\",\n+\t\t\trr.Body.String())\n+\t}\n+}\ndiff --git a\/internal\/usecase\/board_mock_test.go b\/internal\/usecase\/board_mock_test.go\nindex 547d616..7ae85a7 100644\n--- a\/internal\/usecase\/board_mock_test.go\n+++ b\/internal\/usecase\/board_mock_test.go\n@@ -148,3 +148,31 @@ func (m *MockLoginsLimitedBoardRecord) PostLimitLogins() uint8 { return 0 }\n type MockBadPostLimitedBoardRecord struct{}\n \n func (m *MockBadPostLimitedBoardRecord) PostLimitBadPost() uint8 { return 0 }\n+\n+func (repo *MockRepository) GetUserArticles(ctx context.Context, boardID string) ([]bbs.ArticleRecord, error) {\n+\tarticleRecords := []MockArticleRecord{\n+\t\t{\n+\t\t\tfilename:       \"\",\n+\t\t\tmodified:       time.Time{},\n+\t\t\trecommendCount: 10,\n+\t\t\towner:          \"user\",\n+\t\t\tdate:           \"\",\n+\t\t\ttitle:          \"[\u8a0e\u8ad6] \u85aa\u6c34\u592a\u5c11\",\n+\t\t\tmoney:          0,\n+\t\t},\n+\t\t{\n+\t\t\tfilename:       \"\",\n+\t\t\tmodified:       time.Time{},\n+\t\t\trecommendCount: -20,\n+\t\t\towner:          \"9487\",\n+\t\t\tdate:           \"\",\n+\t\t\ttitle:          \"[\u554f\u984c] \u6211\u4e0d\u6703\u5beb\u7a0b\u5f0f\",\n+\t\t\tmoney:          0,\n+\t\t},\n+\t}\n+\tresult := make([]bbs.ArticleRecord, len(articleRecords))\n+\tfor i, v := range articleRecords {\n+\t\tresult[i] = v\n+\t}\n+\treturn result, nil\n+}\ndiff --git a\/internal\/usecase\/user_test.go b\/internal\/usecase\/user_test.go\nindex 0842db8..46405b6 100644\n--- a\/internal\/usecase\/user_test.go\n+++ b\/internal\/usecase\/user_test.go\n@@ -1,10 +1,10 @@\n package usecase\n \n import (\n-\t\"github.com\/Ptt-official-app\/Ptt-backend\/internal\/config\"\n-\n \t\"context\"\n \t\"testing\"\n+\n+\t\"github.com\/Ptt-official-app\/Ptt-backend\/internal\/config\"\n )\n \n func TestGetUserByID(t *testing.T) {\n@@ -36,3 +36,23 @@ func TestGetUserByID(t *testing.T) {\n \t}\n \n }\n+\n+func TestGetUserArticles(t *testing.T) {\n+\n+\tuserID := \"user\"\n+\tmockRepository := &MockRepository{}\n+\tmockUsecase := NewUsecase(&config.Config{}, mockRepository)\n+\n+\tdataItems, err := mockUsecase.GetUserArticles(context.TODO(), userID)\n+\n+\tif err != nil {\n+\t\tt.Errorf(\"GetUserArticles with userID excepted nil error, got %v\", err)\n+\t\treturn\n+\t}\n+\n+\tif dataItems == nil {\n+\t\tt.Errorf(\"GetUserArticles with userID excepted not nil, got %v\", dataItems)\n+\t\treturn\n+\t}\n+\n+}\n",
      "version":"unknown",
      "environment_setup_commit":"ff1b2cc4fe82a4ccf629fdbd21a144bbfb1f9139",
      "PASS_TO_PASS":[
        "TestGetBoardArticlesBadRequest",
        "TestGetPopularBoardList",
        "TestGetClassesList",
        "TestGetPopularArticles",
        "TestGetUserInformation",
        "TestParseUserPath",
        "TestGetUsers",
        "TestNewConfig"
      ],
      "FAIL_TO_PASS":[
        "TestGetUserArticles",
        "TestSearchArticles",
        "TestGetUserByID"
      ],
      "install_config":{
        "test_cmd":[
          "CGO_ENABLED=1 go test -v .\/internal\/delivery\/http\/... .\/internal\/usecase\/... .\/internal\/repository\/... .\/internal\/config\/... -cover -race"
        ],
        "log_parser":"parse_log_gotest",
        "docker_specs":{
          "go_version":"1.19.13"
        },
        "image_name": "go_1.19.13",
        "install":[
          "export HOME=\/root",
          "export XDG_CACHE_HOME=\/root\/.cache",
          "go mod download",
          "sed -i 's\/%w\/%v\/g' internal\/delivery\/http\/route_boards_test.go",
          "sed -i 's\/%w\/%v\/g' internal\/delivery\/http\/route_classes_test.go",
          "sed -i 's\/%w\/%v\/g' internal\/delivery\/http\/route_popular_articles_test.go",
          "sed -i 's\/%w\/%v\/g' internal\/delivery\/http\/route_users_test.go",
          "go mod download"
        ]
      },
      "lang":"Go"
    }
  ]