FTP | CCD | Buscar | Trucos | Trabajo | Foros |
|
Registrarse | FAQ | Miembros | Calendario | Guía de estilo | Temas de Hoy |
|
Herramientas | Buscar en Tema | Desplegado |
#1
|
|||
|
|||
Leer certificado pfx para webservice
Hola buenas tardes:
Mi problema es el siguiente: Quiero cargar un certificado pfx sin tenerlo instalado en windows para utilizarlo en un componente HTTPRio y acceder a un webservice. Estoy usando RAD STUDIO 10.4 He estado buscando en el foro y ya hay un tema al respecto: https://www.clubdelphi.com/foros/showthread.php?t=95356 La cuestión es que no puedo usar CAPICOM. En este tema se ofrece una solución utilizando la unidad JwaWinCrypt. He buscado información y creo que esta unidad está en el componente JWAPI pero no sé como obtenerlo ni como instalarlo. ¿Alguien me puede ayudar, por favor? Gracias |
#2
|
|||
|
|||
Buenas.
En el ejemplo que puse en el otro hilo comentaba que la mayoría de las funciones estaban declaradas en la unidad CertHelper de Delphi 2010, prueba a ver si existe la unidad en 10.4 y dime que funciones te faltan y te paso las declaraciones. |
#3
|
|||
|
|||
Lo revisé un poco y aquí te dejo una versión probada en Delphi 2010 usando sólo la unidad CertHelper que ya viene de serie.
|
#4
|
|||
|
|||
Muchas gracias Garada, muy amable.
He probado el código y el compilador no reconoce la función InternetSetOption. Seguramente está en otra unidad y me falta añadirla al uses. InternetSetOption(Data, INTERNET_OPTION_CLIENT_CERT_CONTEXT, pCert, SizeOf(CERT_CONTEXT)); Tampoco reconoce la variable Data que no sé de donde la sacas porque no está declarada. Si puedes aclararme esto te estaría muy agradecido. |
#5
|
|||
|
|||
Bueno ya sé de donde vienen la función InternetSetOption. De la unidad WinINet.
Ahora, de momento, solo me falta saber el valor de la variable data para que el procedimiento se pueda compilar. |
#6
|
|||
|
|||
data es un parámetro del evento BeforePost del WebNode del HTTPRIO.
Al menos en D2010 viene declarado así:
Si en D10.4 es diferente pásame la declaración a ver que ha cambiado. |
#7
|
|||
|
|||
Ahora es:
HTTPRIO1HTTPWebNode1BeforePost(const HTTPReqResp: THTTPReqResp; Client: THTTPClient); |
#8
|
|||
|
|||
Pues por lo que veo (he instalado la versión Community Edition) en la versión 10.4 han hecho ese cambio.
Por ahora te puedo decir que todas las funciones del API para gestionar los certificados que estaban en CertHelper y las que te declaré en el ejemplo las vas a encontrar en System.Net.HttpClient.Win No he podido averiguar que hacer con el THTTPClient que si tiene para elegir el certificado de la lista de los instalados en el sistema pero ni idea como pasarle un certificado de un fichero. En un foro en inglés hablan de modificar un evento (DoClientCertificateAccepted) para que haga lo que te interesa. Si averiguo algo más actualizo información. A mi en parte me interesa por si algún día me obligan a actualizar versión de Delphi. |
#9
|
|||
|
|||
Ok, gracias.
Lo ideal sería cargar directamente el certificado del fichero pfx pero una solución menor sería seleccionar directamente el certificado instalado en almacén de windows sin que aparezca la ventana que le pregunta al cliente. Voy a buscar información al respecto, aunque por lo que dices sabes como hacerlo. Gracias por tu interés |
#10
|
|||
|
|||
He conseguido hacerlo, pero no muy elegantemente.
Delphi no da ninguna oportunidad para cargar un certificado, sólo elegir de la lista de instalados en el equipo. La clase que se ocupa de todo no es pública y no se puede acceder a ella. En un foro en inglés alguien comentó que parcheaba modificando las VMT (Virtual Method Table) para que llamara a su función. Pero eso se escapa de mis posibilidades. El método bruto es copiar la unidad System.Net.HttpClient.Win en la carpeta de tu proyecto y Delphi la usará en vez de la suya. El archivo está en "C:\Program Files (x86)\Embarcadero\Studio\22.0\source\rtl\net\System.Net.HttpClient.Win.pas" En tu proyecto, en el evento HTTPRIO1HTTPWebNode1NeedClientCertificate pones el parametro AnIndex a cero. Con eso le indicas que quieres usar el primer certificado ya que si lo dejas como viene (-1) indicas que no elijes ninguno y no saltará la función que modificamos para cargar el certificado.
Y en la unidad que copiaste, System.Net.HttpClient.Win buscas la función TWinHTTPClient.DoClientCertificateAccepted y le pones el código que se usaba con el D2010
|
#11
|
|||
|
|||
Hola Garada, buenos días.
Muchas gracias por tu ayuda. Es maravillosa. Una duda. Has hecho cambios cambios en el código respecto a la primera versión: DataBlob: CRYPT_BIT_BLOB; por DataBlob: CRYPT_DATA_BLOB; PKCS12_INCLUDE_EXTENDED_PROPERTIES = $0010 por 0 Es que entonces habría que cambiar la función: Código:
function PFXImportCertStore(var pPFX: CRYPT_BIT_BLOB; szPassword: LPCWSTR; dwFlags: DWORD): HCERTSTORE; stdcall; external 'Crypt32.dll'; De todas formas si pongo Código:
function PFXImportCertStore( var pPFX: CRYPT_BIT_BLOB; szPassword: LPCWSTR; dwFlags: DWORD): HCERTSTORE; stdcall; external 'Crypt32.dll'; y cambio el tipo del datablob por un CRYPT_BIT_BLOB me sigue dando un error de "types of actual and formal var parameters must be identical" |
#12
|
|||
|
|||
Hola,
He cambiado el sitio donde declaro la función PFXImportCertStore. Lo he cambiado aquí: Código:
function CryptUIDlgSelectCertificateFromStore(hCertStore: HCERTSTORE; hwnd: HWND; pwszTitle: LPCWSTR; pwszDisplayString: LPCWSTR; dwDontUseColumn: DWORD; dwFlags: DWORD; pvReserved: Pointer): PCCERT_CONTEXT; stdcall; forward; {$NODEFINE CryptUIDlgSelectCertificateFromStore} function PFXImportCertStore( var pPFX: CRYPT_BIT_BLOB; szPassword: LPCWSTR; dwFlags: DWORD): HCERTSTORE; stdcall; external 'Crypt32.dll'; Lo que pasa es que me da otros errores: No me reconoce ni Enter ni Exit del tipo Monitor: Código:
TMonitor.Enter(ARequest); Código:
TMonitor.Exit(ARequest); En dicha unidad: System.Net.HttpClient.Win; |
#13
|
|||
|
|||
Al final he dejado el evento así:
Quería comprobar que existía el archivo del certificado antes de leerlo pero si declaro en el uses Vcl.Forms para que reconozca el objeto Application no me compila. |
#14
|
|||
|
|||
No hay que darlas.
Los cambios viene a que en esa unidad (System.Net.HttpClient.Win) ya viene declarado todo lo que necesitas y que antes estaba en Certhelper o declaradas a mano. PKCS12_INCLUDE_EXTENDED_PROPERTIES es opcional, sólo informa que se quiere importar el certificado con las propiedades extendidas. Este flag sí que no estaba declarado en la unidad y como no afecta al funcionamiento, lo descarté. El resto pues lo comentado, ya está en la unidad sin necesidad de añadir nada al USES. De hecho me extraña que no te diera error por declarar dos veces la función PFXImportCertStore Si tienes que añadir alguna unidad al USES, añádela al principio de la lista para que no afecte a los originales (tendrán preferencia sobre las tuyos) Cita:
|
#15
|
|||
|
|||
Entonces es que supongo que no tenemos el mismo código en la unidad System.Net.HttpClient.Win
En el que tengo yo en el que no aparece por ningún lado la función PFXImportCertStore, como te decía la he tenido que declarar yo y si declaro la unidad Vcl.Forms
me salen errores:
Undeclared identifier: 'Enter' |
#16
|
|||
|
|||
Ok, yo lo he probado en Delphi 11 ya que la 10.4 ya me había caducado.
Es correcto entonces añadir la función a mano. Para el error, cambia Tmonitor por System.Tmonitor |
#17
|
|||
|
|||
ok, muchas gracias.
Con el system me funciona bien. Ahora me piden otra cosa al respecto que me he jodido todo el invento. Poder seleccionar un certificado según un parámetro que se le pasaría al componente HTTPRIO. Si condición elegimos el certificado 1 sino el 2 El problema es que al estar el procedimiento en la unidad System.Net.HttpClient.Win y no en el componente HTTPRIO no puedo pasarle ningún parámetro. He pensado en pasar el procedimiento al evento OnNeedClientCertificate pero los parámetros no son los mismos:
El problema estaría aquí:
|
#18
|
|||
|
|||
Si añades al USES la unidad con el HTTPRIO puedes acceder a él.
Si quieres separarlo, podrías usar el parámetro AnIndex que es numérico y ya lees el certificado que te interese según su valor. |
#19
|
|||
|
|||
El problema es que he creado una clase heredada de THTTPRIO
TMi_HTTPRIO = class(THTTPRIO) Y como envío y recibo varios tipos de XML, voy creando distintos HTTPRIOs para cada tipo de XML que se envía, con lo cual no puedo acceder a él desde System.Net.HttpClient.Win, principalmente porque no sé ni cuál es. Otra cosa, he comentado el procedimiento DoClientCertificateAccepted que he modificado en la unidad System.Net.HttpClient.Win y la aplicación elige el primer certificado que encuentra en el almacén y lo utiliza sin preguntar. O sea que realmente no sé si realiza alguna función. ¿Si desinstalo los certificados que tengo instalados cogerá el certificado que le indico del fichero pfx? De todas formas sin poder seleccionarlo desde esta unidad no me sirve para mucho. Por otra parte si utilizo el evento httWebNodeNeedClientCertificate cómo sé qué certificado tengo en según el parámetro AnIndex. Es decir si tengo varios instalados como sé cuál es cuál. |
#20
|
|||
|
|||
Sin ver todo en su conjunto, algunas ideas.
1 - Hacer publica la clase TWinHTTPClient para que puedas acceder a ella desde los eventos del httprio y asignarle p.e. una propiedad con el fichero o el stream del certificado. 2 - Pasar el HTTPRio como Integer a través del parametro AnIndex para acceder a él desde el TWinHttpClient Yo haría la 1 Y aparte, sobre el AnIndex. En la unidad sin modificar, te pasa la lista de certificados y devuelves el índice del que quieres usar o -1 si no eliges. En la versión modificada que te pase, le da igual cual elijas (excepto -1) pq va a cargar el certificado del fichero siempre. En la modificación puedes usar 1 o 2 para que después tu elijas tu certificado 1 o 2. (esa es una tercera opción) Última edición por Garada fecha: 02-10-2023 a las 23:05:58. |
|
|
Temas Similares | ||||
Tema | Autor | Foro | Respuestas | Último mensaje |
acceder a Webservice con certificado de cliente | iMia | Internet | 8 | 13-09-2022 12:20:58 |
Conectar Webservice con httpRio+Certificado | gasal | Internet | 2 | 20-07-2018 18:11:08 |
Como leer un TRemotable que proviene de un webservice | apicito | Internet | 17 | 02-09-2011 23:48:41 |
SOAP POST - Webservice con Certificado y SSL | JXJ | Varios | 5 | 09-05-2011 21:11:08 |
|