1
0
mirror of https://github.com/AvengeMedia/DankMaterialShell.git synced 2025-12-05 21:15:38 -05:00
Files
DankMaterialShell/core/internal/server/loginctl/handlers_test.go
2025-12-01 11:04:37 -05:00

501 lines
13 KiB
Go

package loginctl
import (
"bytes"
"encoding/json"
"net"
"sync"
"testing"
"time"
mockdbus "github.com/AvengeMedia/DankMaterialShell/core/internal/mocks/github.com/godbus/dbus/v5"
"github.com/AvengeMedia/DankMaterialShell/core/internal/server/models"
"github.com/godbus/dbus/v5"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require"
)
type mockNetConn struct {
net.Conn
readBuf *bytes.Buffer
writeBuf *bytes.Buffer
closed bool
}
func newMockNetConn() *mockNetConn {
return &mockNetConn{
readBuf: &bytes.Buffer{},
writeBuf: &bytes.Buffer{},
}
}
func (m *mockNetConn) Read(b []byte) (n int, err error) {
return m.readBuf.Read(b)
}
func (m *mockNetConn) Write(b []byte) (n int, err error) {
return m.writeBuf.Write(b)
}
func (m *mockNetConn) Close() error {
m.closed = true
return nil
}
func TestRespondError_Loginctl(t *testing.T) {
conn := newMockNetConn()
models.RespondError(conn, 123, "test error")
var resp models.Response[any]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Equal(t, "test error", resp.Error)
assert.Nil(t, resp.Result)
}
func TestRespond_Loginctl(t *testing.T) {
conn := newMockNetConn()
result := SuccessResult{Success: true, Message: "test"}
models.Respond(conn, 123, result)
var resp models.Response[SuccessResult]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Empty(t, resp.Error)
require.NotNil(t, resp.Result)
assert.True(t, resp.Result.Success)
assert.Equal(t, "test", resp.Result.Message)
}
func TestHandleGetState(t *testing.T) {
manager := &Manager{
state: &SessionState{
SessionID: "1",
Locked: false,
Active: true,
SessionType: "wayland",
SessionClass: "user",
UserName: "testuser",
},
stateMutex: sync.RWMutex{},
}
conn := newMockNetConn()
req := Request{ID: 123, Method: "loginctl.getState"}
handleGetState(conn, req, manager)
var resp models.Response[SessionState]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Empty(t, resp.Error)
require.NotNil(t, resp.Result)
assert.Equal(t, "1", resp.Result.SessionID)
assert.False(t, resp.Result.Locked)
assert.True(t, resp.Result.Active)
}
func TestHandleLock(t *testing.T) {
t.Run("successful lock", func(t *testing.T) {
mockSessionObj := mockdbus.NewMockBusObject(t)
mockCall := &dbus.Call{Err: nil}
mockSessionObj.EXPECT().Call("org.freedesktop.login1.Session.Lock", dbus.Flags(0)).Return(mockCall)
manager := &Manager{
state: &SessionState{},
stateMutex: sync.RWMutex{},
sessionObj: mockSessionObj,
}
conn := newMockNetConn()
req := Request{ID: 123, Method: "loginctl.lock"}
handleLock(conn, req, manager)
var resp models.Response[SuccessResult]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Empty(t, resp.Error)
require.NotNil(t, resp.Result)
assert.True(t, resp.Result.Success)
assert.Equal(t, "locked", resp.Result.Message)
})
t.Run("lock fails", func(t *testing.T) {
mockSessionObj := mockdbus.NewMockBusObject(t)
mockCall := &dbus.Call{Err: assert.AnError}
mockSessionObj.EXPECT().Call("org.freedesktop.login1.Session.Lock", dbus.Flags(0)).Return(mockCall)
manager := &Manager{
state: &SessionState{},
stateMutex: sync.RWMutex{},
sessionObj: mockSessionObj,
}
conn := newMockNetConn()
req := Request{ID: 123, Method: "loginctl.lock"}
handleLock(conn, req, manager)
var resp models.Response[SuccessResult]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Contains(t, resp.Error, "failed to lock session")
})
}
func TestHandleUnlock(t *testing.T) {
t.Run("successful unlock", func(t *testing.T) {
mockSessionObj := mockdbus.NewMockBusObject(t)
mockCall := &dbus.Call{Err: nil}
mockSessionObj.EXPECT().Call("org.freedesktop.login1.Session.Unlock", dbus.Flags(0)).Return(mockCall)
manager := &Manager{
state: &SessionState{},
stateMutex: sync.RWMutex{},
sessionObj: mockSessionObj,
}
conn := newMockNetConn()
req := Request{ID: 123, Method: "loginctl.unlock"}
handleUnlock(conn, req, manager)
var resp models.Response[SuccessResult]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Empty(t, resp.Error)
require.NotNil(t, resp.Result)
assert.True(t, resp.Result.Success)
assert.Equal(t, "unlocked", resp.Result.Message)
})
t.Run("unlock fails", func(t *testing.T) {
mockSessionObj := mockdbus.NewMockBusObject(t)
mockCall := &dbus.Call{Err: assert.AnError}
mockSessionObj.EXPECT().Call("org.freedesktop.login1.Session.Unlock", dbus.Flags(0)).Return(mockCall)
manager := &Manager{
state: &SessionState{},
stateMutex: sync.RWMutex{},
sessionObj: mockSessionObj,
}
conn := newMockNetConn()
req := Request{ID: 123, Method: "loginctl.unlock"}
handleUnlock(conn, req, manager)
var resp models.Response[SuccessResult]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Contains(t, resp.Error, "failed to unlock session")
})
}
func TestHandleActivate(t *testing.T) {
t.Run("successful activate", func(t *testing.T) {
mockSessionObj := mockdbus.NewMockBusObject(t)
mockCall := &dbus.Call{Err: nil}
mockSessionObj.EXPECT().Call("org.freedesktop.login1.Session.Activate", dbus.Flags(0)).Return(mockCall)
manager := &Manager{
state: &SessionState{},
stateMutex: sync.RWMutex{},
sessionObj: mockSessionObj,
}
conn := newMockNetConn()
req := Request{ID: 123, Method: "loginctl.activate"}
handleActivate(conn, req, manager)
var resp models.Response[SuccessResult]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Empty(t, resp.Error)
require.NotNil(t, resp.Result)
assert.True(t, resp.Result.Success)
assert.Equal(t, "activated", resp.Result.Message)
})
t.Run("activate fails", func(t *testing.T) {
mockSessionObj := mockdbus.NewMockBusObject(t)
mockCall := &dbus.Call{Err: assert.AnError}
mockSessionObj.EXPECT().Call("org.freedesktop.login1.Session.Activate", dbus.Flags(0)).Return(mockCall)
manager := &Manager{
state: &SessionState{},
stateMutex: sync.RWMutex{},
sessionObj: mockSessionObj,
}
conn := newMockNetConn()
req := Request{ID: 123, Method: "loginctl.activate"}
handleActivate(conn, req, manager)
var resp models.Response[SuccessResult]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Contains(t, resp.Error, "failed to activate session")
})
}
func TestHandleSetIdleHint(t *testing.T) {
t.Run("missing idle parameter", func(t *testing.T) {
manager := &Manager{
state: &SessionState{},
stateMutex: sync.RWMutex{},
}
conn := newMockNetConn()
req := Request{
ID: 123,
Method: "loginctl.setIdleHint",
Params: map[string]any{},
}
handleSetIdleHint(conn, req, manager)
var resp models.Response[any]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Contains(t, resp.Error, "missing or invalid 'idle' parameter")
})
t.Run("successful set idle hint true", func(t *testing.T) {
mockSessionObj := mockdbus.NewMockBusObject(t)
mockCall := &dbus.Call{Err: nil}
mockSessionObj.EXPECT().Call("org.freedesktop.login1.Session.SetIdleHint", dbus.Flags(0), true).Return(mockCall)
manager := &Manager{
state: &SessionState{},
stateMutex: sync.RWMutex{},
sessionObj: mockSessionObj,
}
conn := newMockNetConn()
req := Request{
ID: 123,
Method: "loginctl.setIdleHint",
Params: map[string]any{
"idle": true,
},
}
handleSetIdleHint(conn, req, manager)
var resp models.Response[SuccessResult]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Empty(t, resp.Error)
require.NotNil(t, resp.Result)
assert.True(t, resp.Result.Success)
assert.Equal(t, "idle hint set", resp.Result.Message)
})
t.Run("set idle hint fails", func(t *testing.T) {
mockSessionObj := mockdbus.NewMockBusObject(t)
mockCall := &dbus.Call{Err: assert.AnError}
mockSessionObj.EXPECT().Call("org.freedesktop.login1.Session.SetIdleHint", dbus.Flags(0), false).Return(mockCall)
manager := &Manager{
state: &SessionState{},
stateMutex: sync.RWMutex{},
sessionObj: mockSessionObj,
}
conn := newMockNetConn()
req := Request{
ID: 123,
Method: "loginctl.setIdleHint",
Params: map[string]any{
"idle": false,
},
}
handleSetIdleHint(conn, req, manager)
var resp models.Response[SuccessResult]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Contains(t, resp.Error, "failed to set idle hint")
})
}
func TestHandleTerminate(t *testing.T) {
t.Run("successful terminate", func(t *testing.T) {
mockSessionObj := mockdbus.NewMockBusObject(t)
mockCall := &dbus.Call{Err: nil}
mockSessionObj.EXPECT().Call("org.freedesktop.login1.Session.Terminate", dbus.Flags(0)).Return(mockCall)
manager := &Manager{
state: &SessionState{},
stateMutex: sync.RWMutex{},
sessionObj: mockSessionObj,
}
conn := newMockNetConn()
req := Request{ID: 123, Method: "loginctl.terminate"}
handleTerminate(conn, req, manager)
var resp models.Response[SuccessResult]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Empty(t, resp.Error)
require.NotNil(t, resp.Result)
assert.True(t, resp.Result.Success)
assert.Equal(t, "terminated", resp.Result.Message)
})
t.Run("terminate fails", func(t *testing.T) {
mockSessionObj := mockdbus.NewMockBusObject(t)
mockCall := &dbus.Call{Err: assert.AnError}
mockSessionObj.EXPECT().Call("org.freedesktop.login1.Session.Terminate", dbus.Flags(0)).Return(mockCall)
manager := &Manager{
state: &SessionState{},
stateMutex: sync.RWMutex{},
sessionObj: mockSessionObj,
}
conn := newMockNetConn()
req := Request{ID: 123, Method: "loginctl.terminate"}
handleTerminate(conn, req, manager)
var resp models.Response[SuccessResult]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Contains(t, resp.Error, "failed to terminate session")
})
}
func TestHandleRequest(t *testing.T) {
manager := &Manager{
state: &SessionState{
SessionID: "1",
Locked: false,
},
stateMutex: sync.RWMutex{},
}
t.Run("unknown method", func(t *testing.T) {
conn := newMockNetConn()
req := Request{
ID: 123,
Method: "loginctl.unknown",
}
HandleRequest(conn, req, manager)
var resp models.Response[any]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Contains(t, resp.Error, "unknown method")
})
t.Run("valid method - getState", func(t *testing.T) {
conn := newMockNetConn()
req := Request{
ID: 123,
Method: "loginctl.getState",
}
HandleRequest(conn, req, manager)
var resp models.Response[SessionState]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
assert.Empty(t, resp.Error)
})
t.Run("lock method", func(t *testing.T) {
mockSessionObj := mockdbus.NewMockBusObject(t)
mockCall := &dbus.Call{Err: nil}
mockSessionObj.EXPECT().Call("org.freedesktop.login1.Session.Lock", mock.Anything).Return(mockCall)
manager.sessionObj = mockSessionObj
conn := newMockNetConn()
req := Request{
ID: 123,
Method: "loginctl.lock",
}
HandleRequest(conn, req, manager)
var resp models.Response[any]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
require.NoError(t, err)
assert.Equal(t, 123, resp.ID)
})
}
func TestHandleSubscribe(t *testing.T) {
manager := &Manager{
state: &SessionState{
SessionID: "1",
Locked: false,
},
stateMutex: sync.RWMutex{},
}
conn := newMockNetConn()
req := Request{ID: 123, Method: "loginctl.subscribe"}
done := make(chan bool)
go func() {
handleSubscribe(conn, req, manager)
done <- true
}()
time.Sleep(50 * time.Millisecond)
conn.Close()
if conn.writeBuf.Len() > 0 {
var resp models.Response[SessionEvent]
err := json.NewDecoder(conn.writeBuf).Decode(&resp)
if err == nil {
assert.Equal(t, 123, resp.ID)
require.NotNil(t, resp.Result)
assert.Equal(t, EventStateChanged, resp.Result.Type)
assert.Equal(t, "1", resp.Result.Data.SessionID)
}
}
select {
case <-done:
case <-time.After(100 * time.Millisecond):
}
}