All Downloads are FREE. Search and download functionalities are using the official Maven repository.

ee.jakarta.tck.concurrent.spec.ManagedExecutorService.resourcedef.AppBean Maven / Gradle / Ivy

/*
 * Copyright (c) 2021, 2022 Contributors to the Eclipse Foundation
 *
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License v. 2.0, which is available at
 * http://www.eclipse.org/legal/epl-2.0.
 *
 * This Source Code may also be made available under the following Secondary
 * Licenses when the conditions for such availability set forth in the
 * Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
 * version 2 with the GNU Classpath Exception, which is available at
 * https://www.gnu.org/software/classpath/license.html.
 *
 * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
 */
package ee.jakarta.tck.concurrent.spec.ManagedExecutorService.resourcedef;

import java.time.Duration;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CompletionException;
import java.util.concurrent.CompletionStage;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.Exchanger;
import java.util.concurrent.Semaphore;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.logging.Logger;

import ee.jakarta.tck.concurrent.common.context.IntContext;
import ee.jakarta.tck.concurrent.common.context.StringContext;
import jakarta.enterprise.concurrent.Asynchronous;
import jakarta.enterprise.concurrent.Schedule;
import jakarta.enterprise.context.ApplicationScoped;

@ApplicationScoped
public class AppBean {
    
    private static final Logger log = Logger.getLogger(AppBean.class.getCanonicalName());
    
    private static final long MAX_WAIT_SECONDS = TimeUnit.MINUTES.toSeconds(2);
    
    // Asynchronous Tests

    @Asynchronous(executor = "java:module/concurrent/ExecutorB")
    public CompletionStage addStringContextAndWait(final BlockingQueue queue, final CountDownLatch blocker) {
        String s = StringContext.get();
        try {
            queue.add(s);
            blocker.await(MAX_WAIT_SECONDS, TimeUnit.SECONDS);
            return Asynchronous.Result.complete(s);
        } catch (Exception x) {
            throw new CompletionException(x);
        }
    }

    @Asynchronous
    public void exchange(final Exchanger exchanger, final String value) {
        try {
            exchanger.exchange(value, MAX_WAIT_SECONDS, TimeUnit.SECONDS);
        } catch (InterruptedException | TimeoutException x) {
            throw new CompletionException(x);
        }
    }

    @Asynchronous(executor = "java:app/concurrent/ExecutorA")
    public CompletableFuture waitAndGetIntContext(final Semaphore started, final CountDownLatch blocker) {
        started.release(1);
        CompletableFuture future = Asynchronous.Result.getFuture();
        try {
            while (!future.isDone() && !blocker.await(300, TimeUnit.MILLISECONDS)) {
                System.out.println(
                        Thread.currentThread().getName() + ": waitAndGetIntContext awaiting signal from caller");
            }
            future.complete(IntContext.get());
        } catch (Exception x) {
            future.completeExceptionally(x);
        }
        return future;
    }
    
    // Scheduled Asynchronous Tests
    public static enum RETURN {
            NULL, // Never completes
            COMPLETE_EXCEPTIONALLY, // Completes with exception
            COMPLETE_RESULT, // Completes with result
            INCOMPLETE, // Returns future that is not complete
            THROW_EXCEPTION; // Method throws exception
            
            private String message = "";
            
            public RETURN withMessage(final String m) {
                this.message = m;
                return this;
            }
            
            public String getMessage() {
                return message;
            }
    }
    
    /**
     * A scheduled async method that runs every 5 seconds
     *
     * @param runs - how many times to run before returning
     * @param type - how this method should return (successfully / incomplete / exceptionally)
     * @param counter - The counter provided from the caller to compare against
     *
     * @return A result or exception
     */
    @Asynchronous(runAt = @Schedule(cron = "*/5 * * * * *"))
    public CompletableFuture scheduledEvery5seconds(final int runs, final RETURN type, final AtomicInteger counter) {
        int count = counter.incrementAndGet();
        
        log.info("Executing scheduledEvery5seconds method " + count + "/" + runs + " (Returning: " + type.toString() + ")");
        log.info("  Thread: " + Thread.currentThread().toString());
        
        if (runs != count) {
            return null; // Continue onto next scheduled execution
        }
        
        CompletableFuture future = Asynchronous.Result.getFuture();
        
        switch (type) {
        case NULL: //Never stop executions
            return null;
        case COMPLETE_EXCEPTIONALLY:
            future.completeExceptionally(new Exception(type.getMessage()));
            break;
        case COMPLETE_RESULT:
            future.complete(count);
            break;
        case INCOMPLETE:
            break; //never complete future
        case THROW_EXCEPTION:
            throw new RuntimeException(type.getMessage());
        default:
            break;
        }
        
        return future;
    }
    
