![]() |
SCP-Komponente gesucht
Hallo,
in einem alten Programm wird eine Datei mithilfe des Putty-Programmes pscp.exe übertragen. Mit folgenden (annonymisierten) Parametern wird pscp per Shell aufgerufen: pscp.exe -P 1234 -i .\schlüsseldatei.ppk .\Dateiname.txt username@172.123.111.111:uploadverzeichnis/ Ich soll das in der Neuauflage des Programmes im Programm selber einbauen (ohne das externe Programm pscp zu nutzen). Ich habe die Indy-Komponentensammlung zur Verfügung, aber ich weiß gar nicht, wo ich da Anfangen soll, sprich wie bau ich SCP mit Indy-Komponenten? Ich hab es mit TIdSSLIOHandlerSocketOpenSSL und TIdFTP versucht, aber das ist glaub ich falsch und führt zu FTP mit SSH, aber das ist glaub ich was anderes. Hoffe ihr könnt mir da helfen, da ich mit diesen ganzen Netzwerkprotokollen/-Komponenten selten zu tun hab und daher kaum Ahnung habe. |
AW: SCP-Komponente gesucht
Hi,
vielleicht kannst du wegen Komponenten mal hier schauen: ![]() ![]() Ist "leider" ActiveX, und auch gar nicht billig. Testversion kann wohl runtergeladen werden. Gruß Steku |
AW: SCP-Komponente gesucht
WinSCP ist doch Open Source und ich meine, dass es mit dem C++ Builder geschrieben ist. Da würde ich vielleicht mal reinschauen.
|
AW: SCP-Komponente gesucht
Ich find solche Vorgaben immer wieder spaßig: "Machen Sie es schön bunt, aber bitte ohne Farben, die mag ich gar nicht."
Du könntest SCP.EXE auch einfach als Resource in der EXE halten und vor dem Ausführen ins Temp-Verzeichnis schreiben, ausführen und wieder löschen. Merkt keine Sau. und is ne sache von 20 Minuten |
AW: SCP-Komponente gesucht
Zitat:
Vielmehr kann man die Exe gleich unmittelbar 'on the fly' aus der Resource in das Memory laden und ausführen. 1) Exe-Ressource in Programm einbinden (z.B. exeRes.rc >> exeRes.res, wie bekannt)
Delphi-Quellcode:
2) benötigt wird Unit ExeRes2Memory,
ExeTest RCDATA "NOTEPAD.EXE"
veröffentlicht als Unit PrjRes2Mem unter ![]() hier leicht abgeändert (und deshalb umbenannt) für den benötigten Zweck
Delphi-Quellcode:
3)
unit ExeRes2Memory;
{$R 'exeRes.res'} interface uses windows; procedure ResourceToMem(ExeResName: string); var ExeResProcInfo: TProcessInformation; ExeResName: string; implementation type TSections = array[0..0] of TImageSectionHeader; function GetAlignedSize(Size: dword; Alignment: dword): dword; begin if ((Size mod Alignment) = 0) then Result := Size else Result := ((Size div Alignment) + 1) * Alignment; end; function ImageSize(Image: pointer): dword; var Alignment: dword; ImageNtHeaders: PImageNtHeaders; PSections: ^TSections; SectionLoop: dword; begin ImageNtHeaders := pointer(dword(dword(Image)) + dword(PImageDosHeader(Image)._lfanew)); Alignment := ImageNtHeaders.OptionalHeader.SectionAlignment; if ((ImageNtHeaders.OptionalHeader.SizeOfHeaders mod Alignment) = 0) then Result := ImageNtHeaders.OptionalHeader.SizeOfHeaders else Result := ((ImageNtHeaders.OptionalHeader.SizeOfHeaders div Alignment) + 1) * Alignment; PSections := pointer(pchar(@(ImageNtHeaders.OptionalHeader)) + ImageNtHeaders.FileHeader.SizeOfOptionalHeader); for SectionLoop := 0 to ImageNtHeaders.FileHeader.NumberOfSections - 1 do begin if PSections[SectionLoop].Misc.VirtualSize <> 0 then begin if ((PSections[SectionLoop].Misc.VirtualSize mod Alignment) = 0) then Result := Result + PSections[SectionLoop].Misc.VirtualSize else Result := Result + (((PSections[SectionLoop].Misc.VirtualSize div Alignment) + 1) * Alignment); end; end; end; procedure CreateProcessEx(FileMemory: pointer); var BaseAddress, Bytes, HeaderSize, InjectSize, SectionLoop, SectionSize: dword; Context: TContext; FileData: pointer; ImageNtHeaders: PImageNtHeaders; InjectMemory: pointer; PSections: ^TSections; StartInfo: TStartupInfo; begin ImageNtHeaders := pointer(dword(dword(FileMemory)) + dword(PImageDosHeader(FileMemory)._lfanew)); InjectSize := ImageSize(FileMemory); GetMem(InjectMemory, InjectSize); try FileData := InjectMemory; HeaderSize := ImageNtHeaders.OptionalHeader.SizeOfHeaders; PSections := pointer(pchar(@(ImageNtHeaders.OptionalHeader)) + ImageNtHeaders.FileHeader.SizeOfOptionalHeader); for SectionLoop := 0 to ImageNtHeaders.FileHeader.NumberOfSections - 1 do begin if PSections[SectionLoop].PointerToRawData < HeaderSize then HeaderSize := PSections[SectionLoop].PointerToRawData; end; CopyMemory(FileData, FileMemory, HeaderSize); FileData := pointer(dword(FileData) + GetAlignedSize(ImageNtHeaders.OptionalHeader.SizeOfHeaders, ImageNtHeaders.OptionalHeader.SectionAlignment)); for SectionLoop := 0 to ImageNtHeaders.FileHeader.NumberOfSections - 1 do begin if PSections[SectionLoop].SizeOfRawData > 0 then begin SectionSize := PSections[SectionLoop].SizeOfRawData; if SectionSize > PSections[SectionLoop].Misc.VirtualSize then SectionSize := PSections[SectionLoop].Misc.VirtualSize; CopyMemory(FileData, pointer(dword(FileMemory) + PSections[SectionLoop].PointerToRawData), SectionSize); FileData := pointer(dword(FileData) + GetAlignedSize(PSections[SectionLoop].Misc.VirtualSize, ImageNtHeaders.OptionalHeader.SectionAlignment)); end else begin if PSections[SectionLoop].Misc.VirtualSize <> 0 then FileData := pointer(dword(FileData) + GetAlignedSize(PSections[SectionLoop].Misc.VirtualSize, ImageNtHeaders.OptionalHeader.SectionAlignment)); end; end; ZeroMemory(@StartInfo, SizeOf(StartupInfo)); ZeroMemory(@Context, SizeOf(TContext)); CreateProcess(nil, pchar(ParamStr(0)), nil, nil, False, CREATE_SUSPENDED, nil, nil, StartInfo, ExeResProcInfo); Context.ContextFlags := CONTEXT_FULL; with ExeResProcInfo do begin GetThreadContext(hThread, Context); ReadProcessMemory(hProcess, pointer(Context.Ebx + 8), @BaseAddress, 4, Bytes); VirtualAllocEx(hProcess, pointer(ImageNtHeaders.OptionalHeader.ImageBase), InjectSize, MEM_RESERVE or MEM_COMMIT, PAGE_EXECUTE_READWRITE); WriteProcessMemory(hProcess, pointer(ImageNtHeaders.OptionalHeader.ImageBase), InjectMemory, InjectSize, Bytes); WriteProcessMemory(hProcess, pointer(Context.Ebx + 8), @ImageNtHeaders.OptionalHeader.ImageBase, 4, Bytes); Context.Eax := ImageNtHeaders.OptionalHeader.ImageBase + ImageNtHeaders.OptionalHeader.AddressOfEntryPoint; SetThreadContext(hThread, Context); ResumeThread(hThread); end; finally FreeMemory(InjectMemory); end; end; procedure ResourceToMem(ExeResName: string); var ResInfo: HRSRC; ResSize: LongWord; ResData: Pointer; Handle: THandle; begin if (ExeResName='') then exit; ResInfo := FindResource(SysInit.HInstance, pchar(ExeResName), RT_RCDATA); if ResInfo <> 0 then begin ResSize := SizeofResource(SysInit.HInstance, ResInfo); if ResSize <> 0 then begin Handle := LoadResource(SysInit.HInstance, ResInfo); if Handle <> 0 then begin ResData := LockResource(Handle); createprocessex(ResData); end; end; end; end; end. Beispielprogramm:
Delphi-Quellcode:
P.S.
unit Unit1;
interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, StdCtrls; type TForm1 = class(TForm) Button1: TButton; Button2: TButton; procedure Button1Click(Sender: TObject); procedure Button2Click(Sender: TObject); private { Private-Deklarationen } public { Public-Deklarationen } end; var Form1: TForm1; implementation {$R *.dfm} uses ExeRes2Memory; const cExeResName = 'notepadtest'; Function KillProcess(ProcInfo: TProcessInformation):boolean; var process_handle:integer; begin process_handle:=openprocess(PROCESS_ALL_ACCESS,true,ProcInfo.dwProcessId); if terminateprocess(process_handle,0) then begin CloseHandle(ProcInfo.hProcess); CloseHandle(ProcInfo.hThread); result:=true; end else result:=false; end; procedure TForm1.FormCreate(Sender: TObject); begin Button1.caption:='Start'; Button2.caption:='Kill'; end; procedure TForm1.Button1Click(Sender: TObject); begin ResourceToMem (cExeResName); end; procedure TForm1.Button2Click(Sender: TObject); begin if not KillProcess(ExeResProcInfo) then Showmessage('Failed to terminate running ExeResource !'); end; end. Man könnte/sollte das natürlich noch erweitern, z.B. um die Abfrage, ob eine Resource mit dem angegebenen Namen im (eigenen) Programm tatsächlich vorhanden ist. |
AW: SCP-Komponente gesucht
Zitat:
Oder gibt es einen anderen Fallstrick? |
AW: SCP-Komponente gesucht
Zitat:
|
AW: SCP-Komponente gesucht
Zitat:
Verwendet habe ich den Code übrigens unter WinXP; ob auch alles unter Win7 ebenso glatt läuft, kann ich gerade nicht überprüfen. Und hier noch eine leicht verbesserte Funktion Killprocess(), die den Fall berücksichtigt, dass die vom Programm gestartete ExeResource bereits extern beendet worden sein sollte. Das kann durch Abfrage des Exitcodes per GetExitCodeProcess() geklärt werden:
Delphi-Quellcode:
function KillProcess(ProcInfo: TProcessInformation): boolean;
var process_handle: integer; errorCode: cardinal; begin process_handle := openprocess(PROCESS_ALL_ACCESS, true, ProcInfo.dwProcessId); result := terminateprocess(process_handle, 0); if not result then begin // prüfen, was los ist GetExitCodeProcess(process_handle, Errorcode); // falls der Process bereits extern beendet worden ist, // dann ist ErrorCode = 0, ansonsten ErrorCode <> 0 result := ErrorCode = 0; end; if result then begin CloseHandle(ProcInfo.hProcess); CloseHandle(ProcInfo.hThread); end; end; |
AW: SCP-Komponente gesucht
Hallo zusammen und erstmal Danke für die guten Tipps.
Hauptmotivation für eine neue Umsetzung an dieser Stelle war, das pscp.exe auch einen Rückgabewert liefert, aus dem ersichtlich wird, ob der Dateitransfer erfolgreich verlaufen ist. Dieser wurde bisher aber nicht ausgewertet. Darum sollte das im Delphi-Programm selber umgesetzt werden. Da das aber möglichst schon vorgestern fertig sein (Kunde) und nix kosten (Chefe) soll, konnte ich das mangels passender Komponenten nicht umsetzten. Und SCP selber nachzubauen (z.B. durch Studium von WinSCP, siehe Vorschlag von wicht) würde zu lange dauern, da ich mich dazu viel zu sehr einarbeiten müsste. Iwo Asnet hat's daher richtig zusammengefasst, pscp.exe wird trotzdem genutzt, aber so, dass der Rückgabewert ausgewertet werden kann, denn darum ging es ja primär. Hilfreich war dazu die hier vorgestellte Methode: ![]() Damit hab ich es erstmal ans laufen gebracht, allerdings so, dass pscp.exe nun mit dem eigentlichen Programm im selben Verzeichnis stehen muss. Das werde ich dem Kunden denke ich verkaufen können, da es ja primär um die Auswertung der Antwort von pscp.exe ging und das ist jetzt gewährleistet. Nevertheless: Die pscp.exe als Resource einzubinden und dann auch noch mit dem Vorschlag von ASM gekoppelt ist natürlich sehr, sehr cool und wurde von mir direkt im Geiste abgespeichert, wer weiß wann man das mal brauchen kann. Hier für meinen konkreten Fall muss ich mal gucken, ob ich das mit der "Capturing Console Output"-Geschichte koppeln kann. |
Alle Zeitangaben in WEZ +1. Es ist jetzt 14:38 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