* * * *

Privacy Policy

Blog italiano

Clicca qui se vuoi andare al blog italiano su Lazarus e il pascal.

Forum ufficiale

Se non siete riusciti a reperire l'informazione che cercavate nei nostri articoli o sul nostro forum vi consiglio di visitare il
Forum ufficiale di Lazarus in lingua inglese.

Lazarus 1.0

Trascinare un file nel programma
DB concetti fondamentali e ZeosLib
Recuperare codice HTML da pagina web
Mandare mail con Lazarus
Stabilire il sistema operativo
Esempio lista in pascal
File INI
Codice di attivazione
Realizzare programmi multilingua
Lavorare con le directory
Utilizzare Unità esterne
TTreeView
TTreeview e Menu
Generare controlli RUN-TIME
LazReport, PDF ed immagini
Intercettare tasti premuti
Ampliare Lazarus
Lazarus e la crittografia
System Tray con Lazarus
UIB: Unified Interbase
Il file: questo sconosciuto
Conferma di chiusura di un applicazione
Liste e puntatori
Overload di funzioni
Funzioni a parametri variabili
Proprietà
Conversione numerica
TImage su Form e Panel
Indy gestiore server FTP lato Client
PopUpMenu sotto Pulsante (TSpeedButton)
Direttiva $macro
Toolbar
Evidenziare voci TreeView
Visualizzare un file Html esterno
StatusBar - aggirare l'errore variabile duplicata
Da DataSource a Excel
Le permutazioni
Brute force
Indy 10 - Invio email con allegati
La gestione degli errori in Lazarus
Pascal Script
Linux + Zeos + Firebird
Dataset virtuale
Overload di operatori
Lavorare con file in formato JSON con Lazarus
Zeos ... dietro le quinte (prima parte)
Disporre le finestre in un blocco unico (come Delphi)
Aspetto retrò (Cmd Line)
Lazarus 1.0
Come interfacciare periferica twain
Ubuntu - aggiornare free pascal e lazarus
fpcup: installazioni parallele di lazarus e fpc
Free Pascal e Lazarus sul Raspberry Pi
Cifratura: breve guida all'uso dell'algoritmo BlowFish con lazarus e free pascal.
Creare un server multithread
guida all'installazione di fpc trunk da subversion in linux gentoo
Indice
DB concetti fondamentali e connessioni standard
Advanced Record Syntax
DB concetti fondamentali e DBGrid
DB concetti fondamentali e TDBEdit, TDBMemo e TDBText
Advanced Record Syntax: un esempio pratico
Superclasse form base per programmi gestionali (e non)
Superclasse form base per programmi gestionali (e non) #2 - log, exception call stack, application toolbox
Superclasse form base per programmi gestionali (e non) #3 - traduzione delle form
Superclasse form base per programmi gestionali (e non) #4 - wait animation
Un dialog per la connessione al database:TfmSimpleDbConnectionDialog
Installare lazarus su mac osx sierra
immagine docker per lavorare con lazarus e free pascal
TDD o Test-Driven Development
Benvenuto! Effettua l'accesso oppure registrati.
Aprile 16, 2024, 08:36:07 am

Inserisci il nome utente, la password e la durata della sessione.

71 Visitatori, 2 Utenti
 

Autore Topic: estrarre il contenuto di un file p7m  (Letto 4671 volte)

nomorelogic

  • Global Moderator
  • Hero Member
  • *****
  • Post: 2870
  • Karma: +20/-4
estrarre il contenuto di un file p7m
« il: Marzo 13, 2019, 07:50:47 pm »
salve gente
volevo segnalare questo thread sul forum ufficiale
http://forum.lazarus.freepascal.org/index.php/topic,44656.msg314145.html#msg314145

lo scopo è sempre quello del thread
http://www.lazaruspascal.it/index.php?topic=2113.msg13052;topicseen#new

ma se si riesce non sarebbe male usare la libreria openssl

Imagination is more important than knowledge (A.Einstein)

bonmario

  • Hero Member
  • *****
  • Post: 1300
  • Karma: +10/-1
Re:estrarre il contenuto di un file p7m
« Risposta #1 il: Marzo 13, 2019, 08:13:45 pm »
Ciao,

se lo scopo di queste 3 righe:

Codice: [Seleziona]
   sl:=TStringList.Create;
   sl.LoadFromFile(Edit1.Text);
   sfile:=sl.Text;
   sl.Free;

è quello di salvare l'intero file in una stringa, credo che usare la TStringList ti potrebbe creare problemi: sl.Text potrebbe contenere i caratteri di fine linea, che di solito vengono usati tra una riga della TStringList e la successiva.

Secondo me, ti conviene usare uno TFileStream, anche se provando ad usarlo p rimane sempre non assegnata.

Questo è il tuo codice "riadattato":
Codice: [Seleziona]
procedure TForm1.Sproteggi2(NomeFilePart, NomeFileDest: String);
var FBIO: PBIO;
    sfile: AnsiString;
    FileStreamPart:TFileStream;
    i: integer;
    p: SslPtr;
    DimFile, BytesLetti:Int64;
begin
  //Apro il file di partenza e faccio in modo che non sia vincolato dal processo in corso
  FileStreamPart:=TFileStream.Create(NomeFilePart, CstApriFileSolaLett);
  try
    DimFile:=FileStreamPart.Size;
    sfile:=StringOfChar(#32, DimFile);
    BytesLetti:=FileStreamPart.Read(PChar(sfile)^, DimFile);

    if (BytesLetti <> DimFile) then begin
      EmettiErrore(0, 'Problemi durante la lettura del file' + NomeFilePart);
      Halt;
    end else begin
      fbio:=BioNew(BioSMem);
      i:=BioRead(FBIO, sfile, Length(sfile));
      p:= d2iPKCS12bio(FBIO, nil);
      BioFreeAll(FBIO);
    end;
  finally
    FreeAndNil(FileStreamPart);
  end;
end;

P.S. Dopo l'esecuzione dell'istruzione
Codice: [Seleziona]
i:=BioRead(FBIO, sfile, Length(sfile));
i vale "-1", quindi probabilmente il problema nasce lì.

Ciao, Mario
« Ultima modifica: Marzo 13, 2019, 08:16:27 pm da bonmario »

giuian

  • Newbie
  • *
  • Post: 14
  • Karma: +0/-0
Re:estrarre il contenuto di un file p7m
« Risposta #2 il: Marzo 14, 2019, 11:05:41 am »
Questa funziona
 
fBIO := BIO_new_file(nomefile, 'r');  // Read from stream

nomorelogic

  • Global Moderator
  • Hero Member
  • *****
  • Post: 2870
  • Karma: +20/-4
Re:estrarre il contenuto di un file p7m
« Risposta #3 il: Marzo 14, 2019, 11:31:08 am »
grazie, provo subito
Imagination is more important than knowledge (A.Einstein)

bonmario

  • Hero Member
  • *****
  • Post: 1300
  • Karma: +10/-1
Re:estrarre il contenuto di un file p7m
« Risposta #4 il: Marzo 14, 2019, 07:57:01 pm »
Questa funziona
 
fBIO := BIO_new_file(nomefile, 'r');  // Read from stream

Puoi postare il codice?
Io ho fatto così, ma p rimane sempre "vuota"
Codice: [Seleziona]
        fBIO := BIO_new_file(PChar(NomeFilePart), 'r');  // Read from stream
        p:= d2iPKCS12bio(FBIO, nil);
        BioFreeAll(FBIO);

Ciao, MArio

giuian

  • Newbie
  • *
  • Post: 14
  • Karma: +0/-0
Re:estrarre il contenuto di un file p7m
« Risposta #5 il: Marzo 14, 2019, 11:07:07 pm »
Rimane vuota perché devi usare d2ipkcs7bio ...
Ci sono molte funzioni openssl che non sono presenti in openssl.pas, ma si possono aggiungere chiaramente.

nomorelogic

  • Global Moderator
  • Hero Member
  • *****
  • Post: 2870
  • Karma: +20/-4
Re:estrarre il contenuto di un file p7m
« Risposta #6 il: Marzo 15, 2019, 02:50:36 pm »
ho cercato di mettere un po' d'ordine nei sorgenti e li metto qua per chi voglia fare qualche esperimento
vabbè, ordine si fa per dire... :)

ora dovrei capire bene come procedere, non vi nego che per me il da farsi è ancora un po' oscuro, ma un passetto alla volta spero di arrivare in fondo