    @Asynchronous(runAt = @Schedule(cron = "*/3 * * * * *"))
    public void scheduledEvery3SecondsVoidReturn(final int runs, final RETURN type, final AtomicInteger counter) {
        int count = counter.incrementAndGet();
        
        log.info("Executing scheduledEvery3SecondsVoidReturn method " + count + "/" + runs + " (Returning: " + type.toString() + ")");
        log.info("  Thread: " + Thread.currentThread().toString());
        
        if (runs != count) {
            return; // Continue onto next scheduled execution
        }
        
        CompletableFuture future = Asynchronous.Result.getFuture();
        
        switch (type) {
        case COMPLETE_EXCEPTIONALLY:
            future.completeExceptionally(new Exception(type.getMessage()));
            break;
        case COMPLETE_RESULT:
            future.complete(null);
            break;
        case THROW_EXCEPTION:
            throw new RuntimeException(type.getMessage());
        default:
            break;
        }
    }
    
    /**
     * A scheduled async method that runs every 3 seconds, but takes 5 seconds to complete
     *
     * @param runs - how many times to run before returning
     * @param counter - The counter provided from the caller to compare against
     *
     * @return The number of runs completed
     */
    @Asynchronous(runAt = @Schedule(cron = "*/3 * * * * *"))
    public CompletableFuture scheduledEvery3SecondsTakes5Seconds(final int runs, final AtomicInteger counter) {
        int count = counter.incrementAndGet();
        
        log.info("Executing scheduledEvery3SecondsTakes5Seconds method " + count + "/" + runs);
        log.info("  Thread: " + Thread.currentThread().toString());
        
        if (runs != count) {
            
            try {
                Thread.sleep(Duration.ofSeconds(5).toMillis());
            } catch (InterruptedException e) {
                throw new RuntimeException("Thread was interrupted while waiting", e);
            }
            
            return null; // Continue onto next scheduled execution
        }
        
        CompletableFuture future = Asynchronous.Result.getFuture();
        future.complete(count);
        
        return future;
    }
    
    /**
     * A scheduled async method that runs every 3 seconds
     * Uses executor = "java:module/concurrent/ExecutorB" with max-async = 1
     *
     * @param runs - how many times to run before returning
     * @param counter - The counter provided from the caller to compare against
     *
     * @return completed future of IntContext
     */
    @Asynchronous(executor = "java:module/concurrent/ExecutorB", runAt = @Schedule(cron = "*/3 * * * * *"))
    public CompletableFuture scheduledEvery3Seconds(final int runs, final AtomicInteger counter) {
        int count = counter.incrementAndGet();
        
        log.info("Executing scheduledEvery3Seconds method " + count + "/" + runs);
        log.info("  Thread: " + Thread.currentThread().toString());
        
        if (runs != count) {
            return null; // Continue onto next scheduled execution
        }
        
        CompletableFuture future = Asynchronous.Result.getFuture();
        future.complete(IntContext.get());
        
        return future;
    }
    
    /**
     * A scheduled async method that runs every 3 seconds and every minute
     * Uses executor = "java:app/concurrent/ExecutorA" with max-async = 1
     *
     * @param runs - how many times to run before returning
     * @param counter - The counter provided from the caller to compare against
     *
     * @return completed future of StringContext
     */
    @Asynchronous(executor = "java:module/concurrent/ExecutorB", runAt = {
            @Schedule(cron = "*/3 * * * * *"),
            @Schedule(cron = "0 * * * * *")
    })
    public CompletableFuture scheduledEvery3SecondsAnd1Minute(final int runs, final AtomicInteger counter) {
        int count = counter.incrementAndGet();
        
        log.info("Executing scheduledEvery3SecondsAnd1Minute method " + count + "/" + runs);
        log.info("  Thread: " + Thread.currentThread().toString());
        
        if (runs != count) {
            return null; // Continue onto next scheduled execution
        }
        
        CompletableFuture future = Asynchronous.Result.getFuture();
        future.complete(StringContext.get());
        
        return future;
    }
    
    /**
     * A scheduled async method that should not run due to invalid configuration
     */
    @Asynchronous(executor = "java:app/concurrent/INVALID", runAt = @Schedule(cron = "*/3 * * * * *"))
    public CompletableFuture scheduledInvalidExecutor() {
        throw new UnsupportedOperationException("Should not be able to execute with invalid executor");
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy