PDA

Ver la Versión Completa : WMI y errores que cuelgan toda la aplicación


elrayo76
10-05-2022, 03:21:10
Buenas,

Estoy teniendo problemas con el método ‘ConnectServer’ de la interfaz ‘ISWbemLocator’. Precisamente lo que pasa es que si la máquina no tiene levantado el servicio este se cuelga y hace que se cuelgue toda la aplicación. Esto independientemente del Windows que tenga la máquina (puede ser Windows Server, Windows 7 o Windows 10)

He visto en la documentación de Microsoft que esto debería devolver un código de error, pero como comenté se cuelga y no devuelve nada. Mirando los componentes de Neftalí veo que tampoco tiene atrapado el error como corresponde.

El muchos foros como por ejemplo ‘Stack Overflow’ hablan de esto pero todas las soluciones que proponen ninguna funciona. La única que se me ocurre es antes de hacer la conexión verificar si el servicio está levantado y ahí atrapar el error. Aunque podría no funcionar en una máquina remota.

Si saben de alguna otra forma sería bueno que lo comenten, ya que los otros métodos que se usan para levantar los datos de lo que se quiere también tienen el mismo problema de dejar colgada la aplicación si por alguna razón no está activo el servicio de WMI.

Saludos,
El Rayo

Neftali [Germán.Estévez]
10-05-2022, 12:42:30
Si lo pruebas en local, verás que es fácil obtener un error de tipo EOleException si el servicio no está disponible. Incluso el mensaje que obtienes te lo deja claro.
Baja el servicio desde la administración de servicios y luego ponlo deshabilitado, para evitar que se arranque automáticamente cuando hagas la petición (ya que por defecto tiene macado araque automático.

Ten en cuenta que al "bajar" el servicio, es posible que se paren otros que dependen de él y que luego tendrás que arrancar manualmente (o reiniciar la máquina para que se vuelvan a arrancar automáticamente).

Si pruebas con un código como este:

uses
SysUtils, ActiveX, ComObj, Variants, VCL.Forms, WinAPI.Windows, VCL.Dialogs;
...


var
FSWbemLocator : OLEVariant;
FWMIService : OLEVariant;
begin
FSWbemLocator := CreateOleObject('WbemScripting.SWbemLocator');
try
FWMIService := FSWbemLocator.ConnectServer(WbemComputer, 'root\CIMV2', WbemUser, WbemPassword);
except
on E:exception do begin

MessageDlg('Error, no se puede conectar al servicio de wmi. ' + sLineBreak +
'(' + E.ClassName + ') ' + E.Message,
mtError, [mbOK], 0);
Exit;
end;
end;
...

Verás que al ejecutarlo con el servicio parado (y deshabilitado) te genera el siguiente error:

https://i.imgur.com/8kdCyhd.png

Neftali [Germán.Estévez]
10-05-2022, 12:44:59
Otro tema es que al intentar arrancar el servicio (que la petición de conexión lo intenta "levantar") se te esté quedando bloqueado por otros temas.
Necesidad de permisos, temas de red, elevación de permisos, servicio no disponible o "roto",...

Ya tendrías que averiguar, ese bloqueo o "cuelgue" porqué está pasando.

elrayo76
15-05-2022, 01:23:57
Buenas, perdón por la demora en responder pero estuve de viaje estos dos últimos días por trabajo.


Neftalí, esto es lo que hago para conectar con WMI.



var
objLocator: ISWbemLocator;

begin
FConnected := False;
objLocator := CoSWbemLocator.Create;

try
// Hace la conexión del WMI con 'root\cimv2'.
FServices := objLocator.ConnectServer(FHost, GetRoot, FUser, FPassword, '', '', 0, nil);
except
on e: Exception do
MessageError(SysErrorMessage(GetLastError));
end;




El tema es que si le doy pausa o lo detengo al servicio no llega nunca al error. Se queda colgado en donde se carga la variable FService


¿Tendrá algo que ver la forma en que llamo a la clase CoSWbemLocator? igual veo que es como lo haces tú en las GLibWMI.


Como ya comente antes, en algunos otros foros preguntan por este mismo problema y ninguno puede dar una solución al respecto. Por eso mismo es que pienso que hay que ver por otro lado si el servicio esta corriendo antes de intentar la conexión.



Saludos,
El Rayo