2024-07-12 02:59:32 +00:00
|
|
|
package handler_test
|
|
|
|
|
|
|
|
import (
|
2024-07-12 11:26:27 +00:00
|
|
|
"encoding/hex"
|
|
|
|
"fmt"
|
2024-07-12 02:59:32 +00:00
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"testing"
|
|
|
|
|
2024-07-12 11:26:27 +00:00
|
|
|
"git.ekzyis.com/ekzyis/delphi.market/server/auth"
|
2024-07-12 02:59:32 +00:00
|
|
|
"git.ekzyis.com/ekzyis/delphi.market/server/router/context"
|
|
|
|
"git.ekzyis.com/ekzyis/delphi.market/server/router/handler"
|
|
|
|
"git.ekzyis.com/ekzyis/delphi.market/test"
|
2024-07-12 11:26:27 +00:00
|
|
|
"github.com/decred/dcrd/dcrec/secp256k1/v4"
|
2024-07-12 02:59:32 +00:00
|
|
|
"github.com/labstack/echo/v4"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
test.Init(&db)
|
|
|
|
}
|
|
|
|
|
2024-07-12 11:26:27 +00:00
|
|
|
func TestLnAuthSignup(t *testing.T) {
|
2024-07-12 02:59:32 +00:00
|
|
|
var (
|
|
|
|
assert = assert.New(t)
|
2024-07-12 11:26:27 +00:00
|
|
|
sc = context.Context{Db: db}
|
2024-07-12 02:59:32 +00:00
|
|
|
e *echo.Echo
|
|
|
|
c echo.Context
|
|
|
|
req *http.Request
|
|
|
|
rec *httptest.ResponseRecorder
|
|
|
|
cookies []*http.Cookie
|
|
|
|
sessionId string
|
|
|
|
dbSessionId string
|
|
|
|
err error
|
|
|
|
)
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
e, req, rec = test.HTTPMocks("GET", "/signup/lightning", nil)
|
|
|
|
c = e.NewContext(req, rec)
|
|
|
|
c.SetParamNames("method")
|
|
|
|
c.SetParamValues("lightning")
|
|
|
|
|
2024-07-12 18:32:46 +00:00
|
|
|
handler.HandleAuth(sc, "register")(c)
|
|
|
|
assert.Equal(http.StatusOK, rec.Code, "wrong status code")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
// Set-Cookie header present
|
|
|
|
cookies = rec.Result().Cookies()
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.Equal(1, len(cookies), "wrong number of Set-Cookie headers")
|
|
|
|
assert.Equal("session", cookies[0].Name, "wrong cookie name")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
// new challenge inserted which matches cookie value
|
|
|
|
sessionId = cookies[0].Value
|
|
|
|
err = db.QueryRow("SELECT session_id FROM lnauth WHERE session_id = $1", sessionId).Scan(&dbSessionId)
|
|
|
|
assert.NoError(err)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.Equal(sessionId, dbSessionId, "wrong session id")
|
2024-07-12 11:26:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestLnAuthSignupCallbackUserNotExists(t *testing.T) {
|
|
|
|
var (
|
|
|
|
assert = assert.New(t)
|
|
|
|
e *echo.Echo
|
|
|
|
c echo.Context
|
|
|
|
sc context.Context
|
|
|
|
req *http.Request
|
|
|
|
rec *httptest.ResponseRecorder
|
|
|
|
lnAuth *auth.LnAuth
|
|
|
|
sk *secp256k1.PrivateKey
|
|
|
|
pk *secp256k1.PublicKey
|
|
|
|
sig string
|
|
|
|
key string
|
|
|
|
sessionId string
|
|
|
|
userId int
|
|
|
|
count int
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
lnAuth, err = auth.NewLnAuth("register")
|
|
|
|
assert.NoErrorf(err, "error creating challenge")
|
|
|
|
|
|
|
|
err = db.QueryRow(
|
|
|
|
"INSERT INTO lnauth(k1, lnurl) VALUES($1, $2) RETURNING session_id",
|
|
|
|
lnAuth.K1, lnAuth.LNURL).Scan(&sessionId)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error creating challenge")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
sk, pk, err = test.GenerateKeyPair()
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error generating keypair")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
sig, err = test.Sign(sk, lnAuth.K1)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error signing k1")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
key = hex.EncodeToString(pk.SerializeCompressed())
|
|
|
|
|
2024-07-12 02:59:32 +00:00
|
|
|
sc = context.Context{Db: db}
|
2024-07-12 18:21:12 +00:00
|
|
|
e, req, rec = test.HTTPMocks(
|
|
|
|
"GET",
|
2024-07-12 18:35:47 +00:00
|
|
|
fmt.Sprintf("/api/lnauth?tag=login&k1=%s&key=%s&sig=%s&action=%s", lnAuth.K1, key, sig, "register"),
|
2024-07-12 18:21:12 +00:00
|
|
|
nil)
|
2024-07-12 11:26:27 +00:00
|
|
|
c = e.NewContext(req, rec)
|
|
|
|
|
2024-07-12 18:32:46 +00:00
|
|
|
handler.HandleLnAuthCallback(sc)(c)
|
|
|
|
assert.Equal(http.StatusOK, rec.Code, "wrong status code")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
// user created
|
|
|
|
err = db.QueryRow("SELECT id FROM users WHERE ln_pubkey = $1", key).Scan(&userId)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error fetching user")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
// session created
|
|
|
|
err = db.QueryRow("SELECT COUNT(1) FROM sessions WHERE id = $1 AND user_id = $2", sessionId, userId).Scan(&count)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error fetching session")
|
|
|
|
assert.Equal(1, count, "invalid session count")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
// challenge deleted
|
|
|
|
err = db.QueryRow("SELECT COUNT(1) FROM lnauth WHERE k1 = $1", lnAuth.K1).Scan(&count)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error fetching challenge")
|
|
|
|
assert.Equal(count, 0, "challenge not deleted")
|
2024-07-12 11:26:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestLnAuthSignupCallbackUserExists(t *testing.T) {
|
|
|
|
var (
|
|
|
|
assert = assert.New(t)
|
|
|
|
e *echo.Echo
|
|
|
|
c echo.Context
|
|
|
|
sc context.Context
|
|
|
|
req *http.Request
|
|
|
|
rec *httptest.ResponseRecorder
|
|
|
|
lnAuth *auth.LnAuth
|
|
|
|
sk *secp256k1.PrivateKey
|
|
|
|
pk *secp256k1.PublicKey
|
|
|
|
sig string
|
|
|
|
key string
|
|
|
|
sessionId string
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
lnAuth, err = auth.NewLnAuth("register")
|
|
|
|
assert.NoErrorf(err, "error creating challenge")
|
|
|
|
|
|
|
|
err = db.QueryRow(
|
|
|
|
"INSERT INTO lnauth(k1, lnurl) VALUES($1, $2) RETURNING session_id",
|
|
|
|
lnAuth.K1, lnAuth.LNURL).Scan(&sessionId)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error creating challenge")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
sk, pk, err = test.GenerateKeyPair()
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error generating keypair")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
sig, err = test.Sign(sk, lnAuth.K1)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error signing k1")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
key = hex.EncodeToString(pk.SerializeCompressed())
|
|
|
|
|
2024-07-12 18:30:34 +00:00
|
|
|
// create user before signup
|
2024-07-12 11:26:27 +00:00
|
|
|
_, err = db.Exec("INSERT INTO users(ln_pubkey) VALUES($1) RETURNING id", key)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error creating user")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
sc = context.Context{Db: db}
|
2024-07-12 18:21:12 +00:00
|
|
|
e, req, rec = test.HTTPMocks(
|
|
|
|
"GET",
|
2024-07-12 18:35:47 +00:00
|
|
|
fmt.Sprintf("/api/lnauth?tag=login&k1=%s&key=%s&sig=%s&action=%s", lnAuth.K1, key, sig, "register"),
|
2024-07-12 18:21:12 +00:00
|
|
|
nil)
|
2024-07-12 11:26:27 +00:00
|
|
|
c = e.NewContext(req, rec)
|
|
|
|
|
2024-07-12 18:30:34 +00:00
|
|
|
// does not throw an error for UX reasons
|
2024-07-12 18:10:47 +00:00
|
|
|
handler.HandleLnAuthCallback(sc)(c)
|
2024-07-12 18:30:34 +00:00
|
|
|
assert.Equal(http.StatusOK, rec.Code, "wrong status code")
|
2024-07-12 11:26:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestLnAuthLogin(t *testing.T) {
|
|
|
|
var (
|
|
|
|
assert = assert.New(t)
|
|
|
|
sc = context.Context{Db: db}
|
|
|
|
e *echo.Echo
|
|
|
|
c echo.Context
|
|
|
|
req *http.Request
|
|
|
|
rec *httptest.ResponseRecorder
|
|
|
|
cookies []*http.Cookie
|
|
|
|
sessionId string
|
|
|
|
dbSessionId string
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
e, req, rec = test.HTTPMocks("GET", "/login/lightning", nil)
|
2024-07-12 02:59:32 +00:00
|
|
|
c = e.NewContext(req, rec)
|
2024-07-12 11:26:27 +00:00
|
|
|
c.SetParamNames("method")
|
|
|
|
c.SetParamValues("lightning")
|
2024-07-12 02:59:32 +00:00
|
|
|
|
2024-07-12 11:26:27 +00:00
|
|
|
err = handler.HandleAuth(sc, "login")(c)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "handler returned error")
|
2024-07-12 02:59:32 +00:00
|
|
|
|
|
|
|
// Set-Cookie header present
|
|
|
|
cookies = rec.Result().Cookies()
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.Equal(len(cookies), 1, "wrong number of Set-Cookie headers")
|
|
|
|
assert.Equal("session", cookies[0].Name, "wrong cookie name")
|
2024-07-12 02:59:32 +00:00
|
|
|
|
2024-07-12 11:26:27 +00:00
|
|
|
// new challenge inserted which matches cookie value
|
2024-07-12 02:59:32 +00:00
|
|
|
sessionId = cookies[0].Value
|
|
|
|
err = db.QueryRow("SELECT session_id FROM lnauth WHERE session_id = $1", sessionId).Scan(&dbSessionId)
|
2024-07-12 11:26:27 +00:00
|
|
|
assert.NoError(err)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.Equal(sessionId, dbSessionId, "wrong session id")
|
2024-07-12 02:59:32 +00:00
|
|
|
}
|
|
|
|
|
2024-07-12 11:26:27 +00:00
|
|
|
func TestLnAuthLoginCallbackUserNotExists(t *testing.T) {
|
|
|
|
var (
|
|
|
|
assert = assert.New(t)
|
|
|
|
e *echo.Echo
|
|
|
|
c echo.Context
|
|
|
|
sc context.Context
|
|
|
|
req *http.Request
|
|
|
|
rec *httptest.ResponseRecorder
|
|
|
|
lnAuth *auth.LnAuth
|
|
|
|
sk *secp256k1.PrivateKey
|
|
|
|
pk *secp256k1.PublicKey
|
|
|
|
sig string
|
|
|
|
key string
|
|
|
|
sessionId string
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
lnAuth, err = auth.NewLnAuth("login")
|
|
|
|
assert.NoErrorf(err, "error creating challenge")
|
|
|
|
|
|
|
|
err = db.QueryRow(
|
|
|
|
"INSERT INTO lnauth(k1, lnurl) VALUES($1, $2) RETURNING session_id",
|
|
|
|
lnAuth.K1, lnAuth.LNURL).Scan(&sessionId)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error creating challenge")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
sk, pk, err = test.GenerateKeyPair()
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error generating keypair")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
sig, err = test.Sign(sk, lnAuth.K1)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error signing k1")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
key = hex.EncodeToString(pk.SerializeCompressed())
|
|
|
|
|
|
|
|
sc = context.Context{Db: db}
|
2024-07-12 18:21:12 +00:00
|
|
|
e, req, rec = test.HTTPMocks(
|
|
|
|
"GET",
|
2024-07-12 18:35:47 +00:00
|
|
|
fmt.Sprintf("/api/lnauth?tag=login&k1=%s&key=%s&sig=%s&action=%s", lnAuth.K1, key, sig, "login"),
|
2024-07-12 18:21:12 +00:00
|
|
|
nil)
|
2024-07-12 11:26:27 +00:00
|
|
|
c = e.NewContext(req, rec)
|
|
|
|
|
|
|
|
// must throw error because user does not exist
|
2024-07-12 18:10:47 +00:00
|
|
|
handler.HandleLnAuthCallback(sc)(c)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.Equal(http.StatusNotFound, rec.Code, "wrong status code")
|
2024-07-12 18:10:47 +00:00
|
|
|
assert.Contains(rec.Body.String(), "\"reason\":\"user not found\"", "user check failed")
|
2024-07-12 11:26:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestLnAuthLoginCallbackUserExists(t *testing.T) {
|
|
|
|
var (
|
|
|
|
assert = assert.New(t)
|
|
|
|
e *echo.Echo
|
|
|
|
c echo.Context
|
|
|
|
sc context.Context
|
|
|
|
req *http.Request
|
|
|
|
rec *httptest.ResponseRecorder
|
|
|
|
lnAuth *auth.LnAuth
|
|
|
|
sk *secp256k1.PrivateKey
|
|
|
|
pk *secp256k1.PublicKey
|
|
|
|
sig string
|
|
|
|
key string
|
|
|
|
sessionId string
|
|
|
|
userId int
|
|
|
|
count int
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
lnAuth, err = auth.NewLnAuth("login")
|
|
|
|
assert.NoErrorf(err, "error creating challenge")
|
|
|
|
|
|
|
|
err = db.QueryRow(
|
|
|
|
"INSERT INTO lnauth(k1, lnurl) VALUES($1, $2) RETURNING session_id",
|
|
|
|
lnAuth.K1, lnAuth.LNURL).Scan(&sessionId)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error creating challenge")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
sk, pk, err = test.GenerateKeyPair()
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error generating keypair")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
sig, err = test.Sign(sk, lnAuth.K1)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error signing k1")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
key = hex.EncodeToString(pk.SerializeCompressed())
|
|
|
|
|
|
|
|
// create user such that login does not fail
|
|
|
|
err = db.QueryRow("INSERT INTO users(ln_pubkey) VALUES($1) RETURNING id", key).Scan(&userId)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error creating user")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
sc = context.Context{Db: db}
|
2024-07-12 18:21:12 +00:00
|
|
|
e, req, rec = test.HTTPMocks(
|
|
|
|
"GET",
|
2024-07-12 18:35:47 +00:00
|
|
|
fmt.Sprintf("/api/lnauth?tag=login&k1=%s&key=%s&sig=%s&action=%s", lnAuth.K1, key, sig, "login"),
|
2024-07-12 18:21:12 +00:00
|
|
|
nil)
|
2024-07-12 11:26:27 +00:00
|
|
|
c = e.NewContext(req, rec)
|
|
|
|
|
2024-07-12 18:32:46 +00:00
|
|
|
handler.HandleLnAuthCallback(sc)(c)
|
|
|
|
assert.Equal(http.StatusOK, rec.Code, "wrong status code")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
// session created
|
|
|
|
err = db.QueryRow("SELECT COUNT(1) FROM sessions WHERE id = $1 AND user_id = $2", sessionId, userId).Scan(&count)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error fetching session")
|
|
|
|
assert.Equal(1, count, "invalid session count")
|
2024-07-12 11:26:27 +00:00
|
|
|
|
|
|
|
// challenge deleted
|
|
|
|
err = db.QueryRow("SELECT COUNT(1) FROM lnauth WHERE k1 = $1", lnAuth.K1).Scan(&count)
|
2024-07-12 18:21:12 +00:00
|
|
|
assert.NoError(err, "error fetching challenge")
|
|
|
|
assert.Equal(count, 0, "challenge not deleted")
|
2024-07-12 02:59:32 +00:00
|
|
|
}
|