mirror of
https://github.com/getAlby/lndhub.go.git
synced 2025-12-19 05:35:16 +01:00
157 lines
4.8 KiB
Go
157 lines
4.8 KiB
Go
package service
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"fmt"
|
|
"math"
|
|
|
|
"github.com/getAlby/lndhub.go/common"
|
|
"github.com/getAlby/lndhub.go/db/models"
|
|
"github.com/getAlby/lndhub.go/lib/security"
|
|
"github.com/getAlby/lndhub.go/lnd"
|
|
"github.com/uptrace/bun"
|
|
passwordvalidator "github.com/wagslane/go-password-validator"
|
|
)
|
|
|
|
func (svc *LndhubService) CreateUser(ctx context.Context, login string, password string) (user *models.User, err error) {
|
|
|
|
user = &models.User{}
|
|
|
|
// generate user login/password if not provided
|
|
user.Login = login
|
|
if login == "" {
|
|
randLoginBytes, err := randBytesFromStr(20, alphaNumBytes)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
user.Login = string(randLoginBytes)
|
|
}
|
|
|
|
if password == "" {
|
|
randPasswordBytes, err := randBytesFromStr(20, alphaNumBytes)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
password = string(randPasswordBytes)
|
|
} else {
|
|
if svc.Config.MinPasswordEntropy > 0 {
|
|
entropy := passwordvalidator.GetEntropy(password)
|
|
if entropy < float64(svc.Config.MinPasswordEntropy) {
|
|
return nil, fmt.Errorf("password entropy is too low (%f), required is %d", entropy, svc.Config.MinPasswordEntropy)
|
|
}
|
|
}
|
|
}
|
|
|
|
// we only store the hashed password but return the initial plain text password in the HTTP response
|
|
hashedPassword := security.HashPassword(password)
|
|
user.Password = hashedPassword
|
|
|
|
// Create user and the user's accounts
|
|
// We use double-entry bookkeeping so we use 4 accounts: incoming, current, outgoing and fees
|
|
// Wrapping this in a transaction in case something fails
|
|
err = svc.DB.RunInTx(ctx, &sql.TxOptions{}, func(ctx context.Context, tx bun.Tx) error {
|
|
if _, err := tx.NewInsert().Model(user).Exec(ctx); err != nil {
|
|
return err
|
|
}
|
|
accountTypes := []string{
|
|
common.AccountTypeIncoming,
|
|
common.AccountTypeCurrent,
|
|
common.AccountTypeOutgoing,
|
|
common.AccountTypeFees,
|
|
}
|
|
for _, accountType := range accountTypes {
|
|
account := models.Account{UserID: user.ID, Type: accountType}
|
|
if _, err := tx.NewInsert().Model(&account).Exec(ctx); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
})
|
|
//return actual password in the response, not the hashed one
|
|
user.Password = password
|
|
return user, err
|
|
}
|
|
|
|
func (svc *LndhubService) FindUser(ctx context.Context, userId int64) (*models.User, error) {
|
|
var user models.User
|
|
|
|
err := svc.DB.NewSelect().Model(&user).Where("id = ?", userId).Limit(1).Scan(ctx)
|
|
if err != nil {
|
|
return &user, err
|
|
}
|
|
return &user, nil
|
|
}
|
|
|
|
func (svc *LndhubService) FindUserByLogin(ctx context.Context, login string) (*models.User, error) {
|
|
var user models.User
|
|
|
|
err := svc.DB.NewSelect().Model(&user).Where("login = ?", login).Limit(1).Scan(ctx)
|
|
if err != nil {
|
|
return &user, err
|
|
}
|
|
return &user, nil
|
|
}
|
|
|
|
func (svc *LndhubService) BalanceCheck(ctx context.Context, lnpayReq *lnd.LNPayReq, userId int64) (ok bool, err error) {
|
|
currentBalance, err := svc.CurrentUserBalance(ctx, userId)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
minimumBalance := lnpayReq.PayReq.NumSatoshis
|
|
if svc.Config.FeeReserve {
|
|
minimumBalance += svc.CalcFeeLimit(lnpayReq.PayReq.Destination, lnpayReq.PayReq.NumSatoshis)
|
|
}
|
|
return currentBalance > minimumBalance, nil
|
|
}
|
|
|
|
func (svc *LndhubService) CalcFeeLimit(destination string, amount int64) int64 {
|
|
if destination == svc.IdentityPubkey {
|
|
return 0
|
|
}
|
|
limit := int64(10)
|
|
if amount > 1000 {
|
|
limit = int64(math.Ceil(float64(amount)*float64(0.01)) + 1)
|
|
}
|
|
return limit
|
|
}
|
|
|
|
func (svc *LndhubService) CurrentUserBalance(ctx context.Context, userId int64) (int64, error) {
|
|
var balance int64
|
|
|
|
account, err := svc.AccountFor(ctx, common.AccountTypeCurrent, userId)
|
|
if err != nil {
|
|
return balance, err
|
|
}
|
|
err = svc.DB.NewSelect().Table("account_ledgers").ColumnExpr("sum(account_ledgers.amount) as balance").Where("account_ledgers.account_id = ?", account.ID).Scan(ctx, &balance)
|
|
return balance, err
|
|
}
|
|
|
|
func (svc *LndhubService) AccountFor(ctx context.Context, accountType string, userId int64) (models.Account, error) {
|
|
account := models.Account{}
|
|
err := svc.DB.NewSelect().Model(&account).Where("user_id = ? AND type= ?", userId, accountType).Limit(1).Scan(ctx)
|
|
return account, err
|
|
}
|
|
|
|
func (svc *LndhubService) TransactionEntriesFor(ctx context.Context, userId int64) ([]models.TransactionEntry, error) {
|
|
transactionEntries := []models.TransactionEntry{}
|
|
err := svc.DB.NewSelect().Model(&transactionEntries).Where("user_id = ?", userId).Scan(ctx)
|
|
return transactionEntries, err
|
|
}
|
|
|
|
func (svc *LndhubService) InvoicesFor(ctx context.Context, userId int64, invoiceType string) ([]models.Invoice, error) {
|
|
var invoices []models.Invoice
|
|
|
|
query := svc.DB.NewSelect().Model(&invoices).Where("user_id = ?", userId)
|
|
if invoiceType != "" {
|
|
query.Where("type = ? AND state <> ?", invoiceType, common.InvoiceStateInitialized)
|
|
}
|
|
query.OrderExpr("id DESC").Limit(100)
|
|
err := query.Scan(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return invoices, nil
|
|
}
|