diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 47b2acb..9693cd5 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -6,6 +6,12 @@ jobs: test: name: Test runs-on: ubuntu-latest + container: + image: node:18 + strategy: + matrix: + searchdb-host: [elastic7, elastic8, opensearch1] + fail-fast: false services: postgres: @@ -14,16 +20,21 @@ jobs: POSTGRES_DB: indexer_test POSTGRES_USER: indexer_test POSTGRES_PASSWORD: temba - ports: - - 5432:5432 options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5 - elastic: - image: elasticsearch:7.17.9 - ports: - - 9200:9200 - - 9300:9300 - env: - discovery.type: single-node + elastic7: + image: elasticsearch:7.17.9 + env: + discovery.type: single-node + elastic8: + image: elasticsearch:8.13.4 + env: + discovery.type: single-node + xpack.security.enabled: false + opensearch1: + image: opensearchproject/opensearch:1.3.16 + env: + discovery.type: single-node + plugins.security.disabled: true steps: - name: Checkout code @@ -34,6 +45,11 @@ jobs: with: go-version: ${{ env.go-version }} + - name: Setup environment + run: | + echo "INDEXER_DB=postgres://indexer_test:temba@postgres:5432/indexer_test?sslmode=disable" >> "$GITHUB_ENV" + echo "INDEXER_ELASTIC_URL=http://${{ matrix.searchdb-host }}:9200" >> "$GITHUB_ENV" + - name: Run tests run: go test -p=1 -coverprofile=coverage.text -covermode=atomic ./... diff --git a/indexers/base_test.go b/indexers/base_test.go index a7ba85d..cf7b247 100644 --- a/indexers/base_test.go +++ b/indexers/base_test.go @@ -15,40 +15,52 @@ import ( "github.com/nyaruka/gocommon/httpx" "github.com/nyaruka/gocommon/jsonx" + indexer "github.com/nyaruka/rp-indexer/v9" "github.com/nyaruka/rp-indexer/v9/indexers" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" ) -const elasticURL = "http://localhost:9200" const aliasName = "indexer_test" -func setup(t *testing.T) *sql.DB { +func getenv(key, def string) string { + val := os.Getenv(key) + if val == "" { + return def + } + return val +} + +func setup(t *testing.T) (*indexer.Config, *sql.DB) { + cfg := indexer.NewDefaultConfig() + cfg.DB = getenv("INDEXER_DB", "postgres://indexer_test:temba@localhost:5432/indexer_test?sslmode=disable") + cfg.ElasticURL = getenv("INDEXER_ELASTIC_URL", "http://localhost:9200") + testDB, err := os.ReadFile("../testdb.sql") require.NoError(t, err) - db, err := sql.Open("postgres", "postgres://indexer_test:temba@localhost:5432/indexer_test?sslmode=disable") + db, err := sql.Open("postgres", cfg.DB) require.NoError(t, err) _, err = db.Exec(string(testDB)) require.NoError(t, err) // delete all indexes with our alias prefix - existing := elasticRequest(t, http.MethodGet, "/_aliases", nil) + existing := elasticRequest(t, cfg, http.MethodGet, "/_aliases", nil) for name := range existing { if strings.HasPrefix(name, aliasName) { - elasticRequest(t, http.MethodDelete, "/"+name, nil) + elasticRequest(t, cfg, http.MethodDelete, "/"+name, nil) } } slog.SetDefault(slog.New(slog.NewTextHandler(os.Stdout, &slog.HandlerOptions{Level: slog.LevelDebug}))) - return db + return cfg, db } -func assertQuery(t *testing.T, query []byte, expected []int64, msgAndArgs ...interface{}) { - results := elasticRequest(t, http.MethodPost, "/"+aliasName+"/_search", +func assertQuery(t *testing.T, cfg *indexer.Config, query []byte, expected []int64, msgAndArgs ...interface{}) { + results := elasticRequest(t, cfg, http.MethodPost, "/"+aliasName+"/_search", map[string]any{"query": json.RawMessage(query), "sort": []map[string]any{{"id": "asc"}}}, ) hits := results["hits"].(map[string]any)["hits"].([]any) @@ -63,8 +75,8 @@ func assertQuery(t *testing.T, query []byte, expected []int64, msgAndArgs ...int assert.Equal(t, expected, actual, msgAndArgs...) } -func assertIndexesWithPrefix(t *testing.T, prefix string, expected []string) { - all := elasticRequest(t, http.MethodGet, "/_aliases", nil) +func assertIndexesWithPrefix(t *testing.T, cfg *indexer.Config, prefix string, expected []string) { + all := elasticRequest(t, cfg, http.MethodGet, "/_aliases", nil) actual := []string{} for name := range all { @@ -82,12 +94,12 @@ func assertIndexerStats(t *testing.T, ix indexers.Indexer, expectedIndexed, expe assert.Equal(t, expectedDeleted, actual.Deleted, "deleted mismatch") } -func elasticRequest(t *testing.T, method, path string, data map[string]any) map[string]any { +func elasticRequest(t *testing.T, cfg *indexer.Config, method, path string, data map[string]any) map[string]any { var body io.Reader if data != nil { body = bytes.NewReader(jsonx.MustMarshal(data)) } - req, err := httpx.NewRequest(method, elasticURL+path, body, map[string]string{"Content-Type": "application/json"}) + req, err := httpx.NewRequest(method, cfg.ElasticURL+path, body, map[string]string{"Content-Type": "application/json"}) require.NoError(t, err) trace, err := httpx.DoTrace(http.DefaultClient, req, nil, nil, -1) diff --git a/indexers/contacts_test.go b/indexers/contacts_test.go index 00b9209..8af1abc 100644 --- a/indexers/contacts_test.go +++ b/indexers/contacts_test.go @@ -61,9 +61,9 @@ var contactQueryTests = []struct { } func TestContacts(t *testing.T) { - db := setup(t) + cfg, db := setup(t) - ix1 := indexers.NewContactIndexer(elasticURL, aliasName, 2, 1, 4) + ix1 := indexers.NewContactIndexer(cfg.ElasticURL, aliasName, 2, 1, 4) assert.Equal(t, "indexer_test", ix1.Name()) dbModified, err := ix1.GetDBLastModified(context.Background(), db) @@ -87,10 +87,10 @@ func TestContacts(t *testing.T) { assert.WithinDuration(t, time.Date(2017, 11, 10, 21, 11, 59, 890662000, time.UTC), esModified, 0) assertIndexerStats(t, ix1, 9, 0) - assertIndexesWithPrefix(t, aliasName, []string{expectedIndexName}) + assertIndexesWithPrefix(t, cfg, aliasName, []string{expectedIndexName}) for _, tc := range contactQueryTests { - assertQuery(t, []byte(tc.query), tc.expected, "query mismatch for %s", tc.query) + assertQuery(t, cfg, []byte(tc.query), tc.expected, "query mismatch for %s", tc.query) } lastModified, err := ix1.GetESLastModified(indexName) @@ -112,13 +112,13 @@ func TestContacts(t *testing.T) { time.Sleep(1 * time.Second) - assertIndexesWithPrefix(t, aliasName, []string{expectedIndexName}) + assertIndexesWithPrefix(t, cfg, aliasName, []string{expectedIndexName}) // should only match new john, old john is gone - assertQuery(t, []byte(`{"match": {"name": {"query": "john"}}}`), []int64{2}) + assertQuery(t, cfg, []byte(`{"match": {"name": {"query": "john"}}}`), []int64{2}) // 3 is no longer in our group - assertQuery(t, []byte(`{"match": {"group_ids": {"query": 4}}}`), []int64{1}) + assertQuery(t, cfg, []byte(`{"match": {"group_ids": {"query": 4}}}`), []int64{1}) // change John's name to Eric.. _, err = db.Exec(` @@ -126,7 +126,7 @@ func TestContacts(t *testing.T) { require.NoError(t, err) // and simulate another indexer doing a parallel rebuild - ix2 := indexers.NewContactIndexer(elasticURL, aliasName, 2, 1, 4) + ix2 := indexers.NewContactIndexer(cfg.ElasticURL, aliasName, 2, 1, 4) indexName2, err := ix2.Index(db, true, false) assert.NoError(t, err) @@ -136,20 +136,20 @@ func TestContacts(t *testing.T) { time.Sleep(1 * time.Second) // check we have a new index but the old index is still around - assertIndexesWithPrefix(t, aliasName, []string{expectedIndexName, expectedIndexName + "_1"}) + assertIndexesWithPrefix(t, cfg, aliasName, []string{expectedIndexName, expectedIndexName + "_1"}) // and the alias points to the new index - assertQuery(t, []byte(`{"match": {"name": {"query": "eric"}}}`), []int64{2}) + assertQuery(t, cfg, []byte(`{"match": {"name": {"query": "eric"}}}`), []int64{2}) // simulate another indexer doing a parallel rebuild with cleanup - ix3 := indexers.NewContactIndexer(elasticURL, aliasName, 2, 1, 4) + ix3 := indexers.NewContactIndexer(cfg.ElasticURL, aliasName, 2, 1, 4) indexName3, err := ix3.Index(db, true, true) assert.NoError(t, err) assert.Equal(t, expectedIndexName+"_2", indexName3) // new index used assertIndexerStats(t, ix3, 8, 0) // check we cleaned up indexes besides the new one - assertIndexesWithPrefix(t, aliasName, []string{expectedIndexName + "_2"}) + assertIndexesWithPrefix(t, cfg, aliasName, []string{expectedIndexName + "_2"}) // check that the original indexer now indexes against the new index indexName, err = ix1.Index(db, false, false)