fleet/client/device_client_test.go
Lucas Manuel Rodriguez 0b8c29198b
Make orbit and Fleet Desktop not depend on server/service/ packages (#42231)
Resolves #40396.

No changes file because there should be no user visible changes.

## Testing

- [x] QA'd all new/changed functionality manually

## fleetd/orbit/Fleet Desktop

- [x] Verified compatibility with the latest released version of Fleet
(see [Must
rule](https://github.com/fleetdm/fleet/blob/main/docs/Contributing/workflows/fleetd-development-and-release-strategy.md))
- [X] Verified that fleetd runs on macOS, Linux and Windows
- [X] Verified auto-update works from the released version of component
to the new version (see [tools/tuf/test](../tools/tuf/test/README.md))
2026-03-26 10:59:42 -03:00

210 lines
5.9 KiB
Go

package client
import (
"bytes"
"fmt"
"io"
"net/http"
"net/http/httptest"
"strings"
"sync/atomic"
"testing"
"github.com/stretchr/testify/require"
)
type mockHTTPClient struct {
resBody string
statusCode int
err error
}
func (m *mockHTTPClient) Do(req *http.Request) (*http.Response, error) {
if m.err != nil {
return nil, m.err
}
res := &http.Response{
StatusCode: m.statusCode,
Body: io.NopCloser(bytes.NewBufferString(m.resBody)),
}
return res, nil
}
func TestDeviceClientGetDesktopPayload(t *testing.T) {
client, err := NewDeviceClient("https://test.com", true, "", nil, "")
token := "test_token"
require.NoError(t, err)
mockRequestDoer := &mockHTTPClient{}
client.SetHTTPClient(mockRequestDoer)
t.Run("with wrong license", func(t *testing.T) {
mockRequestDoer.statusCode = http.StatusPaymentRequired
_, err = client.DesktopSummary(token)
require.ErrorIs(t, err, ErrMissingLicense)
})
t.Run("with no failing policies", func(t *testing.T) {
mockRequestDoer.statusCode = http.StatusOK
mockRequestDoer.resBody = `{}`
result, err := client.DesktopSummary(token)
require.NoError(t, err)
require.EqualValues(t, 0, *result.FailingPolicies)
require.False(t, result.Notifications.NeedsMDMMigration)
})
t.Run("with failing policies", func(t *testing.T) {
mockRequestDoer.statusCode = http.StatusOK
mockRequestDoer.resBody = `{"failing_policies_count": 1}`
result, err := client.DesktopSummary(token)
require.NoError(t, err)
require.EqualValues(t, 1, *result.FailingPolicies)
require.False(t, result.Notifications.NeedsMDMMigration)
})
t.Run("with flag to enable MDM migration", func(t *testing.T) {
mockRequestDoer.statusCode = http.StatusOK
mockRequestDoer.resBody = `{"failing_policies_count": 15, "notifications": {"needs_mdm_migration": true}}`
result, err := client.DesktopSummary(token)
require.NoError(t, err)
require.EqualValues(t, 15, *result.FailingPolicies)
require.True(t, result.Notifications.NeedsMDMMigration)
})
t.Run("alternative browser URL gets set from server response", func(t *testing.T) {
mockRequestDoer.statusCode = http.StatusOK
mockRequestDoer.resBody = `{"alternative_browser_host": "gogetit.com:6969"}`
_, err := client.DesktopSummary(token)
require.NoError(t, err)
require.EqualValues(t, "gogetit.com:6969", client.fleetAlternativeBrowserHostFromServer)
})
}
func TestDeviceClientGetFleetHost(t *testing.T) {
testCases := []struct {
alternativeBrowserHostFromEnv string
alternativeBrowserHostFromServer string
expected string
}{
{
alternativeBrowserHostFromEnv: "",
alternativeBrowserHostFromServer: "",
expected: "",
},
{
alternativeBrowserHostFromEnv: "https://example.com",
alternativeBrowserHostFromServer: "",
expected: "https://example.com",
},
{
alternativeBrowserHostFromEnv: "https://example.com",
alternativeBrowserHostFromServer: "https://two.example.com",
expected: "https://two.example.com",
},
{
alternativeBrowserHostFromEnv: "",
alternativeBrowserHostFromServer: "https://two.example.com",
expected: "https://two.example.com",
},
}
for _, tc := range testCases {
client, err := NewDeviceClient("", true, "", nil, tc.alternativeBrowserHostFromEnv)
require.NoError(t, err)
client.fleetAlternativeBrowserHostFromServer = tc.alternativeBrowserHostFromServer
require.Equal(t, tc.expected, client.getAlternativeBrowserHostSetting())
}
}
func TestDeviceClientRetryInvalidToken(t *testing.T) {
var callCounts atomic.Int64
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
callCounts.Add(1)
parts := strings.Split(r.URL.Path, "/")
token := parts[len(parts)-2] // last parts are /.../{token}/desktop
require.NotEmpty(t, token)
if token == "good_token" {
fmt.Fprint(w, `{"failing_policies_count": 1}`)
return
}
w.WriteHeader(http.StatusUnauthorized)
}))
t.Cleanup(srv.Close)
t.Run("no retry, bad token", func(t *testing.T) {
t.Cleanup(func() { callCounts.Store(0) })
client, err := NewDeviceClient(srv.URL, true, "", nil, "")
require.NoError(t, err)
_, err = client.DesktopSummary("bad")
require.Error(t, err)
require.ErrorIs(t, err, ErrUnauthenticated)
require.Equal(t, int64(1), callCounts.Load())
})
t.Run("no retry, good token", func(t *testing.T) {
t.Cleanup(func() { callCounts.Store(0) })
client, err := NewDeviceClient(srv.URL, true, "", nil, "")
require.NoError(t, err)
_, err = client.DesktopSummary("good_token")
require.NoError(t, err)
require.Equal(t, int64(1), callCounts.Load())
})
t.Run("with retry, good after retry", func(t *testing.T) {
t.Cleanup(func() { callCounts.Store(0) })
client, err := NewDeviceClient(srv.URL, true, "", nil, "")
require.NoError(t, err)
client.WithInvalidTokenRetry(func() string {
return "good_token"
})
_, err = client.DesktopSummary("bad")
require.NoError(t, err)
require.Equal(t, int64(2), callCounts.Load())
})
t.Run("with retry, good after 2 retries", func(t *testing.T) {
t.Cleanup(func() { callCounts.Store(0) })
client, err := NewDeviceClient(srv.URL, true, "", nil, "")
require.NoError(t, err)
var newToken string
client.WithInvalidTokenRetry(func() string {
switch newToken {
case "":
newToken = "bad"
default:
newToken = "good_token"
}
return newToken
})
_, err = client.DesktopSummary("bad")
require.NoError(t, err)
require.Equal(t, int64(3), callCounts.Load())
})
t.Run("with retry, always bad", func(t *testing.T) {
t.Cleanup(func() { callCounts.Store(0) })
client, err := NewDeviceClient(srv.URL, true, "", nil, "")
require.NoError(t, err)
client.WithInvalidTokenRetry(func() string {
return "still-bad"
})
_, err = client.DesktopSummary("bad")
require.Error(t, err)
require.ErrorIs(t, err, ErrUnauthenticated)
require.Equal(t, int64(4), callCounts.Load())
})
}