Controlar Timeout mediante un EJB

Iniciado por routerico, 31 Enero 2015, 13:19 PM

0 Miembros y 2 Visitantes están viendo este tema.

routerico

Hola qué tal, estoy intentando controlar un TimeOut mediante un EJB.


Ésta es la clase que tiene mi EJB:

import javax.ejb.Stateless;
import javax.ejb.Timeout;

@Stateless
public class TimeoutRespuesta {
   

    @Timeout
    public int sumador(int resultado){

        return resultado;

    }
}




Exporto el EJB mediante un .jar, me creo un proyecto Java y me creo una clase de pruebas:


public class Suma {

    public static void main(String[] args) {
       
       
        int res = (new TimeoutRespuesta()).sumador(sumar(1,2));
       
        System.out.println(res);
       
    }

    public static int sumar(int a, int b){
       
        return a+b;
    }
}



Aunque es un ejemplo muy sencillo, pero mi idea es controlar el Timeout mediante EJB, si fuese un método más complejo que no deje la aplicación bloqueada, que salte el Timeout del EJB y salga del método.

¿Estaría bien así?.

¿Cuál es el tiempo por defecto del Timeout del EJB?.


Un saludo y muchas gracias.

Usuario Invitado

#1
Cuando se utiliza un método timeout anotado con @timeout, la especificación nos dice que el método debe retornar void y sólo recibir como parámetro un objeto javax.ejb.TimerService o en su defecto, no debe recibir parámetros.

Para registrar un timeout podemos hacerlo de la siguiente manera (de acuerdo a la tercera forma):

Código (java) [Seleccionar]
createTimer(long duracionInicial, long duracionIntervalo, serializable objetoAsociado);

Así que puedes crearlo así:

Código (java) [Seleccionar]
@Resource TimerService timer; // opcional, aquí inyectamos al objeto TimerService

public void doSomething(Objeto obj) {
    timer.createTimer(15*60*1000, 15*60*1000, obj);
}


Lo que se ha hecho es crear un TimerService cuya primera véz de ejecución será un timeout de 15 minutos y el intérvalo entre ejecuciones será de 15 minutos también. Así mismo, hemos asociado al TimerService un objeto que debe implementar Serializable.

Bien, ahora el método que se ejecutará luego del registro del TimerService será:

Código (java) [Seleccionar]
@timeout doOtherThing(Timer timer)  {
    Objeto obj = (Objeto) timer.getInfo();
    // hacer otras cosas...
}


El método getInfo() retorna el objeto asociado al TimerService. Si al momento de crear el TimerService en lugar de un objeto se declara null, el valor devuelto será null también.

Ahora, si estoy en lo correcto, podemos cancelar un TimerService. Por ejemplo si en lugar de un objeto le hemos asociado un nombre, podríamos detenerlo así:

Código (java) [Seleccionar]
public void stop(String timerName) {
    for(Object obj : timerService.getTimers()) {
        Timer t = (Timer)obj;
        if (t.getInfo().equals(timerName)) {
        t.cancel();
        }
    }
}


Pero si no hemos asociado un nombre, podemos detenerlos iterando el conjunto de TimerService:

Código (java) [Seleccionar]
public void stop() { 
        List<Timer> timers = timerService.getTimers(); 
        if (timers != null) 
        { 
            for(Timer t : timers) {
                t.cancel();
            }
        } 
}


Salu2.

PD: Usa las etiquetas Geshi para colocar código.
"La vida es muy peligrosa. No por las personas que hacen el mal, si no por las que se sientan a ver lo que pasa." Albert Einstein

routerico

Cita de: Gus Garsaky en 31 Enero 2015, 15:42 PM
Cuando se utiliza un método timeout anotado con @timeout, la especificación nos dice que el método debe retornar void y sólo recibir como parámetro un objeto javax.ejb.TimerService o en su defecto, no debe recibir parámetros.

Para registrar un timeout podemos hacerlo de la siguiente manera (de acuerdo a la tercera forma):

Código (java) [Seleccionar]
createTimer(long duracionInicial, long duracionIntervalo, serializable objetoAsociado);

Así que puedes crearlo así:

Código (java) [Seleccionar]
@Resource TimerService timer; // opcional, aquí inyectamos al objeto TimerService

