internal/gitlabnet/twofactorverify/client.go (83 lines of code) (raw):

// Package twofactorverify provides functionality for verifying two-factor authentication in GitLab. package twofactorverify import ( "context" "errors" "fmt" "net/http" "gitlab.com/gitlab-org/gitlab-shell/v14/client" "gitlab.com/gitlab-org/gitlab-shell/v14/internal/command/commandargs" "gitlab.com/gitlab-org/gitlab-shell/v14/internal/config" "gitlab.com/gitlab-org/gitlab-shell/v14/internal/gitlabnet" "gitlab.com/gitlab-org/gitlab-shell/v14/internal/gitlabnet/discover" ) // Client represents a client for interacting with the two-factor verification API. type Client struct { config *config.Config client *client.GitlabNetClient } // Response represents the response from the two-factor verification API. type Response struct { Success bool `json:"success"` Message string `json:"message"` } // RequestBody represents the request body for two-factor verification. type RequestBody struct { KeyID string `json:"key_id,omitempty"` UserID int64 `json:"user_id,omitempty"` OTPAttempt string `json:"otp_attempt,omitempty"` } // NewClient creates a new instance of the two-factor verification client. func NewClient(config *config.Config) (*Client, error) { client, err := gitlabnet.GetClient(config) if err != nil { return nil, fmt.Errorf("error creating http client: %v", err) } return &Client{config: config, client: client}, nil } // VerifyOTP verifies the one-time password (OTP) for two-factor authentication. func (c *Client) VerifyOTP(ctx context.Context, args *commandargs.Shell, otp string) error { requestBody, err := c.getRequestBody(ctx, args, otp) if err != nil { return err } response, err := c.client.Post(ctx, "/two_factor_manual_otp_check", requestBody) if err != nil { return err } defer func() { _ = response.Body.Close() }() return parse(response) } // PushAuth performs two-factor push authentication. func (c *Client) PushAuth(ctx context.Context, args *commandargs.Shell) error { requestBody, err := c.getRequestBody(ctx, args, "") if err != nil { return err } response, err := c.client.Post(ctx, "/two_factor_push_otp_check", requestBody) if err != nil { return err } defer func() { _ = response.Body.Close() }() return parse(response) } func parse(hr *http.Response) error { response := &Response{} if err := gitlabnet.ParseJSON(hr, response); err != nil { return err } if !response.Success { return errors.New(response.Message) } return nil } func (c *Client) getRequestBody(ctx context.Context, args *commandargs.Shell, otp string) (*RequestBody, error) { client, err := discover.NewClient(c.config) if err != nil { return nil, err } var requestBody *RequestBody if args.GitlabKeyID != "" { requestBody = &RequestBody{KeyID: args.GitlabKeyID, OTPAttempt: otp} } else { userInfo, err := client.GetByCommandArgs(ctx, args) if err != nil { return nil, err } requestBody = &RequestBody{UserID: userInfo.UserID, OTPAttempt: otp} } return requestBody, nil }