allo stadio attuale si riesce a leggere il file "partenza" e a passare il contenuto da una struttura BIO_in (input)
se non ho capito male ora bisognerebbe fare una trascodifica verso una struttura BIO_out

se qualcuno ne sa più di me mi chiarisca pure :)


Codice: [Seleziona]
procedure TForm1.Sproteggi2(NomeFilePart, NomeFileDest: String);
var FBIO_in, FBIO_out: PBIO;
    sfile: AnsiString;
    FileStreamPart:TFileStream;
    i: integer;
    p: SslPtr;
    DimFile, BytesLetti:Int64;
    pmem: PChar;
begin
   // init
   BytesLetti:=-1;

   // Apro il file di partenza e carico il centenuto in un buffer
  FileStreamPart:=TFileStream.Create(NomeFilePart, fmOpenRead);
  try
    DimFile:=FileStreamPart.Size;
    sfile:=StringOfChar(#0, DimFile);
    FileStreamPart.Position:=0;
    pmem:=addr(sfile[1]);
    BytesLetti:=FileStreamPart.Read(pmem^, DimFile);
  finally
    FreeAndNil(FileStreamPart);
  end;

  // test errore lettura
  if BytesLetti <> DimFile then
     exception.Create('Errore nella lettura del file sorgente');

  // inizio estrazione
  FBIO_in:=BioNew(BioSMem);
  FBIO_out:=BioNew(BioSMem);
  try
    // assegno buffer letto a BIO_IN
    i:=BioWrite(FBIO_in, sfile, BytesLetti);
    if i<=0 then
      exception.Create('Errore scrittura in BIO_in');

    // trascodifica in BIO_OUT
    // p:= d2iPKCS12bio(FBIO, nil);

  finally
    BioFreeAll(FBIO_in);
    BioFreeAll(FBIO_out);
  end;

end;
Imagination is more important than knowledge (A.Einstein)

bonmario

  • Hero Member
  • *****
  • Post: 1300
  • Karma: +10/-1
Re:estrarre il contenuto di un file p7m
« Risposta #7 il: Marzo 15, 2019, 06:32:19 pm »
Grazie !!!
Nel weekend vedo se riesco a fare qualcosa.

Ciao, Mario

bonmario

  • Hero Member
  • *****
  • Post: 1300
  • Karma: +10/-1
Re:estrarre il contenuto di un file p7m
« Risposta #8 il: Marzo 17, 2019, 11:40:37 am »
Niente da fare: ci ho provato, ma non ho ottenuto nessun risultato.

Ciao, Mario

nomorelogic

  • Global Moderator
  • Hero Member
  • *****
  • Post: 2870
  • Karma: +20/-4
Re:estrarre il contenuto di un file p7m
« Risposta #9 il: Marzo 18, 2019, 01:16:21 am »
sono riuscito a fare un'altro passetto:

Edit:
come già riportato da giuian

Citazione
Rimane vuota perché devi usare d2ipkcs7bio ...
Ci sono molte funzioni openssl che non sono presenti in openssl.pas, ma si possono aggiungere chiaramente.

la funzione d2iPKCS7bio non era esportata da ssl_openssl_lib e quindi mi sono arrangiato dichiarando il tutto a mano, seguendo il codice di d2iPKCS12bio.
Per ora non ho fatto modifiche in ssl_openssl_lib, è tutto nel codice quì sotto, lo riporto così come viene compilato nel mio progetto (un form vuoto con un bottone).

Girando per la rete ho trovato questo
https://gist.github.com/HungHuynh/3810968
probabilmente le prossime funzioni che serviranno saranno PEM_read_bio_PKCS7 e/o OBJ_obj2nid



Codice: [Seleziona]
implementation

{$R *.lfm}

uses synafpc;

type
  Td2iPKCS7bio = function(b:PBIO; Pkcs7: SslPtr): SslPtr; cdecl;
  // TPKCS7parse = function(p7: SslPtr; pass: PAnsiChar; var pkey, cert, ca: SslPtr): integer; cdecl;
  // TPKCS12free = procedure(p12: SslPtr); cdecl;

var
  _d2iPKCS7bio: Td2iPKCS7bio = nil;


  function GetProcAddr(module: HModule; const ProcName: string): SslPtr;
  begin
    Result := GetProcAddress(module, PChar(ProcName));
  end;


function d2iPKCS7bio(b:PBIO; Pkcs7: SslPtr): SslPtr;
begin
  if InitSSLInterface and Assigned(_d2iPKCS7bio) then
    Result := _d2iPKCS7bio(b, Pkcs7)
  else
    Result := nil;
end;

{ TForm1 }


procedure TForm1.Button2Click(Sender: TObject);
begin
   Sproteggi2(Edit1.Text, ExtractFilePath(Edit1.Text) + 'out.dat');
end;


procedure TForm1.Sproteggi2(NomeFilePart, NomeFileDest: String);
var FBIO_in, FBIO_out: PBIO;
    sfile: AnsiString;
    FileStreamPart:TFileStream;
    i: integer;
    p: SslPtr;
    DimFile, BytesLetti:Int64;
    pmem: PChar;
begin
   // init
   BytesLetti:=-1;
   _d2iPKCS7bio := Td2iPKCS7bio(GetProcAddress(SSLUtilHandle, 'd2i_PKCS7_bio'));
   // _PKCS12parse := GetProcAddr(SSLUtilHandle, 'PKCS12_parse');
   // _PKCS12free := GetProcAddr(SSLUtilHandle, 'PKCS12_free');

   // Apro il file di partenza e carico il centenuto in un buffer
  FileStreamPart:=TFileStream.Create(NomeFilePart, fmOpenRead);
  try
    DimFile:=FileStreamPart.Size;
    sfile:=StringOfChar(#0, DimFile);
    FileStreamPart.Position:=0;
    pmem:=addr(sfile[1]);
    BytesLetti:=FileStreamPart.Read(pmem^, DimFile);
  finally
    FreeAndNil(FileStreamPart);
  end;

  // test errore lettura
  if BytesLetti <> DimFile then
     exception.Create('Errore nella lettura del file sorgente');

  // inizio estrazione
  FBIO_in:=BioNew(BioSMem);
  FBIO_out:=BioNew(BioSMem);
  try
    // assegno buffer letto a BIO_IN
    i:=BioWrite(FBIO_in, sfile, BytesLetti);
    if i<=0 then
      exception.Create('Errore scrittura in BIO_in');

    // trascodifica pkcs7
    p:= d2iPKCS7bio(FBIO_in, nil);
    if Assigned(p) then
       ShowMessage('d2iPKCS7bio: eureka!');

  finally
    BioFreeAll(FBIO_in);
    BioFreeAll(FBIO_out);
  end;

end;


end.
« Ultima modifica: Marzo 18, 2019, 01:26:20 am da nomorelogic »
Imagination is more important than knowledge (A.Einstein)

giuian

  • Newbie
  • *
  • Post: 14
  • Karma: +0/-0
Re:estrarre il contenuto di un file p7m
« Risposta #10 il: Marzo 18, 2019, 06:32:43 am »
Si proprio così.
Ora bisogna vedere la dimensione del contenuto del pkcs7, a me puntava ad un record vuoto.
La sequenza di funzioni da usare adesso è quella di questo esempio delphi

Codice: [Seleziona]
// Open a PKCS7 file
procedure TPKCS7.Open(Filename: string);
var
  p7file: pBIO;
  objectType: integer;
begin
p7file := BIO_new(BIO_s_file());
if p7file = nil then
  raise EOpenSSL.Create('Unable to create a file handle.');
BIO_read_filename(p7file, PChar(Filename));
if (fEncoding = auto) or (fEncoding = DER) then
  begin
  fPkcs7 := d2i_PKCS7_bio(p7file, nil);
  if (fPkcs7 = nil) and (fEncoding = auto) then
    BIO_reset(p7file);
  end;
if ((fPkcs7 = nil) and (fEncoding = auto)) or (fEncoding = PEM) then
  begin
  fPkcs7 := PEM_read_bio_PKCS7(p7file, nil, nil, nil);
  if (fPkcs7 = nil) and (fEncoding = auto) then
    BIO_reset(p7file);
  end;
if ((fPkcs7 = nil) and (fEncoding = auto)) or (fEncoding = SMIME) then
  begin
  fPkcs7 := SMIME_read_PKCS7(p7file, fDetachedData);  // &indata ????
  end;
if fPkcs7 = nil then
  raise EOpenSSL.Create('Unable to read PKCS7 file');
if p7file <> nil then
  BIO_free(p7file);
objectType := OBJ_obj2nid(fPkcs7.asn1_type);
case objectType of
  NID_pkcs7_signed: fCerts := fPkcs7.sign.cert;
  NID_pkcs7_signedAndEnveloped: fCerts := fPkcs7.signed_and_enveloped.cert;
  end;
end;

procedure TPKCS7.SaveContent(Filename: String);
var
  p7bio, contentfile: pBIO;
  sinfos: pSTACK_OFPKCS7_SIGNER_INFO;
  i: integer;
  buffer: array [0..4096] of char;
begin
if fPkcs7 = nil then
  raise EOpenSSL.Create('No PKCS7 content.');
if OBJ_obj2nid(fPkcs7.asn1_type) <> NID_pkcs7_signed then
  raise EOpenSSL.Create('Wrong PKCS7 format.');
if (PKCS7_get_detached(fPkcs7) <> nil)
    and (fDetachedData = nil) then
  raise EOpenSSL.Create('PKCS7 has no content.');
sinfos := PKCS7_get_signer_info(fPkcs7);
if (sinfos = nil) or (sk_num(sinfos) = 0) then
  raise EOpenSSL.Create('No signature data.');
contentfile := BIO_new(BIO_s_file());
if BIO_write_filename(contentfile, PChar(Filename)) <= 0 then
  raise EOpenSSL.Create('Error creating output file.');
p7bio := PKCS7_dataInit(fPkcs7, fDetachedData);
repeat
  i := BIO_read(p7bio, @buffer, SizeOf(buffer));
  if i > 0 then
    BIO_write(contentfile, @buffer, i);
until i <= 0;

if fDetachedData <> nil then
  BIO_pop(p7bio);
BIO_free_all(p7bio);
BIO_free(contentfile);
end;

In delphi dichiarano una classe

Codice: [Seleziona]
TPKCS7 = class
  private
    fEncoding: TEncoding;
    fPkcs7: pPKCS7;
    fCerts: pSTACK_OFX509;
    fDetachedData: pBIO;
  protected
    function countCerts: integer;
    function getCert(i: integer): TX509Certificate;
  public
    constructor Create;
    destructor Destroy; override;
    property Encoding: TEncoding read fEncoding write fEncoding default auto;
    property CountCertificate: integer read countCerts;
    property Certificate[Index: Integer]: TX509Certificate read getCert;
    procedure Open(FileName: string);
    procedure Save(Filename: String); overload;
    procedure Save(Filename: String; Encoding: TEncoding); overload;
    procedure SaveContent(Filename: String);
    function VerifyData: boolean; overload;
    function VerifyData(Content: pointer): boolean; overload;
  end;

Ma nel mio adattamento qualcosa non va.
La funzione di Openssl è
#define PKCS7_get_detached    (         p   )       PKCS7_ctrl(p,PKCS7_OP_GET_DETACHED_SIGNATURE,0,NULL)


#define PKCS7_OP_SET_DETACHED_SIGNATURE 1
#define PKCS7_OP_GET_DETACHED_SIGNATURE 2

« Ultima modifica: Marzo 18, 2019, 06:41:25 am da giuian »

giuian

  • Newbie
  • *
  • Post: 14
  • Karma: +0/-0
Re:estrarre il contenuto di un file p7m
« Risposta #11 il: Marzo 26, 2019, 07:18:34 pm »
Signori io passo .... fDetachedData vale sempre 0 ...
A questo punto mi arrendo.
Grazie per averci provato.

 

Recenti

How To

Utenti
  • Utenti in totale: 785
  • Latest: gmax
Stats
  • Post in totale: 18769
  • Topic in totale: 2232
  • Online Today: 80
  • Online Ever: 900
  • (Gennaio 21, 2020, 08:17:49 pm)
Utenti Online
Users: 2
Guests: 71
Total: 73

Disclaimer:

Questo blog non rappresenta una testata giornalistica poiché viene aggiornato senza alcuna periodicità. Non può pertanto considerarsi un prodotto editoriale ai sensi della legge n. 62/2001.