public void doSomething(Objeto obj) {
    timer.createTimer(15*60*1000, 15*60*1000, obj);
}


Lo que se ha hecho es crear un TimerService cuya primera véz de ejecución será un timeout de 15 minutos y el intérvalo entre ejecuciones será de 15 minutos también. Así mismo, hemos asociado al TimerService un objeto que debe implementar Serializable.

Bien, ahora el método que se ejecutará luego del registro del TimerService será:

Código (java) [Seleccionar]
@timeout doOtherThing(Timer timer)  {
    Objeto obj = (Objeto) timer.getInfo();
    // hacer otras cosas...
}


El método getInfo() retorna el objeto asociado al TimerService. Si al momento de crear el TimerService en lugar de un objeto se declara null, el valor devuelto será null también.

Ahora, si estoy en lo correcto, podemos cancelar un TimerService. Por ejemplo si en lugar de un objeto le hemos asociado un nombre, podríamos detenerlo así:

Código (java) [Seleccionar]
public void stop(String timerName) {
    for(Object obj : timerService.getTimers()) {
        Timer t = (Timer)obj;
        if (t.getInfo().equals(timerName)) {
        t.cancel();
        }
    }
}


Pero si no hemos asociado un nombre, podemos detenerlos iterando el conjunto de TimerService:

Código (java) [Seleccionar]
public void stop() { 
        List<Timer> timers = timerService.getTimers(); 
        if (timers != null) 
        { 
            for(Timer t : timers) {
                t.cancel();
            }
        } 
}


Salu2.

PD: Usa las etiquetas Geshi para colocar código.

Muchas gracias por tu ayuda, pero tengo algunas dudas de implementación.

Si quisiera controlar el timeout de un método como el anterior, ¿sería algo así?


Código (java) [Seleccionar]


import java.io.Serializable;
import javax.annotation.Resource;
import javax.ejb.Stateless;
import javax.ejb.Timeout;
import javax.ejb.Timer;
import javax.ejb.TimerService;


@Stateless
public class TimeoutRespuesta {


@Resource TimerService timer;

public void doSomething(Object obj) {
    timer.createTimer(1000, 1000, (Serializable) obj);
}

@Timeout void doOtherThing(Timer timer)  {
    Object obj = (Object) timer.getInfo();

}

public int sum(int i){
return i;
}

}





Código (java) [Seleccionar]

import timeout.TimeoutRespuesta;


public class Suma {

public static void main(String[] args) {


int res = (new TimeoutRespuesta()).sum(sumar(1,2));

System.out.println(res);

}

public static int sumar(int a, int b){
return a+b;
}
}





Gracias.

Usuario Invitado

No te entiendo. ¿Dices que quieres controlar el timeout dinámicamente?

Si es así, puedes controlarlo cancelándolo y volviendo a crearlo. Por ejemplo:

Código (java) [Seleccionar]
public void startOrModifyTimer(long initialExpiration, long interval, String name){     
        for (Timer timer: timer.getTimers()) {
            if (timer.getInfo().equals(name)) {
                timer.cancel();
            }
        }
        TimerConfig config = new TimerConfig();
        config.setInfo(name);
        config.setPersistent(false);
        timerService.createIntervalTimer(initialExpiration, interval, config);
}
"La vida es muy peligrosa. No por las personas que hacen el mal, si no por las que se sientan a ver lo que pasa." Albert Einstein

routerico

Cita de: Gus Garsaky en  1 Febrero 2015, 00:08 AM
No te entiendo. ¿Dices que quieres controlar el timeout dinámicamente?

Si es así, puedes controlarlo cancelándolo y volviendo a crearlo. Por ejemplo:

Código (java) [Seleccionar]
public void startOrModifyTimer(long initialExpiration, long interval, String name){     
        for (Timer timer: timer.getTimers()) {
            if (timer.getInfo().equals(name)) {
                timer.cancel();
            }
        }
        TimerConfig config = new TimerConfig();
        config.setInfo(name);
        config.setPersistent(false);
        timerService.createIntervalTimer(initialExpiration, interval, config);
}



Hola, muchas gracias. Ésto es lo que quiero hacer, pongo un ejemplo:

