TbDateTest added
This commit is contained in:
		
							parent
							
								
									1ed577545f
								
							
						
					
					
						commit
						235c6f94d9
					
				@ -0,0 +1,102 @@
 | 
			
		||||
/**
 | 
			
		||||
 * Copyright © 2016-2023 The Thingsboard Authors
 | 
			
		||||
 *
 | 
			
		||||
 * Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
 * you may not use this file except in compliance with the License.
 | 
			
		||||
 * You may obtain a copy of the License at
 | 
			
		||||
 *
 | 
			
		||||
 *     http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 *
 | 
			
		||||
 * Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
 * distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
 * See the License for the specific language governing permissions and
 | 
			
		||||
 * limitations under the License.
 | 
			
		||||
 */
 | 
			
		||||
package org.thingsboard.script.api.tbel;
 | 
			
		||||
 | 
			
		||||
import com.google.common.util.concurrent.FutureCallback;
 | 
			
		||||
import com.google.common.util.concurrent.Futures;
 | 
			
		||||
import com.google.common.util.concurrent.ListenableFuture;
 | 
			
		||||
import com.google.common.util.concurrent.ListeningExecutorService;
 | 
			
		||||
import com.google.common.util.concurrent.MoreExecutors;
 | 
			
		||||
import lombok.extern.slf4j.Slf4j;
 | 
			
		||||
import org.junit.jupiter.api.AfterEach;
 | 
			
		||||
import org.junit.jupiter.api.Test;
 | 
			
		||||
 | 
			
		||||
import java.util.ArrayList;
 | 
			
		||||
import java.util.List;
 | 
			
		||||
import java.util.concurrent.CountDownLatch;
 | 
			
		||||
import java.util.concurrent.ExecutionException;
 | 
			
		||||
import java.util.concurrent.Executors;
 | 
			
		||||
import java.util.concurrent.TimeUnit;
 | 
			
		||||
import java.util.concurrent.TimeoutException;
 | 
			
		||||
 | 
			
		||||
@Slf4j
 | 
			
		||||
class TbDateTest {
 | 
			
		||||
 | 
			
		||||
    ListeningExecutorService executor;
 | 
			
		||||
 | 
			
		||||
    @AfterEach
 | 
			
		||||
    void tearDown() {
 | 
			
		||||
        if (executor != null) {
 | 
			
		||||
            executor.shutdownNow();
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Note: This test simulated the high concurrency calls.
 | 
			
		||||
     * But it not always fails before as the concurrency issue happens far away from the test subject, inside the SimpleDateFormat class.
 | 
			
		||||
     * Few calls later after latch open, the concurrency issue is not well reproduced.
 | 
			
		||||
     * Depends on environment some failure may happen each 2 or 100 runs.
 | 
			
		||||
     * To be highly confident run this test in a ForkMode=method, repeat 100 times. This will provide about 99 failures per 100 runs.
 | 
			
		||||
     * The value of this test is *never* to fail when isoDateFormat.format(this) is properly synchronized
 | 
			
		||||
     * If this test fails time-to-time -- it is a sign that isoDateFormat.format() called concurrently and have to be fixed (synchronized)
 | 
			
		||||
     * The expected exception example:
 | 
			
		||||
     *   Caused by: java.lang.ArrayIndexOutOfBoundsException: Index 14 out of bounds for length 13
 | 
			
		||||
     * 	     at java.base/sun.util.calendar.BaseCalendar.getCalendarDateFromFixedDate(BaseCalendar.java:457)
 | 
			
		||||
     * 	     at java.base/java.util.GregorianCalendar.computeFields(GregorianCalendar.java:2394)
 | 
			
		||||
     * 	     at java.base/java.util.GregorianCalendar.computeFields(GregorianCalendar.java:2309)
 | 
			
		||||
     * 	     at java.base/java.util.Calendar.setTimeInMillis(Calendar.java:1834)
 | 
			
		||||
     * 	     at java.base/java.util.Calendar.setTime(Calendar.java:1800)
 | 
			
		||||
     * 	     at java.base/java.text.SimpleDateFormat.format(SimpleDateFormat.java:974)
 | 
			
		||||
     */
 | 
			
		||||
    @Test
 | 
			
		||||
    void testToISOStringConcurrently() throws ExecutionException, InterruptedException, TimeoutException {
 | 
			
		||||
        int threads = 5;
 | 
			
		||||
        executor = MoreExecutors.listeningDecorator(Executors.newFixedThreadPool(threads));
 | 
			
		||||
        for (int j = 0; j < 1000; j++) {
 | 
			
		||||
            final int iteration = j;
 | 
			
		||||
            CountDownLatch readyLatch = new CountDownLatch(threads);
 | 
			
		||||
            CountDownLatch latch = new CountDownLatch(1);
 | 
			
		||||
            long now = 1709217342000L;
 | 
			
		||||
            List<ListenableFuture<String>> futures = new ArrayList<>(threads);
 | 
			
		||||
            for (int i = 0; i < threads; i++) {
 | 
			
		||||
                long ts = now + TimeUnit.DAYS.toMillis(i * 366) + TimeUnit.MINUTES.toMillis(iteration) + TimeUnit.SECONDS.toMillis(iteration) + iteration;
 | 
			
		||||
                TbDate tbDate = new TbDate(ts);
 | 
			
		||||
                futures.add(executor.submit(() -> {
 | 
			
		||||
                    readyLatch.countDown();
 | 
			
		||||
                    if (!latch.await(30, TimeUnit.SECONDS)) {
 | 
			
		||||
                        throw new RuntimeException("await timeout");
 | 
			
		||||
                    }
 | 
			
		||||
                    return tbDate.toISOString();
 | 
			
		||||
                }));
 | 
			
		||||
            }
 | 
			
		||||
            ListenableFuture<List<String>> future = Futures.allAsList(futures);
 | 
			
		||||
            Futures.addCallback(future, new FutureCallback<List<String>>() {
 | 
			
		||||
                @Override
 | 
			
		||||
                public void onSuccess(List<String> result) {
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                @Override
 | 
			
		||||
                public void onFailure(Throwable t) {
 | 
			
		||||
                    log.error("Failure happens on iteration {}", iteration);
 | 
			
		||||
                }
 | 
			
		||||
            }, MoreExecutors.directExecutor());
 | 
			
		||||
            readyLatch.await(30, TimeUnit.SECONDS);
 | 
			
		||||
            latch.countDown();
 | 
			
		||||
            future.get(30, TimeUnit.SECONDS);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user