2020-03-23 01:33:04 +00:00
|
|
|
package live_query
|
|
|
|
|
|
|
|
|
|
import (
|
2024-02-28 01:35:27 +00:00
|
|
|
"context"
|
2020-03-23 01:33:04 +00:00
|
|
|
"testing"
|
2024-08-26 16:32:57 +00:00
|
|
|
"time"
|
2020-03-23 01:33:04 +00:00
|
|
|
|
2021-12-14 21:30:26 +00:00
|
|
|
"github.com/fleetdm/fleet/v4/server/datastore/redis"
|
2021-06-26 04:46:51 +00:00
|
|
|
"github.com/fleetdm/fleet/v4/server/fleet"
|
2021-12-14 21:30:26 +00:00
|
|
|
redigo "github.com/gomodule/redigo/redis"
|
2020-03-23 01:33:04 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
2020-07-01 17:51:34 +00:00
|
|
|
"github.com/stretchr/testify/require"
|
2020-03-23 01:33:04 +00:00
|
|
|
)
|
|
|
|
|
|
2021-06-06 22:07:29 +00:00
|
|
|
var testFunctions = [...]func(*testing.T, fleet.LiveQueryStore){
|
2020-07-01 17:51:34 +00:00
|
|
|
testLiveQuery,
|
|
|
|
|
testLiveQueryNoTargets,
|
|
|
|
|
testLiveQueryStopQuery,
|
2021-12-14 21:30:26 +00:00
|
|
|
testLiveQueryExpiredQuery,
|
|
|
|
|
testLiveQueryOnlyExpired,
|
2024-02-28 01:35:27 +00:00
|
|
|
testLiveQueryCleanupInactive,
|
2025-10-08 09:36:38 +00:00
|
|
|
testLiveQuerySetBitOnlyIfKeyExists,
|
2020-07-01 17:51:34 +00:00
|
|
|
}
|
|
|
|
|
|
2021-06-06 22:07:29 +00:00
|
|
|
func testLiveQuery(t *testing.T, store fleet.LiveQueryStore) {
|
2020-07-01 17:51:34 +00:00
|
|
|
queries, err := store.QueriesForHost(1)
|
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
assert.Len(t, queries, 0)
|
|
|
|
|
queries, err = store.QueriesForHost(3)
|
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
assert.Len(t, queries, 0)
|
|
|
|
|
|
|
|
|
|
assert.NoError(t, store.RunQuery("test", "select 1", []uint{1, 3}))
|
|
|
|
|
assert.NoError(t, store.RunQuery("test2", "select 2", []uint{3}))
|
|
|
|
|
assert.NoError(t, store.RunQuery("test3", "select 3", []uint{1}))
|
|
|
|
|
assert.NoError(t, store.RunQuery("test4", "select 4", []uint{4}))
|
|
|
|
|
|
|
|
|
|
queries, err = store.QueriesForHost(1)
|
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
assert.Equal(t,
|
|
|
|
|
map[string]string{
|
|
|
|
|
"test": "select 1",
|
|
|
|
|
"test3": "select 3",
|
|
|
|
|
},
|
|
|
|
|
queries,
|
|
|
|
|
)
|
|
|
|
|
queries, err = store.QueriesForHost(2)
|
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
assert.Len(t, queries, 0)
|
|
|
|
|
queries, err = store.QueriesForHost(3)
|
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
assert.Equal(t,
|
|
|
|
|
map[string]string{
|
|
|
|
|
"test": "select 1",
|
|
|
|
|
"test2": "select 2",
|
|
|
|
|
},
|
|
|
|
|
queries,
|
2020-03-23 01:33:04 +00:00
|
|
|
)
|
2020-07-01 17:51:34 +00:00
|
|
|
|
|
|
|
|
assert.NoError(t, store.QueryCompletedByHost("test", 1))
|
|
|
|
|
assert.NoError(t, store.QueryCompletedByHost("test2", 3))
|
|
|
|
|
|
|
|
|
|
queries, err = store.QueriesForHost(1)
|
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
assert.Equal(t,
|
|
|
|
|
map[string]string{
|
|
|
|
|
"test3": "select 3",
|
|
|
|
|
},
|
|
|
|
|
queries,
|
2020-03-23 01:33:04 +00:00
|
|
|
)
|
2020-07-01 17:51:34 +00:00
|
|
|
queries, err = store.QueriesForHost(2)
|
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
assert.Len(t, queries, 0)
|
|
|
|
|
queries, err = store.QueriesForHost(3)
|
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
assert.Equal(t,
|
|
|
|
|
map[string]string{
|
|
|
|
|
"test": "select 1",
|
|
|
|
|
},
|
|
|
|
|
queries,
|
2020-03-23 01:33:04 +00:00
|
|
|
)
|
|
|
|
|
}
|
2020-07-01 17:51:34 +00:00
|
|
|
|
2021-06-06 22:07:29 +00:00
|
|
|
func testLiveQueryNoTargets(t *testing.T, store fleet.LiveQueryStore) {
|
2020-07-01 17:51:34 +00:00
|
|
|
assert.Error(t, store.RunQuery("test", "select 1", []uint{}))
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-06 22:07:29 +00:00
|
|
|
func testLiveQueryStopQuery(t *testing.T, store fleet.LiveQueryStore) {
|
2020-07-01 17:51:34 +00:00
|
|
|
require.NoError(t, store.RunQuery("test", "select 1", []uint{1, 3}))
|
|
|
|
|
require.NoError(t, store.RunQuery("test2", "select 2", []uint{1, 3}))
|
|
|
|
|
require.NoError(t, store.StopQuery("test"))
|
|
|
|
|
|
|
|
|
|
queries, err := store.QueriesForHost(1)
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
assert.Len(t, queries, 1)
|
|
|
|
|
}
|
2021-12-14 21:30:26 +00:00
|
|
|
|
|
|
|
|
func testLiveQueryExpiredQuery(t *testing.T, store fleet.LiveQueryStore) {
|
|
|
|
|
oldModulo := cleanupExpiredQueriesModulo
|
|
|
|
|
cleanupExpiredQueriesModulo = 1 // run the cleanup each time
|
|
|
|
|
t.Cleanup(func() { cleanupExpiredQueriesModulo = oldModulo })
|
|
|
|
|
|
|
|
|
|
require.NoError(t, store.RunQuery("test", "select 1", []uint{1}))
|
|
|
|
|
|
|
|
|
|
// simulate a "test2" live query that has expired but is still in the set
|
|
|
|
|
pool := store.(*redisLiveQuery).pool
|
|
|
|
|
conn := redis.ConfigureDoer(pool, pool.Get())
|
|
|
|
|
defer conn.Close()
|
|
|
|
|
_, err := conn.Do("SADD", activeQueriesKey, "test2")
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
|
|
queries, err := store.QueriesForHost(1)
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
assert.Len(t, queries, 1)
|
|
|
|
|
assert.Equal(t, map[string]string{"test": "select 1"}, queries)
|
|
|
|
|
|
2024-08-29 16:03:45 +00:00
|
|
|
assert.Eventually(t, func() bool {
|
|
|
|
|
activeNames, err := redigo.Strings(conn.Do("SMEMBERS", activeQueriesKey))
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
if len(activeNames) == 1 && activeNames[0] == "test" {
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
return false
|
|
|
|
|
}, 5*time.Second, 100*time.Millisecond)
|
2021-12-14 21:30:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func testLiveQueryOnlyExpired(t *testing.T, store fleet.LiveQueryStore) {
|
|
|
|
|
oldModulo := cleanupExpiredQueriesModulo
|
|
|
|
|
cleanupExpiredQueriesModulo = 1 // run the cleanup each time
|
|
|
|
|
t.Cleanup(func() { cleanupExpiredQueriesModulo = oldModulo })
|
|
|
|
|
|
|
|
|
|
// simulate a "test" live query that has expired but is still in the set
|
|
|
|
|
pool := store.(*redisLiveQuery).pool
|
|
|
|
|
conn := redis.ConfigureDoer(pool, pool.Get())
|
|
|
|
|
defer conn.Close()
|
|
|
|
|
_, err := conn.Do("SADD", activeQueriesKey, "test")
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
|
|
queries, err := store.QueriesForHost(1)
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
assert.Len(t, queries, 0)
|
|
|
|
|
|
2024-08-26 16:32:57 +00:00
|
|
|
assert.Eventually(t, func() bool {
|
|
|
|
|
activeNames, err := redigo.Strings(conn.Do("SMEMBERS", activeQueriesKey))
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
return len(activeNames) == 0
|
|
|
|
|
}, 5*time.Second, 100*time.Millisecond)
|
2021-12-14 21:30:26 +00:00
|
|
|
}
|
2024-02-28 01:35:27 +00:00
|
|
|
|
|
|
|
|
func testLiveQueryCleanupInactive(t *testing.T, store fleet.LiveQueryStore) {
|
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
|
|
// get a raw Redis connection to make direct checks
|
|
|
|
|
pool := store.(*redisLiveQuery).pool
|
|
|
|
|
conn := redis.ConfigureDoer(pool, pool.Get())
|
|
|
|
|
defer conn.Close()
|
|
|
|
|
|
|
|
|
|
// run a few live queries, making them active in Redis
|
|
|
|
|
err := store.RunQuery("1", "SELECT 1", []uint{1, 2, 3})
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
err = store.RunQuery("2", "SELECT 2", []uint{4})
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
err = store.RunQuery("3", "SELECT 3", []uint{5, 6})
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
err = store.RunQuery("4", "SELECT 4", []uint{1, 2, 5})
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
err = store.RunQuery("5", "SELECT 5", []uint{2, 3, 7})
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
|
|
activeNames, err := store.LoadActiveQueryNames()
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
require.ElementsMatch(t, []string{"1", "2", "3", "4", "5"}, activeNames)
|
|
|
|
|
|
|
|
|
|
// sanity-check that the queries are properly stored
|
|
|
|
|
m, err := store.QueriesForHost(1)
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
require.Equal(t, map[string]string{"1": "SELECT 1", "4": "SELECT 4"}, m)
|
|
|
|
|
|
|
|
|
|
// simulate that only campaigns 2 and 4 are still active, cleanup the rest
|
|
|
|
|
err = store.CleanupInactiveQueries(ctx, []uint{1, 3, 5})
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
|
|
activeNames, err = store.LoadActiveQueryNames()
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
require.ElementsMatch(t, []string{"2", "4"}, activeNames)
|
|
|
|
|
|
|
|
|
|
m, err = store.QueriesForHost(1)
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
require.Equal(t, map[string]string{"4": "SELECT 4"}, m)
|
|
|
|
|
|
|
|
|
|
// explicitly mark campaign 4 as stopped
|
|
|
|
|
err = store.StopQuery("4")
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
|
|
// no more queries for host 1
|
|
|
|
|
m, err = store.QueriesForHost(1)
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
require.Empty(t, m)
|
|
|
|
|
|
|
|
|
|
// only campaign 2 remains, for host 4
|
|
|
|
|
m, err = store.QueriesForHost(4)
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
require.Equal(t, map[string]string{"2": "SELECT 2"}, m)
|
|
|
|
|
|
|
|
|
|
// simulate that there are no inactive campaigns to cleanup
|
|
|
|
|
err = store.CleanupInactiveQueries(ctx, nil)
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
|
|
activeNames, err = store.LoadActiveQueryNames()
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
require.ElementsMatch(t, []string{"2"}, activeNames)
|
|
|
|
|
|
|
|
|
|
// simulate that all campaigns are inactive, cleanup all
|
|
|
|
|
err = store.CleanupInactiveQueries(ctx, []uint{1, 2, 3, 4, 5})
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
|
|
activeNames, err = store.LoadActiveQueryNames()
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
require.Empty(t, activeNames)
|
|
|
|
|
|
|
|
|
|
m, err = store.QueriesForHost(4)
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
require.Empty(t, m)
|
|
|
|
|
}
|
2025-10-08 09:36:38 +00:00
|
|
|
|
|
|
|
|
func testLiveQuerySetBitOnlyIfKeyExists(t *testing.T, store fleet.LiveQueryStore) {
|
|
|
|
|
// Create a live query campaign.
|
|
|
|
|
err := store.RunQuery("test", "SELECT 1;", []uint{1})
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
|
|
// Get the query for the host.
|
|
|
|
|
queries, err := store.QueriesForHost(1)
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
require.Equal(t,
|
|
|
|
|
map[string]string{
|
|
|
|
|
"test": "SELECT 1;",
|
|
|
|
|
},
|
|
|
|
|
queries,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// Mark query as completed by host.
|
|
|
|
|
err = store.QueryCompletedByHost("test", 1)
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
|
|
// Query should not be returned anymore as it was marked as completed for this host.
|
|
|
|
|
queries, err = store.QueriesForHost(1)
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
require.Empty(t, queries)
|
|
|
|
|
|
|
|
|
|
// A host could be attempting to write a result for a query that was already deleted.
|
|
|
|
|
err = store.QueryCompletedByHost("test-2", 1)
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
|
|
// Let's test that such key was not created.
|
|
|
|
|
|
|
|
|
|
// get a raw Redis connection to make direct checks
|
|
|
|
|
pool := store.(*redisLiveQuery).pool
|
|
|
|
|
conn := redis.ConfigureDoer(pool, pool.Get())
|
|
|
|
|
t.Cleanup(func() {
|
|
|
|
|
conn.Close()
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
n, err := redigo.Int(conn.Do("EXISTS", queryKeyPrefix+"{test-2}"))
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
require.Zero(t, n)
|
|
|
|
|
}
|