Indy: Unterschied zwischen den Versionen
| Zeile 59: | Zeile 59: | ||
begin | begin | ||
LDestFileName := ADestFile; | LDestFileName := ADestFile; | ||
if LDestFileName = | if LDestFileName = `` then | ||
begin | begin | ||
LDestFileName := Sys.ExtractFileName(ASourceFile); | LDestFileName := Sys.ExtractFileName(ASourceFile); | ||
| Zeile 65: | Zeile 65: | ||
LSourceStream := TReadFileNonExclusiveStream.Create(ASourceFile); | LSourceStream := TReadFileNonExclusiveStream.Create(ASourceFile); | ||
try | try | ||
EIdFTPUploadFileNameCanNotBeEmpty.IfTrue(ADestFile = | EIdFTPUploadFileNameCanNotBeEmpty.IfTrue(ADestFile = ``, RSFTPFileNameCanNotBeEmpty); | ||
LSourceStream.Position := StartPosition; | LSourceStream.Position := StartPosition; | ||
DoBeforePut(LSourceStream); | DoBeforePut(LSourceStream); | ||
Version vom 16. November 2009, 12:24 Uhr
OrgaMon verwendet den HTTP-Server und FTP-Client des Indy Projektes. Dieses Projekt enthält in Delphi geschriebene TCP Protokolle. Im Moment (Stand 2009) benötigen einige Units-Patches aus unterschiedlichen Gründen ...
Wirksamkeit der folgenden Patches
Im Tools->Optionen->Bibliothekspfad müssen die 3 Pfade ...
$(BDS)\source\Indy\Indy10\System
$(BDS)\source\Indy\Indy10\Core
$(BDS)\source\Indy\Indy10\Protocols
... hinzugefügt werden, sonst ist der Patch wirkungslos, da die Quelltexte sonst nicht gesehen werden.
Zeitzonen-Bug
- \System\IdSysVCL.pas
class function TIdSysVCL.OffsetFromUTC: TIdDateTimeBase; var TimeZ : TTimeZoneInformation; OffsetInMinutes: integer; begin // get Offset in Minutes from System if (GetTimeZoneInformation(TimeZ)=2) then OffsetInMinutes := - (TimeZ.Bias + TimeZ.DayLightBias) else OffsetInMinutes := - (TimeZ.Bias + TimeZ.StandardBias); // Build the TDateTime, ensure positive Values Result := EncodeTime(abs(OffsetInMinutes) DIV 60,abs(OffsetInMinutes) MOD 60,0,0); // Ensure Old Sign if (OffsetInMinutes<0) then Result := - Result; end;
FTP-Client kann nun REST(art)
bei grössen Datei-Uploads ist es immer möglich dass der Upload nicht vollständig erfolgreich durchgeführt werden kann. Diese Routine ist wiederansetzfähig und kann in einem 2. 3. Versuch die Datei vervollständigen. Erst wenn sie vollständig hochgeladen wurde erhält sie ihren richtigen Namen ...
- \Protocols\IdFTP.pas
// Im "public" Bereich der TIDFTP-Klasse // am besten unter die anderen Put - Routinen procedure PutRestart(const ASourceFile,ADestFile: string; const StartPosition: Int64);
// hier die Implementierung weiter unten
//
// für Delphi 2007
procedure TIdFTP.PutRestart(const ASourceFile, ADestFile: string; const StartPosition: int64);
var
LSourceStream: TIdStream;
LDestFileName : String;
begin
LDestFileName := ADestFile;
if LDestFileName = `` then
begin
LDestFileName := Sys.ExtractFileName(ASourceFile);
end;
LSourceStream := TReadFileNonExclusiveStream.Create(ASourceFile);
try
EIdFTPUploadFileNameCanNotBeEmpty.IfTrue(ADestFile = ``, RSFTPFileNameCanNotBeEmpty);
LSourceStream.Position := StartPosition;
DoBeforePut(LSourceStream);
SendCmd('REST ' + Sys.IntToStr(StartPosition), [350]); {Do not localize}
InternalPut('STOR ' + ADestFile, LSourceStream, false); {Do not localize}
DoAfterPut;
finally
Sys.FreeAndNil(LSourceStream);
end;
end;
// für Delphi 2010
procedure TIdFTP.PutRestart(const ASourceFile, ADestFile: string; const StartPosition: int64);
var
LSourceStream: TStream;
LDestFileName : String;
begin
LDestFileName := ADestFile;
if LDestFileName = `` then
begin
LDestFileName := ExtractFileName(ASourceFile);
end;
LSourceStream := TIdReadFileNonExclusiveStream.Create(ASourceFile);
try
if ADestFile = `` then
raise EIdFTPUploadFileNameCanNotBeEmpty.create(RSFTPFileNameCanNotBeEmpty);
LSourceStream.Position := StartPosition;
DoBeforePut(LSourceStream);
SendCmd('REST ' + IntToStr(StartPosition), [350]); {Do not localize}
InternalPut('STOR ' + ADestFile, LSourceStream, false); {Do not localize}
DoAfterPut;
finally
FreeAndNil(LSourceStream);
end;
end;
diese eine neue Routine reicht aus! Nun folgt ein BEispiel aus der Praxis. Man beachte, dass hier die globale Variable "FTP_StartOffset" gesetzt wird, diese int64 Variable muss bei den Wartebalken-Routinen immer mit dazugezählt werden (Also bei OnBeginWork und bei OnWork).
// eine typische Anwendung eines FTP Restart sieht so aus ...
function TFormDatensicherung.doUpload(ResultFName: string):boolean;
var
FtpDestFName: string;
rSize: int64;
lSize: int64;
begin
//
result := false;
lSize := FSize(ResultFName);
SolidInit(IdFTP1);
with IdFTP1 do
begin
Host := cFTP_Host;
UserName := cFTP_UserName;
password := cFTP_Password;
try
FtpStartTime := 0;
if connected then
Quit;
connect;
// atomic.begin
FtpDestFName := ExtractFileName(ResultFName);
repeat
rSize := Size(FtpDestFName + '.$$$');
if rSize=lSize then
break;
if rSize>lSize then
raise Exception.create('FTP: remote Datei ist '+inttostr(rSize-lSize)+' Bytes grösser als die lokale');
if (rSize<1) then
begin
FTP_StartOffset := 0;
Put(ResultFName, FtpDestFName + '.$$$');
end else
begin
FTP_StartOffset := rSize;
PutRestart(ResultFName, FtpDestFName + '.$$$',rSize);
end;
until true;
rSize := Size(FtpDestFName + '.$$$');
if (lSize=rSize) then
begin
if (Size(FtpDestFName) >= 0) then
Delete(FtpDestFName);
Rename(FtpDestFName + '.$$$', FtpDestFName);
result := true;
end else
begin
if (rSize>lSize) then
raise Exception.create('FTP: remote Datei ist '+inttostr(rSize-lSize)+' Bytes grösser als die lokale')
else
raise Exception.create('FTP: remote Datei ist '+inttostr(lSize-rSize)+' Bytes kleiner als die lokale');
end;
// atomic.end
Quit;
except
on E: Exception do
begin
Log('ERROR: Ftp Upload Error: ' + e.message);
end;
end;
end;
end;