You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

DataCommunicatorTest.java 11KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333
  1. package com.vaadin.data.provider;
  2. import static org.junit.Assert.assertEquals;
  3. import static org.junit.Assert.assertFalse;
  4. import static org.junit.Assert.assertTrue;
  5. import java.util.ArrayList;
  6. import java.util.Collections;
  7. import java.util.concurrent.Future;
  8. import org.junit.Test;
  9. import org.mockito.Mockito;
  10. import com.vaadin.data.provider.DataCommunicator.ActiveDataHandler;
  11. import com.vaadin.server.MockVaadinSession;
  12. import com.vaadin.server.SerializableConsumer;
  13. import com.vaadin.server.SerializablePredicate;
  14. import com.vaadin.server.VaadinRequest;
  15. import com.vaadin.server.VaadinService;
  16. import com.vaadin.server.VaadinSession;
  17. import com.vaadin.shared.Range;
  18. import com.vaadin.shared.Registration;
  19. import com.vaadin.ui.UI;
  20. import elemental.json.Json;
  21. import elemental.json.JsonArray;
  22. import elemental.json.JsonObject;
  23. /**
  24. * @author Vaadin Ltd
  25. *
  26. */
  27. public class DataCommunicatorTest {
  28. private static final Object TEST_OBJECT = new Object();
  29. private static final Object TEST_OBJECT_TWO = new Object();
  30. public static class TestUI extends UI {
  31. private final VaadinSession session;
  32. public TestUI(VaadinSession session) {
  33. this.session = session;
  34. }
  35. @Override
  36. protected void init(VaadinRequest request) {
  37. }
  38. @Override
  39. public VaadinSession getSession() {
  40. return session;
  41. }
  42. @Override
  43. public Future<Void> access(Runnable runnable) {
  44. runnable.run();
  45. return null;
  46. }
  47. }
  48. private static class TestDataProvider extends ListDataProvider<Object>
  49. implements Registration {
  50. private Registration registration;
  51. public TestDataProvider() {
  52. super(new ArrayList());
  53. addItem(TEST_OBJECT);
  54. }
  55. @Override
  56. public Registration addDataProviderListener(
  57. DataProviderListener<Object> listener) {
  58. registration = super.addDataProviderListener(listener);
  59. return this;
  60. }
  61. @Override
  62. public void remove() {
  63. registration.remove();
  64. registration = null;
  65. }
  66. public boolean isListenerAdded() {
  67. return registration != null;
  68. }
  69. public void setItem(Object item) {
  70. clear();
  71. addItem(item);
  72. }
  73. public void clear() {
  74. getItems().clear();
  75. }
  76. public void addItem(Object item) {
  77. getItems().add(item);
  78. }
  79. }
  80. private static class TestDataCommunicator extends DataCommunicator<Object> {
  81. protected void extend(UI ui) {
  82. super.extend(ui);
  83. }
  84. }
  85. private static class TestDataGenerator implements DataGenerator<Object> {
  86. Object refreshed = null;
  87. Object generated = null;
  88. @Override
  89. public void generateData(Object item, JsonObject jsonObject) {
  90. generated = item;
  91. }
  92. @Override
  93. public void refreshData(Object item) {
  94. refreshed = item;
  95. }
  96. }
  97. private final MockVaadinSession session = new MockVaadinSession(
  98. Mockito.mock(VaadinService.class));
  99. @Test
  100. public void attach_dataProviderListenerIsNotAddedBeforeAttachAndAddedAfter() {
  101. session.lock();
  102. UI ui = new TestUI(session);
  103. TestDataCommunicator communicator = new TestDataCommunicator();
  104. TestDataProvider dataProvider = new TestDataProvider();
  105. communicator.setDataProvider(dataProvider, null);
  106. assertFalse(dataProvider.isListenerAdded());
  107. communicator.extend(ui);
  108. assertTrue(dataProvider.isListenerAdded());
  109. }
  110. @Test
  111. public void detach_dataProviderListenerIsRemovedAfterDetach() {
  112. session.lock();
  113. UI ui = new TestUI(session);
  114. TestDataCommunicator communicator = new TestDataCommunicator();
  115. TestDataProvider dataProvider = new TestDataProvider();
  116. communicator.setDataProvider(dataProvider, null);
  117. communicator.extend(ui);
  118. assertTrue(dataProvider.isListenerAdded());
  119. communicator.detach();
  120. assertFalse(dataProvider.isListenerAdded());
  121. }
  122. @Test
  123. public void refresh_dataProviderListenerCallsRefreshInDataGeneartors() {
  124. session.lock();
  125. UI ui = new TestUI(session);
  126. TestDataCommunicator communicator = new TestDataCommunicator();
  127. communicator.extend(ui);
  128. TestDataProvider dataProvider = new TestDataProvider();
  129. communicator.setDataProvider(dataProvider, null);
  130. TestDataGenerator generator = new TestDataGenerator();
  131. communicator.addDataGenerator(generator);
  132. // Generate initial data.
  133. communicator.beforeClientResponse(true);
  134. assertEquals("DataGenerator generate was not called", TEST_OBJECT,
  135. generator.generated);
  136. generator.generated = null;
  137. // Make sure data does not get re-generated
  138. communicator.beforeClientResponse(false);
  139. assertEquals("DataGenerator generate was called again", null,
  140. generator.generated);
  141. // Refresh a data object to trigger an update.
  142. dataProvider.refreshItem(TEST_OBJECT);
  143. assertEquals("DataGenerator refresh was not called", TEST_OBJECT,
  144. generator.refreshed);
  145. // Test refreshed data generation
  146. communicator.beforeClientResponse(false);
  147. assertEquals("DataGenerator generate was not called", TEST_OBJECT,
  148. generator.generated);
  149. }
  150. @Test
  151. public void refreshDataProviderRemovesOldObjectsFromActiveDataHandler() {
  152. session.lock();
  153. UI ui = new TestUI(session);
  154. TestDataProvider dataProvider = new TestDataProvider();
  155. TestDataCommunicator communicator = new TestDataCommunicator();
  156. communicator.setDataProvider(dataProvider, null);
  157. communicator.extend(ui);
  158. communicator.beforeClientResponse(true);
  159. DataKeyMapper keyMapper = communicator.getKeyMapper();
  160. assertTrue("Object not mapped by key mapper",
  161. keyMapper.has(TEST_OBJECT));
  162. ActiveDataHandler handler = communicator.getActiveDataHandler();
  163. assertTrue("Object not amongst active data",
  164. handler.getActiveData().containsKey(TEST_OBJECT));
  165. dataProvider.setItem(TEST_OBJECT_TWO);
  166. dataProvider.refreshAll();
  167. communicator.beforeClientResponse(false);
  168. // assert that test object is marked as removed
  169. assertTrue("Object not marked as dropped",
  170. handler.getDroppedData().containsKey(TEST_OBJECT));
  171. communicator
  172. .setPushRows(Range.between(0, communicator.getMinPushSize()));
  173. communicator.beforeClientResponse(false);
  174. assertFalse("Object still mapped by key mapper",
  175. keyMapper.has(TEST_OBJECT));
  176. assertTrue("Object not mapped by key mapper",
  177. keyMapper.has(TEST_OBJECT_TWO));
  178. assertFalse("Object still amongst active data",
  179. handler.getActiveData().containsKey(TEST_OBJECT));
  180. assertTrue("Object not amongst active data",
  181. handler.getActiveData().containsKey(TEST_OBJECT_TWO));
  182. }
  183. @Test
  184. public void testDestroyData() {
  185. session.lock();
  186. UI ui = new TestUI(session);
  187. TestDataCommunicator communicator = new TestDataCommunicator();
  188. TestDataProvider dataProvider = new TestDataProvider();
  189. communicator.setDataProvider(dataProvider, null);
  190. communicator.extend(ui);
  191. // Put a test object into a cache
  192. communicator.pushData(1, Collections.singletonList(TEST_OBJECT));
  193. // Put the test object into an update queue
  194. communicator.refresh(TEST_OBJECT);
  195. // Drop the test object from the cache
  196. String key = communicator.getKeyMapper().key(TEST_OBJECT);
  197. JsonArray keys = Json.createArray();
  198. keys.set(0, key);
  199. communicator.onDropRows(keys);
  200. // Replace everything
  201. communicator.setDataProvider(
  202. new ListDataProvider<>(Collections.singleton(new Object())));
  203. // The communicator does not have to throw exceptions during
  204. // request finalization
  205. communicator.beforeClientResponse(false);
  206. assertFalse("Stalled object in KeyMapper",
  207. communicator.getKeyMapper().has(TEST_OBJECT));
  208. }
  209. @Test
  210. public void testFilteringLock() {
  211. session.lock();
  212. UI ui = new TestUI(session);
  213. TestDataCommunicator communicator = new TestDataCommunicator();
  214. communicator.extend(ui);
  215. ListDataProvider<Object> dataProvider = DataProvider.ofItems("one",
  216. "two", "three");
  217. SerializableConsumer<SerializablePredicate<Object>> filterSlot = communicator
  218. .setDataProvider(dataProvider, null);
  219. communicator.beforeClientResponse(true);
  220. // Mock empty request
  221. filterSlot.accept(t -> String.valueOf(t).contains("a"));
  222. communicator.beforeClientResponse(false);
  223. // Assume client clears up the filter
  224. filterSlot.accept(t -> String.valueOf(t).contains(""));
  225. communicator.beforeClientResponse(false);
  226. // And in the next request sets a non-matching filter
  227. // and has the data request for previous change
  228. communicator.onRequestRows(0, 3, 0, 0);
  229. filterSlot.accept(t -> String.valueOf(t).contains("a"));
  230. communicator.beforeClientResponse(false);
  231. // Mark communicator clean
  232. ui.getConnectorTracker().markClean(communicator);
  233. assertTrue("Communicator should be marked for hard reset",
  234. communicator.reset);
  235. assertFalse("DataCommunicator should not be marked as dirty",
  236. ui.getConnectorTracker().isDirty(communicator));
  237. // Set a filter that gets results again.
  238. filterSlot.accept(t -> String.valueOf(t).contains(""));
  239. assertTrue("DataCommunicator should be marked as dirty",
  240. ui.getConnectorTracker().isDirty(communicator));
  241. }
  242. @Test(expected = IllegalStateException.class)
  243. public void requestTooMuchRowsFail() {
  244. TestDataCommunicator communicator = new TestDataCommunicator();
  245. communicator.onRequestRows(0, communicator.getMaximumAllowedRows() + 10,
  246. 0, 0);
  247. }
  248. @Test
  249. public void requestTooMuchRowsOverride() {
  250. TestDataCommunicator communicator = new TestDataCommunicator();
  251. int maxRows = communicator.getMaximumAllowedRows();
  252. communicator.setMaximumAllowedRows(maxRows + 100);
  253. communicator.onRequestRows(0, maxRows + 10, 0, 0);
  254. }
  255. }