TidHTTPServer decode content type multipart/form-data
Come tutti sappiamo, creare un webServer con Delphi è semplicissimo, cosa meno banale è invece decodificare una post che che contiene un multipart/form-data.
Quando una paginetta http esegue una http submit da un form in metodo post, magari con un file in upload il browser genera uno stream codificato in UTF8 in cui ci sono tutti i parametri e tutti gli allegati.
Il nostro server HTTP riceve tale stream nella variabile ARequestInfo.PostStream nell’evento “onCommandGet” (non bisogna farsi forviare dal nome, qualsiasi sia il metodo http utilizzato questo evento viene scatenato lato server).
Ora come maneggiare questa informazione è tutt’altro che banale, o si parsa lo stream tenendo conto degli standard RFC.
Personalmente ho utilizzato una classe già presente in Delphi nella tool dei componenti Indy che è nata per parsare stream molto simili : quelli MIME dei mesaggi di posta (son la stessa cosa alla fine).
Di seguito un esempio, il codice del comando onCommandGet l’ho implementato che restituisce una paginetta Http con un form per uploadare un file se viene chiamata la base path, quando invece viene chiamato basepath/upload gestisce la post del suddetto form.
L’esempio che riporto di seguito, salva lo stream con il nome file che gli arriva dal browser su un file nella cartella dell’applicazione. Ovviamente questo è un codice di esempio, se dovete implementare il webserver per un uso professionale bisogna considerare che è multi thread quindi le operazioni andrebbero sincronizzate con il thread principale, se non lo fate rischiate che si inchiodi prima o poi.
Tools : Delphi XE5 e Indy 10
E ora il codice:
interface
uses
System.SysUtils, System.Classes, Web.Win.Sockets, IdBaseComponent,
IdComponent, IdCustomTCPServer, IdCustomHTTPServer, IdHTTPServer, IdContext,
IdTCPServer,System.Generics.Collections, Data.DB, Datasnap.DBClient,IdHeaderList,idGlobal,
IdIntercept,IdMessage,IdMessageCoderMIME,IdMessageCoder,IdGlobalProtocols;
…
procedure TdmNet.IdHTTPServerCommandGet(AContext: TIdContext;
ARequestInfo: TIdHTTPRequestInfo; AResponseInfo: TIdHTTPResponseInfo);
Var
ms : TMemoryStream;
newdecoder, Decoder: TIdMessageDecoder;
boundary, startboundary : String;
msgEnd : Boolean;
tmp : String;
I : Integer;
fname : String;
tsValues : TStringList;
begin
i := 0;
if pos(‘upload’,lowercase(ARequestInfo.Document)) > 0 then
begin
If ARequestInfo.PostStream = nil then
AResponseInfo.ContentText := ‘
unparsed:’ + ARequestInfo.UnparsedParams +
‘
Encoding:’ + ARequestInfo.ContentEncoding + ARequestInfo.RawHeaders.Values[‘Content-Type’] +
‘
HashCode:’ + IntToStr(ARequestInfo.GetHashCode) +
‘
Params:’ + ARequestInfo.Params.Text + ‘ –>stream nullo
‘
Else
ARequestInfo.PostStream.Position := 0;
msgEnd := False;
boundary := ExtractHeaderSubItem(ARequestInfo.ContentType, ‘boundary’,QuoteHTTP);
startboundary := ‘–‘ + boundary;
repeat
tmp := ReadLnFromStream(ARequestInfo.PostStream, -1, True);
until tmp = startboundary;
decoder := TIdMessageDecoderMIME.Create(nil);
TIdMessageDecoderMIME(decoder).MIMEBoundary := boundary;
tsValues := TStringList.Create;
try
repeat
decoder.SourceStream := ARequestInfo.PostStream;
decoder.FreeSourceStream := false;
decoder.ReadHeader;
inc(I);
case Decoder.PartType of
mcptAttachment,mcptText : begin
ms := TMemoryStream.Create;
ms.Position := 0;
newdecoder := Decoder.ReadBody(ms,msgEnd);
tmp := Decoder.Headers.Text;
fname := decoder.Filename;
decoder.Free;
decoder := newdecoder;
if decoder <> nil then
TIdMessageDecoderMIME(decoder).MIMEBoundary := boundary;
sleep(100);
if fname <> ” then
begin
ms.SaveToFile(fname);
//msgEnd := true;
end
else
begin
ms.SaveToFile(inttostr(i) + ‘.txt’);
end;
ms.Free;
end;
mcptIgnore: Begin
try
FreeAndNil(decoder);
decoder := TIdMessageDecoderMIME.Create(nil);
TIdMessageDecoderMIME(decoder).MIMEBoundary := boundary;
finally
ms.Free;
end;
End;
mcptEOF: begin FreeAndNil(decoder); msgEnd := True end;
end;
until (decoder = nil) or(msgEnd);
finally
if decoder <> nil then
decoder.Free;
end;
AResponseInfo.ContentText := AResponseInfo.ContentText + ‘‘;
AResponseInfo.ContentText := ‘
unparsed:’ + ARequestInfo.UnparsedParams +
‘
Encoding:’ + ARequestInfo.ContentEncoding + ‘
Conte’ + ARequestInfo.RawHeaders.Values[‘Content-Type’] +
‘
HashCode:’ + IntToStr(ARequestInfo.GetHashCode) +
‘
Params:’ + ARequestInfo.Params.Text + ‘ –>stream nullo
‘;
end
Else
Begin
AResponseInfo.ContentText :=
‘ ‘ + #10#13 +
‘ ‘ + #10#13 +
‘ ‘ + #10#13 +
‘ ‘ + #10#13 +
‘Pagina di prova ‘ + #10#13 +
‘ ‘ + #10#13 +
‘ ‘ + #10#13 +
‘ ‘ + #10#13 +
‘
Test multipart from Synaptica Srl
‘ + #10#13 +
‘
‘ + #10#13 +
‘ ‘ + #10#13 +
‘ ‘;
End;
end;