From 4d40159772923282973b8591a96c52d93fa6b751 Mon Sep 17 00:00:00 2001 From: OXYgen Date: Mon, 11 Nov 2024 16:41:51 +0500 Subject: [PATCH 1/4] Added AddTask and DeleteTask. Without db tests --- internal/contests/delivery.go | 2 ++ internal/contests/delivery/grpc/handlers.go | 16 ++++++++++ internal/contests/pg_repository.go | 2 ++ internal/contests/repository/pg_repository.go | 29 +++++++++++++++++++ internal/contests/usecase.go | 2 ++ internal/contests/usecase/usecase.go | 8 +++++ 6 files changed, 59 insertions(+) diff --git a/internal/contests/delivery.go b/internal/contests/delivery.go index bbb81f3..4314ccd 100644 --- a/internal/contests/delivery.go +++ b/internal/contests/delivery.go @@ -10,4 +10,6 @@ type ContestHandlers interface { CreateContest(ctx context.Context, req *contestv1.CreateContestRequest) (*contestv1.CreateContestResponse, error) ReadContest(ctx context.Context, req *contestv1.ReadContestRequest) (*contestv1.ReadContestResponse, error) DeleteContest(ctx context.Context, req *contestv1.DeleteContestRequest) (*emptypb.Empty, error) + AddTask(ctx context.Context, req *contestv1.AddTaskRequest) (*contestv1.AddTaskResponse, error) + DeleteTask(ctx context.Context, req *contestv1.DeleteTaskRequest) (*emptypb.Empty, error) } diff --git a/internal/contests/delivery/grpc/handlers.go b/internal/contests/delivery/grpc/handlers.go index 67141e4..ceb381f 100644 --- a/internal/contests/delivery/grpc/handlers.go +++ b/internal/contests/delivery/grpc/handlers.go @@ -49,3 +49,19 @@ func (h *ContestHandlers) DeleteContest(ctx context.Context, req *contestv1.Dele } return &emptypb.Empty{}, nil } + +func (h *ContestHandlers) AddTask(ctx context.Context, req *contestv1.AddTaskRequest) (*contestv1.AddTaskResponse, error) { + id, err := h.contestUC.AddTask(ctx, req.GetContestId(), req.GetProblemId()) + if err != nil { + return nil, err + } + return &contestv1.AddTaskResponse{Id: id}, nil +} + +func (h *ContestHandlers) DeleteTask(ctx context.Context, req *contestv1.DeleteTaskRequest) (*emptypb.Empty, error) { + err := h.contestUC.DeleteTask(ctx, req.GetTaskId()) + if err != nil { + return nil, err + } + return &emptypb.Empty{}, nil +} diff --git a/internal/contests/pg_repository.go b/internal/contests/pg_repository.go index 1dc1368..ccfdc24 100644 --- a/internal/contests/pg_repository.go +++ b/internal/contests/pg_repository.go @@ -9,4 +9,6 @@ type ContestRepository interface { CreateContest(ctx context.Context, title string) (int32, error) ReadContestById(ctx context.Context, id int32) (*models.Contest, error) DeleteContest(ctx context.Context, id int32) error + AddTask(ctx context.Context, contestId int32, taskId int32) (int32, error) + DeleteTask(ctx context.Context, taskId int32) error } diff --git a/internal/contests/repository/pg_repository.go b/internal/contests/repository/pg_repository.go index e3f1c82..abd9013 100644 --- a/internal/contests/repository/pg_repository.go +++ b/internal/contests/repository/pg_repository.go @@ -68,6 +68,35 @@ func (r *ContestRepository) DeleteContest(ctx context.Context, id int32) error { return nil } +const addTaskQuery = "INSERT INTO task (problem_id, contest_id,position) VALUES (?, ?,(SELECT COALESCE(MAX(position),0) + 1 FROM task) ) RETURNING id" + +func (r *ContestRepository) AddTask(ctx context.Context, contestId int32, taskId int32) (int32, error) { + query := r.db.Rebind(addTaskQuery) + rows, err := r.db.QueryxContext(ctx, query, taskId, contestId) + if err != nil { + return 0, handlePgErr(err) + } + defer rows.Close() + var id int32 + rows.Next() + err = rows.Scan(&id) + if err != nil { + return 0, handlePgErr(err) + } + return id, nil +} + +const deleteTaskQuery = "DELETE FROM task WHERE id=?" + +func (r *ContestRepository) DeleteTask(ctx context.Context, taskId int32) error { + query := r.db.Rebind(deleteTaskQuery) + _, err := r.db.ExecContext(ctx, query, taskId) + if err != nil { + return handlePgErr(err) + } + return nil +} + func handlePgErr(err error) error { var pgErr *pgconn.PgError if !errors.As(err, &pgErr) { diff --git a/internal/contests/usecase.go b/internal/contests/usecase.go index c076721..18d230a 100644 --- a/internal/contests/usecase.go +++ b/internal/contests/usecase.go @@ -9,4 +9,6 @@ type ContestUseCase interface { CreateContest(ctx context.Context, title string) (int32, error) ReadContestById(ctx context.Context, id int32) (*models.Contest, error) DeleteContest(ctx context.Context, id int32) error + AddTask(ctx context.Context, contestId int32, taskId int32) (int32, error) + DeleteTask(ctx context.Context, taskId int32) error } diff --git a/internal/contests/usecase/usecase.go b/internal/contests/usecase/usecase.go index 5437e17..5da3654 100644 --- a/internal/contests/usecase/usecase.go +++ b/internal/contests/usecase/usecase.go @@ -29,3 +29,11 @@ func (uc *ContestUseCase) ReadContestById(ctx context.Context, id int32) (*model func (uc *ContestUseCase) DeleteContest(ctx context.Context, id int32) error { return uc.contestRepo.DeleteContest(ctx, id) } + +func (uc *ContestUseCase) AddTask(ctx context.Context, contestId int32, taskId int32) (id int32, err error) { + return uc.contestRepo.AddTask(ctx, contestId, taskId) +} + +func (uc *ContestUseCase) DeleteTask(ctx context.Context, taskId int32) error { + return uc.contestRepo.DeleteTask(ctx, taskId) +} From f7dd1bc8068b81ad6b0ec4aa9031e43e9a5afda6 Mon Sep 17 00:00:00 2001 From: OXYgen Date: Thu, 14 Nov 2024 17:09:52 +0500 Subject: [PATCH 2/4] Added AddParticipant and DeleteParticipant --- internal/contests/delivery.go | 2 ++ internal/contests/delivery/grpc/handlers.go | 16 ++++++++++ internal/contests/pg_repository.go | 2 ++ internal/contests/repository/pg_repository.go | 31 ++++++++++++++++++- internal/contests/usecase.go | 2 ++ internal/contests/usecase/usecase.go | 8 +++++ 6 files changed, 60 insertions(+), 1 deletion(-) diff --git a/internal/contests/delivery.go b/internal/contests/delivery.go index 4314ccd..874d361 100644 --- a/internal/contests/delivery.go +++ b/internal/contests/delivery.go @@ -12,4 +12,6 @@ type ContestHandlers interface { DeleteContest(ctx context.Context, req *contestv1.DeleteContestRequest) (*emptypb.Empty, error) AddTask(ctx context.Context, req *contestv1.AddTaskRequest) (*contestv1.AddTaskResponse, error) DeleteTask(ctx context.Context, req *contestv1.DeleteTaskRequest) (*emptypb.Empty, error) + AddParticipant(ctx context.Context, req *contestv1.AddParticipantRequest) (*contestv1.AddParticipantResponse, error) + DeleteParticipant(ctx context.Context, req *contestv1.DeleteParticipantRequest) (*emptypb.Empty, error) } diff --git a/internal/contests/delivery/grpc/handlers.go b/internal/contests/delivery/grpc/handlers.go index ceb381f..36047a4 100644 --- a/internal/contests/delivery/grpc/handlers.go +++ b/internal/contests/delivery/grpc/handlers.go @@ -65,3 +65,19 @@ func (h *ContestHandlers) DeleteTask(ctx context.Context, req *contestv1.DeleteT } return &emptypb.Empty{}, nil } + +func (h *ContestHandlers) AddParticipant(ctx context.Context, req *contestv1.AddParticipantRequest) (*contestv1.AddParticipantResponse, error) { + id, err := h.contestUC.AddParticipant(ctx, req.GetContestId(), req.GetUserId()) + if err != nil { + return nil, err + } + return &contestv1.AddParticipantResponse{Id: id}, nil +} + +func (h *ContestHandlers) DeleteParticipant(ctx context.Context, req *contestv1.DeleteParticipantRequest) (*emptypb.Empty, error) { + err := h.contestUC.DeleteParticipant(ctx, req.GetParticipantId()) + if err != nil { + return nil, err + } + return &emptypb.Empty{}, nil +} diff --git a/internal/contests/pg_repository.go b/internal/contests/pg_repository.go index ccfdc24..0443e32 100644 --- a/internal/contests/pg_repository.go +++ b/internal/contests/pg_repository.go @@ -11,4 +11,6 @@ type ContestRepository interface { DeleteContest(ctx context.Context, id int32) error AddTask(ctx context.Context, contestId int32, taskId int32) (int32, error) DeleteTask(ctx context.Context, taskId int32) error + AddParticipant(ctx context.Context, contestId int32, userId int32) (int32, error) + DeleteParticipant(ctx context.Context, participantId int32) error } diff --git a/internal/contests/repository/pg_repository.go b/internal/contests/repository/pg_repository.go index abd9013..8d28144 100644 --- a/internal/contests/repository/pg_repository.go +++ b/internal/contests/repository/pg_repository.go @@ -68,7 +68,7 @@ func (r *ContestRepository) DeleteContest(ctx context.Context, id int32) error { return nil } -const addTaskQuery = "INSERT INTO task (problem_id, contest_id,position) VALUES (?, ?,(SELECT COALESCE(MAX(position),0) + 1 FROM task) ) RETURNING id" +const addTaskQuery = "INSERT INTO task (problem_id, contest_id,position) VALUES (?, ?,(SELECT COALESCE(MAX(position),0) + 1 FROM task)) RETURNING id" func (r *ContestRepository) AddTask(ctx context.Context, contestId int32, taskId int32) (int32, error) { query := r.db.Rebind(addTaskQuery) @@ -97,6 +97,35 @@ func (r *ContestRepository) DeleteTask(ctx context.Context, taskId int32) error return nil } +const addParticipantQuery = "INSERT INTO participant (user_id ,contest_id, name) VALUES (?, ?, ?) RETURNING id" + +func (r *ContestRepository) AddParticipant(ctx context.Context, contestId int32, userId int32) (int32, error) { + query := r.db.Rebind(addParticipantQuery) + name := "" + rows, err := r.db.QueryxContext(ctx, query, contestId, userId, name) + if err != nil { + return 0, handlePgErr(err) + } + defer rows.Close() + var id int32 + err = rows.Scan(&id) + if err != nil { + return 0, err + } + return id, nil +} + +const deleteParticipantQuery = "DELETE FROM participant WHERE id=?" + +func (r *ContestRepository) DeleteParticipant(ctx context.Context, participantId int32) error { + query := r.db.Rebind(deleteParticipantQuery) + _, err := r.db.ExecContext(ctx, query, participantId) + if err != nil { + return handlePgErr(err) + } + return nil +} + func handlePgErr(err error) error { var pgErr *pgconn.PgError if !errors.As(err, &pgErr) { diff --git a/internal/contests/usecase.go b/internal/contests/usecase.go index 18d230a..7892518 100644 --- a/internal/contests/usecase.go +++ b/internal/contests/usecase.go @@ -11,4 +11,6 @@ type ContestUseCase interface { DeleteContest(ctx context.Context, id int32) error AddTask(ctx context.Context, contestId int32, taskId int32) (int32, error) DeleteTask(ctx context.Context, taskId int32) error + AddParticipant(ctx context.Context, contestId int32, userId int32) (int32, error) + DeleteParticipant(ctx context.Context, participantId int32) error } diff --git a/internal/contests/usecase/usecase.go b/internal/contests/usecase/usecase.go index 5da3654..633bab3 100644 --- a/internal/contests/usecase/usecase.go +++ b/internal/contests/usecase/usecase.go @@ -37,3 +37,11 @@ func (uc *ContestUseCase) AddTask(ctx context.Context, contestId int32, taskId i func (uc *ContestUseCase) DeleteTask(ctx context.Context, taskId int32) error { return uc.contestRepo.DeleteTask(ctx, taskId) } + +func (uc *ContestUseCase) AddParticipant(ctx context.Context, contestId int32, userId int32) (id int32, err error) { + return uc.contestRepo.AddParticipant(ctx, contestId, userId) +} + +func (uc *ContestUseCase) DeleteParticipant(ctx context.Context, participantId int32) error { + return uc.contestRepo.DeleteParticipant(ctx, participantId) +} From e201b12db5249ec2e5d13cfa81af58efbd6f2400 Mon Sep 17 00:00:00 2001 From: OXYgen Date: Thu, 14 Nov 2024 20:25:47 +0500 Subject: [PATCH 3/4] Added tests to new AddTask, DeleteTask, AddParticipant, DeleteParticipant --- internal/contests/repository/pg_repository.go | 7 +- .../contests/repository/pg_repository_test.go | 96 +++++++++++++++++++ 2 files changed, 100 insertions(+), 3 deletions(-) diff --git a/internal/contests/repository/pg_repository.go b/internal/contests/repository/pg_repository.go index 8d28144..2777562 100644 --- a/internal/contests/repository/pg_repository.go +++ b/internal/contests/repository/pg_repository.go @@ -68,11 +68,11 @@ func (r *ContestRepository) DeleteContest(ctx context.Context, id int32) error { return nil } -const addTaskQuery = "INSERT INTO task (problem_id, contest_id,position) VALUES (?, ?,(SELECT COALESCE(MAX(position),0) + 1 FROM task)) RETURNING id" +const addTaskQuery = "INSERT INTO task (problem_id, contest_id, position) VALUES (?, ?,(SELECT COALESCE(MAX(position),0) + 1 FROM task)) RETURNING id" -func (r *ContestRepository) AddTask(ctx context.Context, contestId int32, taskId int32) (int32, error) { +func (r *ContestRepository) AddTask(ctx context.Context, contestId int32, problem_id int32) (int32, error) { query := r.db.Rebind(addTaskQuery) - rows, err := r.db.QueryxContext(ctx, query, taskId, contestId) + rows, err := r.db.QueryxContext(ctx, query, problem_id, contestId) if err != nil { return 0, handlePgErr(err) } @@ -108,6 +108,7 @@ func (r *ContestRepository) AddParticipant(ctx context.Context, contestId int32, } defer rows.Close() var id int32 + rows.Next() err = rows.Scan(&id) if err != nil { return 0, err diff --git a/internal/contests/repository/pg_repository_test.go b/internal/contests/repository/pg_repository_test.go index beee36a..2440713 100644 --- a/internal/contests/repository/pg_repository_test.go +++ b/internal/contests/repository/pg_repository_test.go @@ -56,3 +56,99 @@ func TestContestRepository_DeleteContest(t *testing.T) { require.NoError(t, err) }) } + +func TestContestRepository_AddTask(t *testing.T) { + t.Parallel() + + db, mock, err := sqlmock.New(sqlmock.QueryMatcherOption(sqlmock.QueryMatcherEqual)) + require.NoError(t, err) + defer db.Close() + + sqlxDB := sqlx.NewDb(db, "sqlmock") + defer sqlxDB.Close() + + contestRepo := NewContestRepository(sqlxDB, zap.NewNop()) + + t.Run("valid task additional", func(t *testing.T) { + taskId := int32(1) + contestId := int32(1) + + rows := sqlmock.NewRows([]string{"id"}).AddRow(1) + + mock.ExpectQuery(sqlxDB.Rebind(addTaskQuery)).WithArgs(taskId, contestId).WillReturnRows(rows) + + id, err := contestRepo.AddTask(context.Background(), contestId, taskId) + + require.NoError(t, err) + require.Equal(t, int32(1), id) + + }) +} + +func TestContestRepository_DeleteTask(t *testing.T) { + t.Parallel() + db, mock, err := sqlmock.New(sqlmock.QueryMatcherOption(sqlmock.QueryMatcherEqual)) + require.NoError(t, err) + defer db.Close() + + sqlxDB := sqlx.NewDb(db, "sqlmock") + defer sqlxDB.Close() + contestRepo := NewContestRepository(sqlxDB, zap.NewNop()) + t.Run("valid task deletion", func(t *testing.T) { + id := int32(1) + rows := sqlmock.NewResult(1, 1) + + mock.ExpectExec(sqlxDB.Rebind(deleteTaskQuery)).WithArgs(id).WillReturnResult(rows) + + err = contestRepo.DeleteTask(context.Background(), id) + require.NoError(t, err) + }) +} + +func TestContestRepository_AddParticipant(t *testing.T) { + t.Parallel() + db, mock, err := sqlmock.New(sqlmock.QueryMatcherOption(sqlmock.QueryMatcherEqual)) + require.NoError(t, err) + defer db.Close() + + sqlxDB := sqlx.NewDb(db, "sqlmock") + defer sqlxDB.Close() + contestRepo := NewContestRepository(sqlxDB, zap.NewNop()) + + t.Run("valid participant addition", func(t *testing.T) { + contestId := int32(1) + userId := int32(1) + name := "" + + rows := sqlmock.NewRows([]string{"id"}).AddRow(1) + + mock.ExpectQuery(sqlxDB.Rebind(addParticipantQuery)).WithArgs(contestId, userId, name).WillReturnRows(rows) + + id, err := contestRepo.AddParticipant(context.Background(), contestId, userId) + + require.NoError(t, err) + require.Equal(t, int32(1), id) + }) +} + +func TestContestRepository_DeleteParticipant(t *testing.T) { + t.Parallel() + db, mock, err := sqlmock.New(sqlmock.QueryMatcherOption(sqlmock.QueryMatcherEqual)) + require.NoError(t, err) + defer db.Close() + + sqlxDB := sqlx.NewDb(db, "sqlmock") + defer sqlxDB.Close() + + contestRepo := NewContestRepository(sqlxDB, zap.NewNop()) + + t.Run("valid participant deletion", func(t *testing.T) { + id := int32(1) + rows := sqlmock.NewResult(1, 1) + + mock.ExpectExec(sqlxDB.Rebind(deleteParticipantQuery)).WithArgs(id).WillReturnResult(rows) + + err = contestRepo.DeleteParticipant(context.Background(), id) + require.NoError(t, err) + }) +} From a1ebd514040b1e0a3ea82acdeba094246b38afe5 Mon Sep 17 00:00:00 2001 From: OXYgen Date: Fri, 15 Nov 2024 13:39:22 +0500 Subject: [PATCH 4/4] Fixed bug in AddTask, fixed finding new task position --- internal/contests/repository/pg_repository.go | 4 ++-- internal/contests/repository/pg_repository_test.go | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/internal/contests/repository/pg_repository.go b/internal/contests/repository/pg_repository.go index 2777562..48ff720 100644 --- a/internal/contests/repository/pg_repository.go +++ b/internal/contests/repository/pg_repository.go @@ -68,11 +68,11 @@ func (r *ContestRepository) DeleteContest(ctx context.Context, id int32) error { return nil } -const addTaskQuery = "INSERT INTO task (problem_id, contest_id, position) VALUES (?, ?,(SELECT COALESCE(MAX(position),0) + 1 FROM task)) RETURNING id" +const addTaskQuery = "INSERT INTO task (problem_id, contest_id, position) VALUES (?, ?,COALESCE(SELECT MAX(position) FROM task WHERE contest_id = ? ,0) + 1) RETURNING id" func (r *ContestRepository) AddTask(ctx context.Context, contestId int32, problem_id int32) (int32, error) { query := r.db.Rebind(addTaskQuery) - rows, err := r.db.QueryxContext(ctx, query, problem_id, contestId) + rows, err := r.db.QueryxContext(ctx, query, problem_id, contestId, contestId) if err != nil { return 0, handlePgErr(err) } diff --git a/internal/contests/repository/pg_repository_test.go b/internal/contests/repository/pg_repository_test.go index 2440713..be2cccd 100644 --- a/internal/contests/repository/pg_repository_test.go +++ b/internal/contests/repository/pg_repository_test.go @@ -75,7 +75,7 @@ func TestContestRepository_AddTask(t *testing.T) { rows := sqlmock.NewRows([]string{"id"}).AddRow(1) - mock.ExpectQuery(sqlxDB.Rebind(addTaskQuery)).WithArgs(taskId, contestId).WillReturnRows(rows) + mock.ExpectQuery(sqlxDB.Rebind(addTaskQuery)).WithArgs(taskId, contestId, contestId).WillReturnRows(rows) id, err := contestRepo.AddTask(context.Background(), contestId, taskId)