package service import ( "context" "fmt" "net/http" "strconv" "github.com/getAlby/lndhub.go/rabbitmq" "github.com/getAlby/lndhub.go/db/models" "github.com/getAlby/lndhub.go/lib/responses" "github.com/getAlby/lndhub.go/lib/tokens" "github.com/getAlby/lndhub.go/lnd" "github.com/labstack/echo/v4" "github.com/labstack/gommon/random" "github.com/uptrace/bun" "github.com/ziflex/lecho/v3" "golang.org/x/crypto/bcrypt" ) const alphaNumBytes = random.Alphanumeric type LndhubService struct { Config *Config DB *bun.DB LndClient lnd.LightningClientWrapper RabbitMQClient rabbitmq.Client Logger *lecho.Logger InvoicePubSub *Pubsub } func (svc *LndhubService) GenerateToken(ctx context.Context, login, password, inRefreshToken string) (accessToken, refreshToken string, err error) { var user models.User switch { case login != "" || password != "": { if err := svc.DB.NewSelect().Model(&user).Where("login = ?", login).Scan(ctx); err != nil { return "", "", fmt.Errorf("bad auth") } if bcrypt.CompareHashAndPassword([]byte(user.Password), []byte(password)) != nil { return "", "", fmt.Errorf("bad auth") } } case inRefreshToken != "": { userId, err := tokens.GetUserIdFromToken(svc.Config.JWTSecret, inRefreshToken) if err != nil { return "", "", fmt.Errorf("bad auth") } if err := svc.DB.NewSelect().Model(&user).Where("id = ?", userId).Scan(ctx); err != nil { return "", "", fmt.Errorf("bad auth") } } default: { return "", "", fmt.Errorf("login and password or refresh token is required") } } if user.Deactivated || user.Deleted { return "", "", fmt.Errorf(responses.AccountDeactivatedError.Message) } accessToken, err = tokens.GenerateAccessToken(svc.Config.JWTSecret, svc.Config.JWTAccessTokenExpiry, &user) if err != nil { return "", "", err } refreshToken, err = tokens.GenerateRefreshToken(svc.Config.JWTSecret, svc.Config.JWTRefreshTokenExpiry, &user) if err != nil { return "", "", err } return accessToken, refreshToken, nil } func (svc *LndhubService) ParseInt(value interface{}) (int64, error) { switch v := value.(type) { case float64: return int64(v), nil case string: c, err := strconv.ParseInt(v, 10, 64) if err != nil { return 0, err } return c, nil default: return 0, fmt.Errorf("conversion to int from %T not supported", v) } } func (svc *LndhubService) ValidateUserMiddleware() echo.MiddlewareFunc { return func(next echo.HandlerFunc) echo.HandlerFunc { return func(c echo.Context) error { userId := c.Get("UserID").(int64) if userId == 0 { return echo.ErrUnauthorized } user, err := svc.FindUser(c.Request().Context(), userId) if err != nil { return echo.NewHTTPError(http.StatusUnauthorized, echo.Map{ "error": true, "code": 1, "message": "bad auth", }) } if user.Deactivated || user.Deleted { return echo.NewHTTPError(http.StatusUnauthorized, echo.Map{ "error": true, "code": 1, "message": "bad auth", }) } return next(c) } } }