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
}