Cita:
Empezado por razorxxx
Dado que aún no he conseguido hacer funcionar el procedimiento OnBeforePost del HTTPRIO en Delphi Rio, estoy tratando de prescindir de ese método, y asignando (justo antes de la petición al WebService) la llamada al HTTPRIO1.HTTPWebNode.ClientCertificate.SerialNum, usando la función
Código:
function BUSCAR_CERTIFICADO(Nombre_Certificado: String): string;
que ya han mencionado en un post anterior.
Sin embargo, cuando hago
Código:
HTTPRIO1.HTTPWebNode.ClientCertificate.SerialNum := Buscar_Certificado(Cert.Text)
, me devuelve una cadena vacía. Según me he estado informando, esto solo sirve para cuando el certificado está en el almacén de certificados del equipo, pero no para cuando trabajamos con un fichero PFX/P12.
¿Alguien tiene alguna idea acerca de cargar un .pfx o .p12 en un objeto THTTPRIO, usando HTTPWebNode.ClientCertificate y prescindiendo de la llamada al OnBeforePost?
|
Buenas a todos.
Hasta ahora, para cargar un certificado desde archivo antes de realizar una petición con un objeto HTTPRIO utilizaba la librería capicom.dll desde el método OnBeforePost. Pero como comenté en su día, el método había cambiado desde la versión 10.3 de Delphi, ya que el Data: Pointer lo cambiaron a Client: THTTPClient.
Pues bien, después de mucho batallar, he conseguido hacerlo funcionar y además prescindiendo de la antigua librería capicom, simplemente usando métodos nativos de la librería crypt32.dll de Windows.
Primeramente, antes de llamar al bloque "implementation" de nuestro código, poner la siguiente línea:
Código Delphi
[-]function PFXImportCertStore(var pPFX: CRYPT_DATA_BLOB; szPassword: LPCWSTR; dwFlags: DWORD): HCERTSTORE; stdcall; external 'Crypt32.dll';
Luego en el método
Código Delphi
[-]HTTPRIOHTTPWebNodeBeforePost(const HTTPReqResp: THTTPReqResp; Client: THTTPClient)
se pone lo siguiente:
Código Delphi
[-]
const
PKCS12_INCLUDE_EXTENDED_PROPERTIES = $0010;
CERT_COMPARE_HAS_PRIVATE_KEY = 21;
CERT_FIND_HAS_PRIVATE_KEY = CERT_COMPARE_HAS_PRIVATE_KEY shl CERT_COMPARE_SHIFT;
var
Almacen: HCERTSTORE;
Certificado: PCERT_CONTEXT;
DataBlob: CRYPT_DATA_BLOB;
PFX: TBytes;
begin
Almacen := nil;
Certificado := nil;
PFX := TFile.ReadAllBytes(Cert.Text);
Try
DataBlob.cbData := Length(PFX);
DataBlob.pbData := @PFX[0];
Almacen := PFXImportCertStore(DataBlob, PWideChar(Pwd.Text), PKCS12_INCLUDE_EXTENDED_PROPERTIES);
If not Assigned(Almacen) Then
Salida.Lines.Add('[ERROR] No se pudo importar el certificado seleccionado.') Else
Begin
Certificado := CertFindCertificateInStore(Almacen, X509_ASN_ENCODING, 0, CERT_FIND_HAS_PRIVATE_KEY, nil, nil);
If not Assigned(Certificado) Then
Salida.Lines.Add('[ERROR] No se pudo encontrar el certificado digital en el contexto actual, o bien no tiene clave privada.'); End;
Finally
If Assigned(Certificado) Then CertFreeCertificateContext(Certificado);
If Assigned(Almacen) Then CertCloseStore(Almacen, 0);
End;
end;
, sabiendo que la ruta completa al certificado .pfx o .p12 está en un campo TEdit llamado 'Cert', y su contraseña en otro TEdit llamado 'Pwd'.
No olviden asignar este método al OnBeforePost del objeto HTTPRIO que realiza la petición al webservice, de lo contrario recibirán el error
Received content of invalid Content-Type setting: text/html - SOAP expects "text/xml".
Espero haber ayudado. Saludos.