diff --git a/web/client-api/src/test/java/io/deephaven/web/ClientIntegrationTestSuite.java b/web/client-api/src/test/java/io/deephaven/web/ClientIntegrationTestSuite.java index 3c3aa8791ca..7f3eb1c60a4 100644 --- a/web/client-api/src/test/java/io/deephaven/web/ClientIntegrationTestSuite.java +++ b/web/client-api/src/test/java/io/deephaven/web/ClientIntegrationTestSuite.java @@ -9,6 +9,7 @@ import io.deephaven.web.client.api.storage.JsStorageServiceTestGwt; import io.deephaven.web.client.api.subscription.ConcurrentTableTestGwt; import io.deephaven.web.client.api.subscription.ViewportTestGwt; +import io.deephaven.web.client.api.widget.plot.ChartDataTestGwt; import io.deephaven.web.client.fu.LazyPromiseTestGwt; import junit.framework.Test; import junit.framework.TestSuite; @@ -32,6 +33,7 @@ public static Test suite() { suite.addTestSuite(InputTableTestGwt.class); suite.addTestSuite(ColumnStatisticsTestGwt.class); suite.addTestSuite(GrpcTransportTestGwt.class); + suite.addTestSuite(ChartDataTestGwt.class); // This should be a unit test, but it requires a browser environment to run on GWT 2.9 // GWT 2.9 doesn't have proper bindings for Promises in HtmlUnit, so we need to use the IntegrationTest suite diff --git a/web/client-api/src/test/java/io/deephaven/web/CustomTransportTest.gwt.xml b/web/client-api/src/test/java/io/deephaven/web/CustomTransportTest.gwt.xml new file mode 100644 index 00000000000..36e00484fd6 --- /dev/null +++ b/web/client-api/src/test/java/io/deephaven/web/CustomTransportTest.gwt.xml @@ -0,0 +1,4 @@ + + + + diff --git a/web/client-api/src/test/java/io/deephaven/web/client/api/grpc/GrpcTransportTestGwt.java b/web/client-api/src/test/java/io/deephaven/web/client/api/grpc/GrpcTransportTestGwt.java index 020c65b771a..2a9ff5d5792 100644 --- a/web/client-api/src/test/java/io/deephaven/web/client/api/grpc/GrpcTransportTestGwt.java +++ b/web/client-api/src/test/java/io/deephaven/web/client/api/grpc/GrpcTransportTestGwt.java @@ -16,7 +16,8 @@ public class GrpcTransportTestGwt extends AbstractAsyncGwtTestCase { @Override public String getModuleName() { - return "io.deephaven.web.DeephavenIntegrationTest"; + // This test runs in its own module to avoid risking poisoning other tests with its broken custom transports + return "io.deephaven.web.CustomTransportTest"; } /** diff --git a/web/client-api/src/test/java/io/deephaven/web/client/api/widget/plot/ChartDataTestGwt.java b/web/client-api/src/test/java/io/deephaven/web/client/api/widget/plot/ChartDataTestGwt.java new file mode 100644 index 00000000000..d9faf49257e --- /dev/null +++ b/web/client-api/src/test/java/io/deephaven/web/client/api/widget/plot/ChartDataTestGwt.java @@ -0,0 +1,92 @@ +// +// Copyright (c) 2016-2024 Deephaven Data Labs and Patent Pending +// +package io.deephaven.web.client.api.widget.plot; + +import elemental2.promise.Promise; +import io.deephaven.web.client.api.AbstractAsyncGwtTestCase; +import io.deephaven.web.client.api.subscription.AbstractTableSubscription; +import io.deephaven.web.client.api.subscription.TableSubscription; + +public class ChartDataTestGwt extends AbstractAsyncGwtTestCase { + private final AbstractAsyncGwtTestCase.TableSourceBuilder tables = new AbstractAsyncGwtTestCase.TableSourceBuilder() + .script("from deephaven import time_table") + .script("appending_table", "time_table('PT0.1s').update([\"A = i % 2\", \"B = `` + A\"])") + .script("updating_table", "appending_table.last_by('A')"); + + @Override + public String getModuleName() { + return "io.deephaven.web.DeephavenIntegrationTest"; + } + + public void testAppendingChartData() { + connect(tables) + .then(table("appending_table")) + .then(table -> { + ChartData chartData = new ChartData(table); + TableSubscription sub = table.subscribe(table.getColumns()); + + // Handle at least one event with data + int[] count = {0}; + + return new Promise((resolve, reject) -> { + delayTestFinish(12301); + sub.addEventListener(TableSubscription.EVENT_UPDATED, event -> { + AbstractTableSubscription.SubscriptionEventData data = + (AbstractTableSubscription.SubscriptionEventData) event.getDetail(); + chartData.update(data); + + if (count[0] > 0) { + // Don't test on the first update, could still be empty + assertTrue(chartData.getColumn("A", arr -> arr, data).length > 0); + + resolve.onInvoke(data); + } + count[0]++; + }); + }).then(data -> { + sub.close(); + table.close(); + return Promise.resolve(data); + }); + }) + .then(this::finish).catch_(this::report); + } + + public void testModifiedChartData() { + connect(tables) + .then(table("updating_table")) + .then(table -> { + ChartData chartData = new ChartData(table); + TableSubscription sub = table.subscribe(table.getColumns()); + + int[] count = {0}; + return new Promise((resolve, reject) -> { + delayTestFinish(12302); + sub.addEventListener(TableSubscription.EVENT_UPDATED, event -> { + + AbstractTableSubscription.SubscriptionEventData data = + (AbstractTableSubscription.SubscriptionEventData) event.getDetail(); + + chartData.update(data); + if (count[0] > 0) { + // Don't test on the first update, could still be empty + assertTrue(chartData.getColumn("A", arr -> arr, data).length > 0); + } + + if (count[0] > 2) { + // We've seen at least three updates, and must have modified rows at least once + resolve.onInvoke((AbstractTableSubscription.SubscriptionEventData) event.getDetail()); + } + count[0]++; + }); + }) + .then(data -> { + sub.close(); + table.close(); + return Promise.resolve(data); + }); + }) + .then(this::finish).catch_(this::report); + } +}