Tengo un proyecto web en Java, ahora bien, tengo un método que hace una consulta a la base de datos, pues imagínate que se queda pillado el método y no devuelve nunca el resultado de la consulta, en vez de quedarse bloqueado, lo que quiero es mediante el timeout de un EJB, controlar y decir si se invoca el método y pasan 20 segundos y no se recibe respuesta que se salga del método y no se quede pillado ahí.


Saludos.

Usuario Invitado

Yo lo haría así:

1.- Hacer la consulta en un Thread.
2.- Haces la consulta.
3.- Registras el Timer a 20 segundos.
4.- En el método @Timeout cancelas el Thread y por consiguiente la consulta.
"La vida es muy peligrosa. No por las personas que hacen el mal, si no por las que se sientan a ver lo que pasa." Albert Einstein

routerico

Cita de: Gus Garsaky en  1 Febrero 2015, 17:52 PM
Yo lo haría así:

1.- Hacer la consulta en un Thread.
2.- Haces la consulta.
3.- Registras el Timer a 20 segundos.
4.- En el método @Timeout cancelas el Thread y por consiguiente la consulta.

Vale gracias, hice los pasos:


1.- Hacer la consulta en un Thread.

Código (java) [Seleccionar]


public class ConsultaBBDD implements Runnable{

public void run() {
String datos = consultarBBDD();

}

public String consultarBBDD(){
return "datosBBDD";
}
}








2.- Haces la consulta.


Código (java) [Seleccionar]

public class ClasePrincipal {

public String obtenerDatos(){
Thread t = new Thread(new ConsultaBBDD());
t.start();

return "";
}

}





3.- Registras el Timer a 20 segundos.
4.- En el método @Timeout cancelas el Thread y por consiguiente la consulta.


Código (java) [Seleccionar]

import java.io.Serializable;
import javax.annotation.Resource;
import javax.ejb.Stateless;
import javax.ejb.Timeout;
import javax.ejb.Timer;
import javax.ejb.TimerService;


@Stateless
public class TimeoutRespuesta {

@Resource TimerService timer;

public void doSomething(Object obj) {
    timer.createTimer(20000, 20000, (Serializable) obj);
}

@Timeout void doOtherThing(Timer timer)  {
    Thread t = new Thread();
    t.stop();
}
}





¿Sería algo parecido a lo que puse?. Lo que me falta sería la invocación del EJB dentro de la ClasePrincipal, pero es que no sé exactamente cómo sería, ¿qué método suyo tendrían que invocar?.


Muchas gracias.

Saludos.


Usuario Invitado

#7
Exactamente. Me captaste la idea  ;D

Para invocar al método @Timeout no es necesario invocar nada. En el momento en que registras un TimerService, el método @Timeout se invocará automáticamente cuando expire el tiempo de espera.

Otra cosa. Un TimerService tiene que ser usado en un servidor de aplicaciones ya que, éste será inyectado por medio de @Resource por el contenedor del servidor. Además creo que para usar un EJB Timer es necesario especificar si es un Bean local o remoto por medio de sus interfaces. Te dejo un ejemplo:

Interface local:
Código (java) [Seleccionar]
import javax.ejb.Local;
import javax.ejb.Timer;

@Local
public interface BeanLocal {
public void start(String name);
public void execute(Timer timer);
public void stop(String name);
}


El EJB
Código (java) [Seleccionar]
import javax.annotation.Resource;
import javax.ejb.Singleton;
import javax.ejb.Timeout;
import javax.ejb.Timer;
import javax.ejb.TimerService;

@Singleton
public class TimerBean implements BeanLocal {

@Resource TimerService timer;

@Override
public void start(String name) {
boolean exists = false;
for(Timer temp : timer.getTimers()) {
if(temp.getInfo().equals(name))
exists = true;
}
if(!exists)
timer.createTimer(15000, 3000, name);
}

@Override @Timeout
public void execute(Timer timer) {
System.out.println("Método @Timeout invocado: "+timer.getInfo());
}

@Override
public void stop(String name) {
for(Timer temp : timer.getTimers()) {
if(temp.getInfo().equals(name))
temp.cancel();
}
}

}


Luego, donde uses el EJB Timer lo inyectas también:

Código (=java) [Seleccionar]
@Inject private BeanLocal myEJB;

Para registrar un TimerService, simplemente haría:

Código (=java) [Seleccionar]

myEJB.start("identificadorTimer");


Y para cancelar un TimerService:

Código (=java) [Seleccionar]
myEJB.stop("identificadorTimer");

Nota: Yo le puse @Singleton porque quería que solo exista una instancia del EJB Timer. Funciona igual con EJB Stateless y Stateful.

UPDATE

De acuerdo a la especificación de EJB 3.1, si va a ser un EJB local solamente, se puede obviar la interface @Local. Solo basta ponerle al EJB @LocalBean para indicar que es un EJB local.

Código (=java) [Seleccionar]
@LocalBean
@Singleton
public class TimerBean {
"La vida es muy peligrosa. No por las personas que hacen el mal, si no por las que se sientan a ver lo que pasa." Albert Einstein

routerico

Cita de: Gus Garsaky en  1 Febrero 2015, 19:18 PM
Exactamente. Me captaste la idea  ;D

Para invocar al método @Timeout no es necesario invocar nada. En el momento en que registras un TimerService, el método @Timeout se invocará automáticamente cuando expire el tiempo de espera.

Otra cosa. Un TimerService tiene que ser usado en un servidor de aplicaciones ya que, éste será inyectado por medio de @Resource por el contenedor del servidor. Además creo que para usar un EJB Timer es necesario especificar si es un Bean local o remoto por medio de sus interfaces. Te dejo un ejemplo:

Interface local:
Código (java) [Seleccionar]
import javax.ejb.Local;
import javax.ejb.Timer;

@Local
public interface BeanLocal {
public void start(String name);
public void execute(Timer timer);
public void stop(String name);
}


El EJB
Código (java) [Seleccionar]
import javax.annotation.Resource;
import javax.ejb.Singleton;
import javax.ejb.Timeout;
import javax.ejb.Timer;
import javax.ejb.TimerService;

@Singleton
public class TimerBean implements BeanLocal {

@Resource TimerService timer;

@Override
public void start(String name) {
boolean exists = false;
for(Timer temp : timer.getTimers()) {
if(temp.getInfo().equals(name))
exists = true;
}
if(!exists)
timer.createTimer(15000, 3000, name);
}

@Override @Timeout
public void execute(Timer timer) {
System.out.println("Método @Timeout invocado: "+timer.getInfo());
}

@Override
public void stop(String name) {
for(Timer temp : timer.getTimers()) {
if(temp.getInfo().equals(name))
temp.cancel();
}
}

}


Luego, donde uses el EJB Timer lo inyectas también:

Código (=java) [Seleccionar]
@Inject private BeanLocal myEJB;

Para registrar un TimerService, simplemente haría:

Código (=java) [Seleccionar]

myEJB.start("identificadorTimer");


Y para cancelar un TimerService:

Código (=java) [Seleccionar]
myEJB.stop("identificadorTimer");

Nota: Yo le puse @Singleton porque quería que solo exista una instancia del EJB Timer. Funciona igual con EJB Stateless y Stateful.

De acuerdo, muchísimas gracias. Mañana lo voy a probar a ver qué sucede.

Un saludo!!

routerico

Cita de: Gus Garsaky en  1 Febrero 2015, 19:18 PM


Código (=java) [Seleccionar]
@Inject private BeanLocal myEJB;

Para registrar un TimerService, simplemente haría:

Código (=java) [Seleccionar]

myEJB.start("identificadorTimer");


Y para cancelar un TimerService:

Código (=java) [Seleccionar]
myEJB.stop("identificadorTimer");

Nota: Yo le puse @Singleton porque quería que solo exista una instancia del EJB Timer. Funciona igual con EJB Stateless y Stateful.

UPDATE

De acuerdo a la especificación de EJB 3.1, si va a ser un EJB local solamente, se puede obviar la interface @Local. Solo basta ponerle al EJB @LocalBean para indicar que es un EJB local.

Código (=java) [Seleccionar]
@LocalBean
@Singleton
public class TimerBean {





Hola compañero, me está dando NullPointerException cuando en el proyecto web.


- Me creo el atributo del EJB
Código (=java) [Seleccionar]
@Inject private BeanLocal myEJB;


Pero cuando invoco al método start, ahí me da NullPointerException

Código (=java) [Seleccionar]

myEJB.start("identificadorTimer");



¿Cómo tendría que instanciar el EJB dentro de una clase del Proyecto Web para que no me de null?

Un saludo y muchas gracias de nuevo.