![]() |
Große Datei über WebBroker als 'application/octet-stream' zur Verfügung stellen
Hi,
kurzer Umgebungsüberblick. - Delphi Version 10.2 - WebBroker aktuell als eigenständiger Webserver lauffähig (später ISAPI Modul im IIS) - Dateien mit Größe 100mb + Also, ich versuche ein Funktion zu erstellen, die mir eine Datei auf dem Filesystem als 'application/octet-stream' zur Verfügung stellt. Funktioniert soweit auch schon. Verwende dafür einfach einen
Delphi-Quellcode:
.
TByteStream.LoadFromFile('FilePath')
das ganze sieht dann so aus:
Delphi-Quellcode:
Funktioniert Problemlos bis Dateien von ca. 50MB. Dann bekomme ich einen Socket Fehler 10054 Die Verbindung wurde von Peer zurückgesetzt. Ich nehme mal an, das hängt mit dem reservierten Speicher zusammen. Kennt das jemand und wie würde man das umgehen?
Response.Content := '';
Response.ContentType := 'application/octet-stream'; Response.ContentStream := TBytesStream.Create; TBytesStream(Response.ContentStream).LoadFromFile('FilePath'); Die Abfrage in Chunks aufteilen könnte helfen, aber ich weiß nicht wie ich das mit dem Websocket umsetzen kann, der Sendet ja zum schluss erst alles gebündelt raus und nicht schon zwischenzeitlich, oder? vG PJM |
AW: Große Datei über WebBroker als 'application/octet-stream' zur Verfügung stellen
Kenne Indy leider nicht. Timeoutfehler?
Wieso verwendest du nicht einen TFileStream und zeigst dann mit Contentstream auf diesen? Loadfromfile entfällt dann. Anderer Ansatz: Falls auf Clientseite ein Browser genutzt wird: Du könntest als Antwort auch einfach auf die Location des Files zeigen (vielleicht willst du das nicht). Also zum Beispiel Statuscode 302 (oder sonst einen Redirectcode) und Response.Location:=<filename> zurückgeben. |
AW: Große Datei über WebBroker als 'application/octet-stream' zur Verfügung stellen
Wie geht denn der Code weiter? Gibst du vielleicht "Response.ContentStream" selber frei (und das bevor er komplett übertragen ist)?
|
AW: Große Datei über WebBroker als 'application/octet-stream' zur Verfügung stellen
Zitat:
|
AW: Große Datei über WebBroker als 'application/octet-stream' zur Verfügung stellen
Zitat:
|
AW: Große Datei über WebBroker als 'application/octet-stream' zur Verfügung stellen
Ich habe rasch dies mit deinem Code getestet. Delphi 11.2. Win 11 Pro, aktueller IIS. Als CGI in IIS.
Delphi-Quellcode:
program filedownloadcgi;
{$APPTYPE CONSOLE} uses WebBroker, CGIApp, WebModuleUnit1 in 'WebModuleUnit1.pas' {WebModule1: TWebModule}; {$R *.res} begin Application.Initialize; Application.WebModuleClass := WebModuleClass; Application.Run; end.
Delphi-Quellcode:
Funktioniert bei mir auch mit 300MB Files ohne Probleme.
unit WebModuleUnit1;
interface uses System.SysUtils, System.Classes, HTTPApp; type TWebModule1 = class(TWebModule) procedure WebModuleBeforeDispatch(Sender: TObject; Request: TWebRequest; Response: TWebResponse; var Handled: Boolean); private { Private-Deklarationen } public { Public-Deklarationen } end; var WebModuleClass: TComponentClass = TWebModule1; implementation {$R *.dfm} procedure TWebModule1.WebModuleBeforeDispatch(Sender: TObject; Request: TWebRequest; Response: TWebResponse; var Handled: Boolean); var fn : string; begin fn := paramstr(1) + 'test.bin'; Response.Content := ''; Response.ContentType := 'application/octet-stream'; Response.ContentStream := TBytesStream.Create; TBytesStream(Response.ContentStream).LoadFromFile(fn); Handled := true; end; end. Ist nicht 100%ig das, was du machst - aber Webbroker funktioniert also. Falls du immer noch glaubst, dass es am Speicher liegt, dann prüf doch mal, ob der Tipp mit Response.Location :=... funktioniert. Dann regelt IIS für dich die Übertragung der Files. Daran denken: In IIS die MIME Typen setzen. Der von dir erwähnte WSock Error 10054 sollte doch als Windows Ereignis geloggt werden? Findest du dort keine Infos? Und halt doch noch einmal wegen Speicher... Wenn ich mit perfmon beobachte, was passiert, wenn ich deinen LoadFromFile Ansatz verwende, dann sind nach dem LoadFromFile die 300MB Dateidaten zusätzlich im Arbeitsspeicher (erwartet). Falls ich perfmon vertrauen kann, dann ist es mit TfileStream (erwartet) nicht so. Ich hab's nun auch überprüft: Setze in Web.HttpApp bei procedure TWebResponse.SendStream(AStream: TStream); einen Breakpoint, dann siehst du wie Buffer für Buffer (1MB) gesendet wird und so der Arbeitsspeicher vom Server nicht überlastet wird. Also so:
Delphi-Quellcode:
Ich setze hier zusätzlich auch noch den gewünschten Filenamen (muss natürlich nicht dem Filenamen auf dem Server entsprechen).
procedure TWebModule1.WebModuleBeforeDispatch(Sender: TObject;
Request: TWebRequest; Response: TWebResponse; var Handled: Boolean); var fn : string; tf : tfileStream; begin fn := 'C:\inetpub\wwwroot\test.bin'; Response.Content := ''; Response.ContentType := 'application/octet-stream'; Response.CustomHeaders.Values['Content-Disposition'] := 'attachment; filename=test.bin'; tf := tfilestream.Create( fn, fmopenread or fmShareDenyWrite ); Response.ContentStream := tf; handled := true; end; |
AW: Große Datei über WebBroker als 'application/octet-stream' zur Verfügung stellen
Zitat:
danke für deine Hilfe! Ich habe jetzt mal deinen Code getestet, gleiches Verhalten. Wie bist du in das SendStream rein gekommen? Ich habe hier lediglich ein
Delphi-Quellcode:
bzw ein
procedure SendResponse; virtual; abstract;
Delphi-Quellcode:
. Leider nur die deklaration und keine Implementierung in der Web.HTTPApp.
procedure SendStream(AStream: TStream); virtual; abstract;
Ich habe mir auch in dem WEbmodul einen Pfad definiert, mache es also nicht im BeforeDispatch, aber das sollte ja eigentlich nicht stören oder? vG PJM |
AW: Große Datei über WebBroker als 'application/octet-stream' zur Verfügung stellen
Hi nochmal,
ich verwendet jetzt die Filestream Lösung, weil wie von Michael schon beschrieben, besseres Speichermanagement. Das Problem lag am Postman. Ich teste mit diesem Programm die Abfragen und wie die Meldung vom Websocket schon korrekterweise beschreibt:
Delphi-Quellcode:
wurde die Verbindung vom Peer, also der anfragenden Seite, zurückgesetzt. :|
Socket Fehler 10054 Die Verbindung wurde von Peer zurückgesetzt
War ja aber nicht ganz Umsonst, da das Speichermanagement jetzt besser läuft :-) Vielen Dank! |
Alle Zeitangaben in WEZ +1. Es ist jetzt 15:00 Uhr. |
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024-2025 by Thomas Breitkreuz