PDA

Ver la Versión Completa : Problema raro con Firebird


marcial
06-02-2013, 09:19:47
Hola a todos:
Trabajo con D5, firebird 2.1 y componentes FibPlus.
El problema raro es el siguiente y sólo ocurre cuando trato de conectar con una BD Firebird remota: (En local funciona perfectamente)
la ruta de conexion que utilizo es: xxx.xxx.xxx.xxx:C:\Directorio\BaseDeDatos.FDB siendo xxx.xxx.xxx.xxx la ip pública de internet.
El programa es un mantenimiento de clientes que tiene un componente EventAlerter de Fibplus para que cuando haya algún cambio en los datos del cliente actualice automáticamente el DBGrid de presentación. En la Base de datos es un trigger que está así:
SET TERM ^ ;

CREATE TRIGGER ACTUALIZA_CLIENTES FOR CLIENTES
ACTIVE AFTER INSERT OR UPDATE OR DELETE
POSITION 0
AS
BEGIN
POST_EVENT 'ACTUALIZA_CLIENTES';
END^

SET TERM ; ^

y en el programa, en el OnShow tengo:


Eventos1.Events.Add('ACTUALIZA_CLIENTES');
Eventos1.RegisterEvents;

en el OnClose tengo:

Eventos1.UnregisterEvents;

y en el OnEventAlerter del componente tengo:

if EventName='ACTUALIZA_CLIENTES' then
begin
Clientes.Refresh;
end;


Pues bien, si pongo un "exit" al principio de el OnShow y no registra los eventos, el programa funciona perfectamente, sin embargo, si el programa entra por ahí y los registra, se queda colgado en el CLOSE cuando hace la siguiente consulta de clientes:


SQL := 'Select * from Clientes ORDER BY NOMBRE ASCENDING';

Clientes.CLose;
if pFIBTransaction1.InTransaction then pFIBTransaction1.CommitRetaining;
pFIBTransaction1.StartTransaction;
CLIENTES.SelectSQL.Clear;
CLIENTES.SelectSQL.Add(SQL);
CLIENTES.QSelect.ExecQuery;
pFIBtransaction1.CommitRetaining;
Clientes.Open;

Repito, sólo cuando registra los eventos.

¿Alguien puede ayudarme?
Gracias.

Casimiro Notevi
06-02-2013, 09:54:58
Ese tema se ha tratado en diversas ocasiones, el problema es que los "eventos" se transmiten por otros puertos que no es el 3050.
Haz una búsqueda por post_event a ver si lo encuentras.

marcial
06-02-2013, 09:58:58
Gracias Casimiro, me pongo a ello como un loco.

duilioisola
06-02-2013, 12:06:55
También puede ser que estés registrando más de una vez el evento.
El OnShow se ejecuta cada vez que se muestra el formulario (si minimizas y vuelves a maximizarlo o si haces un Form.Hide; Form.Show;).
Quizás debas poner el registro de eventos en OnActivate.

También puede ser que se esté ejecutando el evento al mismo tiempo que haces un Close y el evento trate de hacer un Refresh.
Para probar puedes crear una variable global a la unidad


var
TenerEnCuentaEventos : Boolean;

OnCreate
begin
TenerEnCuentaEventos := True;
end;

OnEvent
begin
if ((TenerEnCuentaEventos) and (EventName='ACTUALIZA_CLIENTES')) then
begin
Clientes.Refresh;
end;
end;

Consulta
begin
SQL := 'Select * from Clientes ORDER BY NOMBRE ASCENDING';

// "desactivo" eventos
TenerEnCuentaEventos := False;
try
Clientes.CLose;
if pFIBTransaction1.InTransaction then pFIBTransaction1.CommitRetaining;
pFIBTransaction1.StartTransaction;
CLIENTES.SelectSQL.Clear;
CLIENTES.SelectSQL.Add(SQL);
CLIENTES.QSelect.ExecQuery;
pFIBtransaction1.CommitRetaining;
Clientes.Open;
finally
// "activo" nuevamente los eventos
TenerEnCuentaEventos := True;
end;
end;

marcial
06-02-2013, 12:10:44
Por fin solucionado:

En el equipo donde está la BAse de Datos:

Con el Firebird.Config: Establecer el "Remote Auxc Port" por ejemplo 3051
En el Router: Abrir el puerto 3051 y direccionarlo a la IP local del Servidor donde esta la base de datos 192.168.X.XXX
Firewall: permitir/abrir el puerto 3051

Los programas no hay que tocarlos de como registran e interceptan el evento en los anteriores post.

Con esto ya se puede acceder remotamente

Gracias

marcial
06-02-2013, 12:14:48
Diuisola
Tal como he comentado, el problema de la conexión (fisica) está resuelto, pero con tu aporte creo que ya está perfecto. Tendré en cuenta tus recomendaciones.
Gracias

Lepe
06-02-2013, 13:00:56
En el OnShow y Activate se puede hacer un lío, creo lo mejor es en el OnCreate del Form, (así lo tengo yo).

En el OnClose ( o quizás en el OnCloseQuery si preguntas al usuario), antes de nada, quita el registro de eventos, así no interfiere con el resto de código.

Te aconsejo no juegues mucho con el RegisterEvents y el UnRegisterEvents porque pueden hacer pupa (lanzar excepciones), por eso, actívalos una vez y quítalo una vez nada más.

Además recuerda que solo puedes registrar 15 eventos por cada EventAlert.

Edito: Yo, no suelo hacer eso con los eventos... imagina que uno está editando un registro y le viene un mensaje de refresco... adios edición. Sí puedes poner un botón de refresco, o bien un mensaje de que hay nuevos registros desde la red. Si no usas controles DBaware, entonces sí puedes hacerlo perfectamente.

Saludos