From a5027cc6b424488a5ef4282f7635ed22a2e21b68 Mon Sep 17 00:00:00 2001 From: Kosmatoff Date: Wed, 20 Dec 2023 16:30:39 +0300 Subject: [PATCH 1/2] add test grpc auth --- .../auth/delivery/grpc/grpc_handler_test.go | 486 ++++++++++++++++++ 1 file changed, 486 insertions(+) diff --git a/internal/microservices/auth/delivery/grpc/grpc_handler_test.go b/internal/microservices/auth/delivery/grpc/grpc_handler_test.go index 21e034e4..568c8af1 100644 --- a/internal/microservices/auth/delivery/grpc/grpc_handler_test.go +++ b/internal/microservices/auth/delivery/grpc/grpc_handler_test.go @@ -1 +1,487 @@ package grpc + +import ( + "context" + "errors" + "github.com/go-park-mail-ru/2023_2_Hamster/internal/common/logger" + "github.com/go-park-mail-ru/2023_2_Hamster/internal/microservices/auth" + proto "github.com/go-park-mail-ru/2023_2_Hamster/internal/microservices/auth/delivery/grpc/generated" + mocks "github.com/go-park-mail-ru/2023_2_Hamster/internal/microservices/auth/mocks" + "github.com/go-park-mail-ru/2023_2_Hamster/internal/models" + "github.com/golang/mock/gomock" + "github.com/google/uuid" + "github.com/stretchr/testify/assert" + "google.golang.org/grpc/codes" + "google.golang.org/grpc/status" + "testing" +) + +func TestSignUp(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedID := uuid.New() + expectedLogin := "testuser" + expectedUsername := "Test User" + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + mockAuthServices.EXPECT(). + SignUp(gomock.Any(), gomock.Any()). + Return(expectedID, expectedLogin, expectedUsername, nil) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + signUpRequest := &proto.SignUpRequest{ + Login: expectedLogin, + Username: expectedUsername, + Password: "testpassword", + } + + signUpResponse, err := authGRPCInstance.SignUp(context.Background(), signUpRequest) + + assert.NoError(t, err) + assert.NotNil(t, signUpResponse) + + assert.Equal(t, "200", signUpResponse.Status) + assert.Equal(t, expectedID.String(), signUpResponse.Body.Id) + assert.Equal(t, expectedLogin, signUpResponse.Body.Login) + assert.Equal(t, expectedUsername, signUpResponse.Body.Username) +} + +func TestSignUp_ErrorCase(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedLogin := "testuser" + expectedUsername := "Test User" + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + expectedError := errors.New("signup failed") + mockAuthServices.EXPECT(). + SignUp(gomock.Any(), gomock.Any()). + Return(uuid.Nil, "", "", expectedError) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + signUpRequest := &proto.SignUpRequest{ + Login: expectedLogin, + Username: expectedUsername, + Password: "testpassword", + } + + signUpResponse, err := authGRPCInstance.SignUp(context.Background(), signUpRequest) + + assert.Error(t, err) + assert.Nil(t, signUpResponse) + +} + +func TestSignUp_UserAlreadyExistsError(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedLogin := "testuser" + expectedUsername := "Test User" + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + expectedError := &models.UserAlreadyExistsError{} + mockAuthServices.EXPECT(). + SignUp(gomock.Any(), gomock.Any()). + Return(uuid.Nil, "", "", expectedError) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + signUpRequest := &proto.SignUpRequest{ + Login: expectedLogin, + Username: expectedUsername, + Password: "testpassword", + } + + signUpResponse, err := authGRPCInstance.SignUp(context.Background(), signUpRequest) + + assert.Error(t, err) + assert.Nil(t, signUpResponse) + + var errUserAlreadyExists *models.UserAlreadyExistsError + assert.Equal(t, "user already exists", errUserAlreadyExists.Error()) +} + +func TestLogin_Success(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedLogin := "testuser" + expectedUsername := "Test User" + expectedID := uuid.New() + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + mockAuthServices.EXPECT(). + Login(gomock.Any(), gomock.Any(), gomock.Any()). + Return(expectedID, expectedLogin, expectedUsername, nil) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + loginRequest := &proto.LoginRequest{ + Login: expectedLogin, + Password: "testpassword", + } + + loginResponse, err := authGRPCInstance.Login(context.Background(), loginRequest) + + assert.NoError(t, err) + assert.NotNil(t, loginResponse) + + assert.Equal(t, "200", loginResponse.Status) + assert.Equal(t, expectedID.String(), loginResponse.Body.Id) + assert.Equal(t, expectedLogin, loginResponse.Body.Login) + assert.Equal(t, expectedUsername, loginResponse.Body.Username) +} + +func TestLogin_UserNotFound(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedLogin := "nonexistentuser" + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + expectedError := &models.NoSuchUserError{} + mockAuthServices.EXPECT(). + Login(gomock.Any(), gomock.Eq(expectedLogin), gomock.Any()). + Return(uuid.Nil, "", "", expectedError) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + loginRequest := &proto.LoginRequest{ + Login: expectedLogin, + Password: "testpassword", + } + + loginResponse, err := authGRPCInstance.Login(context.Background(), loginRequest) + + assert.Error(t, err) + assert.Nil(t, loginResponse) + + assert.Equal(t, codes.NotFound, status.Code(err)) +} + +func TestLogin_IncorrectPassword(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedLogin := "testuser" + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + expectedError := &models.IncorrectPasswordError{} + mockAuthServices.EXPECT(). + Login(gomock.Any(), gomock.Eq(expectedLogin), gomock.Any()). + Return(uuid.Nil, "", "", expectedError) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + loginRequest := &proto.LoginRequest{ + Login: expectedLogin, + Password: "wrongpassword", + } + + loginResponse, err := authGRPCInstance.Login(context.Background(), loginRequest) + + assert.Error(t, err) + assert.Nil(t, loginResponse) + + assert.Equal(t, codes.PermissionDenied, status.Code(err)) + assert.Contains(t, err.Error(), "incorrect password") +} + +func TestLogin_SomeError(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedLogin := "testuser" + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + // Simulate an IncorrectPasswordError during Login + expectedError := errors.New("some") + mockAuthServices.EXPECT(). + Login(gomock.Any(), gomock.Eq(expectedLogin), gomock.Any()). + Return(uuid.Nil, "", "", expectedError) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + loginRequest := &proto.LoginRequest{ + Login: expectedLogin, + Password: "wrongpassword", + } + + loginResponse, err := authGRPCInstance.Login(context.Background(), loginRequest) + + assert.Error(t, err) + assert.Nil(t, loginResponse) + + assert.Contains(t, err.Error(), "some") +} + +func TestCheckLoginUnique_Success(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedLogin := "uniqueuser" + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + mockAuthServices.EXPECT(). + CheckLoginUnique(gomock.Any(), gomock.Eq(expectedLogin)). + Return(true, nil) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + uniqCheckRequest := &proto.UniqCheckRequest{ + Login: expectedLogin, + } + + uniqCheckResponse, err := authGRPCInstance.CheckLoginUnique(context.Background(), uniqCheckRequest) + + assert.NoError(t, err) + assert.NotNil(t, uniqCheckResponse) + + assert.Equal(t, "200", uniqCheckResponse.Status) + assert.False(t, uniqCheckResponse.Body) +} + +func TestCheckLoginUnique_NotUnique(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedLogin := "existinguser" + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + // Simulate a non-unique login during CheckLoginUnique + mockAuthServices.EXPECT(). + CheckLoginUnique(gomock.Any(), gomock.Eq(expectedLogin)). + Return(false, nil) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + uniqCheckRequest := &proto.UniqCheckRequest{ + Login: expectedLogin, + } + + uniqCheckResponse, err := authGRPCInstance.CheckLoginUnique(context.Background(), uniqCheckRequest) + + assert.NoError(t, err) + assert.NotNil(t, uniqCheckResponse) + + assert.Equal(t, "200", uniqCheckResponse.Status) + assert.True(t, uniqCheckResponse.Body) +} + +func TestCheckLoginUnique_Error(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedLogin := "testuser" + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + expectedError := errors.New("check login unique failed") + mockAuthServices.EXPECT(). + CheckLoginUnique(gomock.Any(), gomock.Eq(expectedLogin)). + Return(false, expectedError) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + uniqCheckRequest := &proto.UniqCheckRequest{ + Login: expectedLogin, + } + + uniqCheckResponse, err := authGRPCInstance.CheckLoginUnique(context.Background(), uniqCheckRequest) + + assert.Error(t, err) + assert.Nil(t, uniqCheckResponse) + + assert.Equal(t, codes.Internal, status.Code(err)) + assert.Contains(t, err.Error(), "check login unique failed") +} + +func TestGetByID_Success(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedID := uuid.New() + expectedLogin := "testuser" + expectedUsername := "Test User" + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + // Simulate a successful GetByID + mockAuthServices.EXPECT(). + GetByID(gomock.Any(), gomock.Eq(expectedID)). + Return(&models.User{ID: expectedID, Login: expectedLogin, Username: expectedUsername}, nil) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + userIDRequest := &proto.UserIdRequest{ + Id: expectedID.String(), + } + + userResponse, err := authGRPCInstance.GetByID(context.Background(), userIDRequest) + + assert.NoError(t, err) + assert.NotNil(t, userResponse) + + assert.Equal(t, "200", userResponse.Status) + assert.Equal(t, expectedID.String(), userResponse.Body.Id) + assert.Equal(t, expectedLogin, userResponse.Body.Login) + assert.Equal(t, expectedUsername, userResponse.Body.Username) +} + +func TestGetByID_InvalidID(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + invalidID := "invalid-id" + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + userIDRequest := &proto.UserIdRequest{ + Id: invalidID, + } + + userResponse, err := authGRPCInstance.GetByID(context.Background(), userIDRequest) + + assert.Error(t, err) + assert.Nil(t, userResponse) + +} + +func TestGetByID_UserNotFound(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedID := uuid.New() + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + expectedError := &models.NoSuchUserError{} + mockAuthServices.EXPECT(). + GetByID(gomock.Any(), gomock.Eq(expectedID)). + Return(nil, expectedError) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + userIDRequest := &proto.UserIdRequest{ + Id: expectedID.String(), + } + + userResponse, err := authGRPCInstance.GetByID(context.Background(), userIDRequest) + + assert.Error(t, err) + assert.Nil(t, userResponse) + + assert.Equal(t, codes.InvalidArgument, status.Code(err)) +} + +func TestGetByID_Error(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedID := uuid.New() + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + expectedError := errors.New("get user by ID failed") + mockAuthServices.EXPECT(). + GetByID(gomock.Any(), gomock.Eq(expectedID)). + Return(nil, expectedError) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + userIDRequest := &proto.UserIdRequest{ + Id: expectedID.String(), + } + + userResponse, err := authGRPCInstance.GetByID(context.Background(), userIDRequest) + + assert.Error(t, err) + assert.Nil(t, userResponse) + + assert.Equal(t, codes.Internal, status.Code(err)) + assert.Contains(t, err.Error(), "get user by ID failed") +} + +func TestChangePassword_Success(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedLogin := "testuser" + oldPassword := "oldpassword" + newPassword := "newpassword" + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + mockAuthServices.EXPECT(). + ChangePassword(gomock.Any(), auth.ChangePasswordInput{ + Login: expectedLogin, + OldPassword: oldPassword, + NewPassword: newPassword, + }). + Return(nil) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + changePasswordRequest := &proto.ChangePasswordRequest{ + Login: expectedLogin, + OldPassword: oldPassword, + NewPassword: newPassword, + } + + emptyResponse, err := authGRPCInstance.ChangePassword(context.Background(), changePasswordRequest) + + assert.NoError(t, err) + assert.NotNil(t, emptyResponse) +} + +func TestChangePassword_Error(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + expectedLogin := "testuser" + oldPassword := "oldpassword" + newPassword := "newpassword" + + mockAuthServices := mocks.NewMockUsecase(ctrl) + + expectedError := errors.New("change password failed") + mockAuthServices.EXPECT(). + ChangePassword(gomock.Any(), auth.ChangePasswordInput{ + Login: expectedLogin, + OldPassword: oldPassword, + NewPassword: newPassword, + }). + Return(expectedError) + + authGRPCInstance := NewAuthGRPC(mockAuthServices, *logger.NewLogger(context.TODO())) + + changePasswordRequest := &proto.ChangePasswordRequest{ + Login: expectedLogin, + OldPassword: oldPassword, + NewPassword: newPassword, + } + + emptyResponse, err := authGRPCInstance.ChangePassword(context.Background(), changePasswordRequest) + + assert.Error(t, err) + assert.Nil(t, emptyResponse) + + assert.Equal(t, codes.Internal, status.Code(err)) + assert.Contains(t, err.Error(), "change password failed") +} From b1d34dfe3b17a0cc24d9964fb8e234c24479826c Mon Sep 17 00:00:00 2001 From: Kosmatoff Date: Wed, 20 Dec 2023 17:25:50 +0300 Subject: [PATCH 2/2] test 67.4% --- .../user/delivery/http/handlers_test.go | 323 ++++++++++++++++++ 1 file changed, 323 insertions(+) diff --git a/internal/microservices/user/delivery/http/handlers_test.go b/internal/microservices/user/delivery/http/handlers_test.go index 63145d17..fb36ba49 100644 --- a/internal/microservices/user/delivery/http/handlers_test.go +++ b/internal/microservices/user/delivery/http/handlers_test.go @@ -5,6 +5,7 @@ import ( "context" "errors" "fmt" + "github.com/gorilla/mux" "image" "image/color" "image/png" @@ -1048,3 +1049,325 @@ func TestUserHandler_UpdateProfilePhoto(t *testing.T) { }) } } + +func TestHandler_AddUserInAccount(t *testing.T) { + uuidTest := uuid.New() + user := &models.User{ID: uuidTest} + + tests := []struct { + name string + user *models.User + requestBody io.Reader + expectedCode int + expectedBody string + mockUsecaseFn func(*mocks.MockUsecase) + }{ + { + name: "Invalid JSON body", + user: user, + requestBody: strings.NewReader("invalid json data"), + expectedCode: http.StatusBadRequest, + expectedBody: `{"status":400,"message":"invalid input body"}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) {}, + }, + { + name: "Successful Add User in Account", + user: user, + requestBody: strings.NewReader(`{ + "field1": "value1", + "field2": "value2" + }`), + expectedCode: http.StatusOK, + expectedBody: `{"status":200,"body":{}}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) { + mockUsecase.EXPECT().AddUserInAccount(gomock.Any(), gomock.Any(), gomock.Any()).Return(nil) + }, + }, + { + name: "Unauthorized Request", + user: nil, + requestBody: strings.NewReader(`{ + "field1": "value1", + "field2": "value2" + }`), + expectedCode: http.StatusUnauthorized, + expectedBody: `{"status":401,"message":"unauthorized"}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) {}, + }, + { + name: "Error adding user in account - Not Found", + user: user, + requestBody: strings.NewReader(`{ + "field1": "value1", + "field2": "value2" + }`), + expectedCode: http.StatusNotFound, + expectedBody: `{"status":404,"message":"no user found with this login"}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) { + mockUsecase.EXPECT().AddUserInAccount(gomock.Any(), gomock.Any(), gomock.Any()).Return(&models.NoSuchUserInLogin{}) + }, + }, + { + name: "Error adding user in account - Forbidden", + user: user, + requestBody: strings.NewReader(`{ + "field1": "value1", + "field2": "value2" + }`), + expectedCode: http.StatusForbidden, + expectedBody: `{"status":403,"message":"user has no rights"}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) { + mockUsecase.EXPECT().AddUserInAccount(gomock.Any(), gomock.Any(), gomock.Any()).Return(&models.ForbiddenUserError{}) + }, + }, + { + name: "Error adding user in account - Duplicate", + user: user, + requestBody: strings.NewReader(`{ + "field1": "value1", + "field2": "value2" + }`), + expectedCode: http.StatusBadRequest, + expectedBody: `{"status":400,"message":"this user has already been added to the account"}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) { + mockUsecase.EXPECT().AddUserInAccount(gomock.Any(), gomock.Any(), gomock.Any()).Return(&models.DuplicateError{}) + }, + }, + { + name: "Error adding user in account - Internal Server Error", + user: user, + requestBody: strings.NewReader(`{ + "field1": "value1", + "field2": "value2" + }`), + expectedCode: http.StatusInternalServerError, + expectedBody: `{"status":500,"message":"can't get user"}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) { + mockUsecase.EXPECT().AddUserInAccount(gomock.Any(), gomock.Any(), gomock.Any()).Return(errors.New("internal server error")) + }, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + mockService := mocks.NewMockUsecase(ctrl) + tt.mockUsecaseFn(mockService) + + mockHandler := NewHandler(mockService, *logger.NewLogger(context.TODO())) + + req := httptest.NewRequest("POST", "/api/user/add", tt.requestBody) + + if tt.user != nil { + ctx := context.WithValue(req.Context(), models.ContextKeyUserType{}, tt.user) + req = req.WithContext(ctx) + } + + recorder := httptest.NewRecorder() + + mockHandler.AddUserInAccount(recorder, req) + + actual := strings.TrimSpace(recorder.Body.String()) + + assert.Equal(t, tt.expectedCode, recorder.Code) + assert.Equal(t, tt.expectedBody, actual) + }) + } +} + +func TestHandler_Unsubscribe(t *testing.T) { + uuidTest := uuid.New() + user := &models.User{ID: uuidTest} + //accountID := uuid.New() + + tests := []struct { + name string + user *models.User + accountID string + expectedCode int + expectedBody string + mockUsecaseFn func(*mocks.MockUsecase) + }{ + { + name: "Invalid URL parameter", + user: user, + accountID: uuid.Nil.String(), + expectedCode: http.StatusBadRequest, + expectedBody: `{"status":400,"message":"invalid url parameter"}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) {}, + }, + { + name: "Unauthorized Request", + user: nil, + accountID: "account_id", + expectedCode: http.StatusUnauthorized, + expectedBody: `{"status":401,"message":"unauthorized"}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) {}, + }, + { + name: "Successful Unsubscribe", + user: user, + accountID: "account_id", + expectedCode: http.StatusOK, + expectedBody: `{"status":200,"body":{}}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) { + mockUsecase.EXPECT().Unsubscribe(gomock.Any(), gomock.Any(), gomock.Any()).Return(nil) + }, + }, + { + name: "Error unsubscribing user - Internal Server Error", + user: user, + accountID: "account_id", + expectedCode: http.StatusInternalServerError, + expectedBody: `{"status":500,"message":"can't get user"}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) { + mockUsecase.EXPECT().Unsubscribe(gomock.Any(), gomock.Any(), gomock.Any()).Return(errors.New("internal server error")) + }, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + mockService := mocks.NewMockUsecase(ctrl) + tt.mockUsecaseFn(mockService) + + mockHandler := NewHandler(mockService, *logger.NewLogger(context.TODO())) + + url := "/api/account/delete" + req := httptest.NewRequest("GET", url, nil) + req = mux.SetURLVars(req, map[string]string{tt.accountID: uuidTest.String()}) + + // Create a new request with the specified URL + + if tt.user != nil { + ctx := context.WithValue(req.Context(), models.ContextKeyUserType{}, tt.user) + req = req.WithContext(ctx) + } + + recorder := httptest.NewRecorder() + + mockHandler.Unsubscribe(recorder, req) + actual := strings.TrimSpace(recorder.Body.String()) + + assert.Equal(t, tt.expectedCode, recorder.Code) + assert.Equal(t, tt.expectedBody, actual) + }) + } +} + +func TestHandler_DeleteUserInAccount(t *testing.T) { + uuidTest := uuid.New() + user := &models.User{ID: uuidTest} + accountID := uuid.New() + + tests := []struct { + name string + user *models.User + requestBody io.Reader + expectedCode int + expectedBody string + mockUsecaseFn func(*mocks.MockUsecase) + }{ + { + name: "Unauthorized Request", + user: nil, + requestBody: strings.NewReader(`{"user_id": "testuser", "account_id": "account123"}`), + expectedCode: http.StatusUnauthorized, + expectedBody: `{"status":401,"message":"unauthorized"}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) {}, + }, + { + name: "Successful User Deletion in Account", + user: user, + requestBody: strings.NewReader(`{ + "user_id": "` + user.ID.String() + `", + "account_id": "` + accountID.String() + `" + }`), + expectedCode: http.StatusOK, + expectedBody: `{"status":200,"body":{}}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) { + mockUsecase.EXPECT().DeleteUserInAccount(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).Return(nil) + }, + }, + { + name: "User Not Found - Bad Request", + user: user, + requestBody: strings.NewReader(`{ + "user_id": "` + uuid.New().String() + `", + "account_id": "` + accountID.String() + `" + }`), + expectedCode: http.StatusBadRequest, + expectedBody: `{"status":400,"message":"no such user"}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) { + mockUsecase.EXPECT().DeleteUserInAccount(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).Return(&models.NoSuchUserInLogin{}) + }, + }, + { + name: "User Not Found - Bad Request", + user: user, + requestBody: strings.NewReader(`{ + "user_id": "` + uuid.New().String() + `", + "account_id": "` + accountID.String() + `" + }`), + expectedCode: http.StatusInternalServerError, + expectedBody: `{"status":500,"message":"can't get user"}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) { + mockUsecase.EXPECT().DeleteUserInAccount(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).Return(errors.New("some")) + }, + }, + { + name: "Forbidden User - Forbidden", + user: user, + requestBody: strings.NewReader(`{ + "user_id": "` + user.ID.String() + `", + "account_id": "` + uuid.New().String() + `" + }`), + expectedCode: http.StatusForbidden, + expectedBody: `{"status":403,"message":"user has no rights"}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) { + mockUsecase.EXPECT().DeleteUserInAccount(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).Return(&models.ForbiddenUserError{}) + }, + }, + { + name: "Invalid JSON body", + user: user, + requestBody: strings.NewReader("invalid json data"), + expectedCode: http.StatusBadRequest, + expectedBody: `{"status":400,"message":"invalid input body"}`, + mockUsecaseFn: func(mockUsecase *mocks.MockUsecase) {}, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + mockService := mocks.NewMockUsecase(ctrl) + tt.mockUsecaseFn(mockService) + + mockHandler := NewHandler(mockService, *logger.NewLogger(context.TODO())) + + req := httptest.NewRequest("DELETE", "/api/user/delete", tt.requestBody) + + if tt.user != nil { + ctx := context.WithValue(req.Context(), models.ContextKeyUserType{}, tt.user) + req = req.WithContext(ctx) + } + + recorder := httptest.NewRecorder() + + mockHandler.DeleteUserInAccount(recorder, req) + + actual := strings.TrimSpace(recorder.Body.String()) + + assert.Equal(t, tt.expectedCode, recorder.Code) + assert.Equal(t, tt.expectedBody, actual) + }) + } +}