| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356 | //---------------------------------------------------------------------------#include <vcl.h>#pragma hdrstop#include "SessionData.h"#include "Common.h"#include "Exceptions.h"#include "FileBuffer.h"#include "CoreMain.h"#include "TextsCore.h"#include "PuttyIntf.h"#include "RemoteFiles.h"#include "SFTPFileSystem.h"#include <Soap.EncdDecd.hpp>#include <StrUtils.hpp>#include <XMLDoc.hpp>#include <StrUtils.hpp>#include <algorithm>//---------------------------------------------------------------------------#pragma package(smart_init)//---------------------------------------------------------------------------const wchar_t * PingTypeNames = L"Off;Null;Dummy";const wchar_t * ProxyMethodNames = L"None;SOCKS4;SOCKS5;HTTP;Telnet;Cmd";const wchar_t * DefaultName = L"Default Settings";const UnicodeString CipherNames[CIPHER_COUNT] = {L"WARN", L"3des", L"blowfish", L"aes", L"des", L"arcfour", L"chacha20"};const UnicodeString KexNames[KEX_COUNT] = {L"WARN", L"dh-group1-sha1", L"dh-group14-sha1", L"dh-gex-sha1", L"rsa", L"ecdh"};const wchar_t SshProtList[][10] = {L"1", L"1>2", L"2>1", L"2"};const TCipher DefaultCipherList[CIPHER_COUNT] =  { cipAES, cipChaCha20, cipBlowfish, cip3DES, cipWarn, cipArcfour, cipDES };const TKex DefaultKexList[KEX_COUNT] =  { kexECDH, kexDHGEx, kexDHGroup14, kexRSA, kexWarn, kexDHGroup1 };const wchar_t FSProtocolNames[FSPROTOCOL_COUNT][16] = { L"SCP", L"SFTP (SCP)", L"SFTP", L"", L"", L"FTP", L"WebDAV" };const int SshPortNumber = 22;const int FtpPortNumber = 21;const int FtpsImplicitPortNumber = 990;const int HTTPPortNumber = 80;const int HTTPSPortNumber = 443;const int TelnetPortNumber = 23;const int DefaultSendBuf = 262144;const int ProxyPortNumber = 80;const UnicodeString AnonymousUserName(L"anonymous");const UnicodeString AnonymousPassword(L"[email protected]");const UnicodeString PuttySshProtocol(L"ssh");const UnicodeString PuttyTelnetProtocol(L"telnet");const UnicodeString SftpProtocol(L"sftp");const UnicodeString ScpProtocol(L"scp");const UnicodeString FtpProtocol(L"ftp");const UnicodeString FtpsProtocol(L"ftps");const UnicodeString FtpesProtocol(L"ftpes");const UnicodeString SshProtocol(L"ssh");const UnicodeString WinSCPProtocolPrefix(L"winscp-");const wchar_t UrlParamSeparator = L';';const wchar_t UrlParamValueSeparator = L'=';const UnicodeString UrlHostKeyParamName(L"fingerprint");const UnicodeString UrlSaveParamName(L"save");const UnicodeString PassphraseOption(L"passphrase");//---------------------------------------------------------------------TDateTime __fastcall SecToDateTime(int Sec){  return TDateTime(double(Sec) / SecsPerDay);}//--- TSessionData ----------------------------------------------------__fastcall TSessionData::TSessionData(UnicodeString aName):  TNamedObject(aName){  Default();  FModified = true;}//---------------------------------------------------------------------_fastcall TSessionData::~TSessionData(){}//---------------------------------------------------------------------int __fastcall TSessionData::Compare(TNamedObject * Other){  int Result;  // To avoid using CompareLogicalText on hex names of sessions in workspace.  // The session 000A would be sorted before 0001.  if (IsWorkspace && DebugNotNull(dynamic_cast<TSessionData *>(Other))->IsWorkspace)  {    Result = CompareText(Name, Other->Name);  }  else  {    Result = TNamedObject::Compare(Other);  }  return Result;}//---------------------------------------------------------------------TSessionData * __fastcall TSessionData::Clone(){  std::unique_ptr<TSessionData> Data(new TSessionData(L""));  Data->Assign(this);  return Data.release();}//---------------------------------------------------------------------void __fastcall TSessionData::Default(){  HostName = L"";  PortNumber = SshPortNumber;  UserName = L"";  Password = L"";  PingInterval = 30;  PingType = ptOff;  Timeout = 15;  TryAgent = true;  AgentFwd = false;  AuthTIS = false;  AuthKI = true;  AuthKIPassword = true;  AuthGSSAPI = false;  GSSAPIFwdTGT = false;  ChangeUsername = false;  Compression = false;  SshProt = ssh2only;  Ssh2DES = false;  SshNoUserAuth = false;  for (int Index = 0; Index < CIPHER_COUNT; Index++)  {    Cipher[Index] = DefaultCipherList[Index];  }  for (int Index = 0; Index < KEX_COUNT; Index++)  {    Kex[Index] = DefaultKexList[Index];  }  PublicKeyFile = L"";  Passphrase = L"";  FPuttyProtocol = L"";  TcpNoDelay = false;  SendBuf = DefaultSendBuf;  SshSimple = true;  HostKey = L"";  FingerprintScan = false;  FOverrideCachedHostKey = true;  Note = L"";  ProxyMethod = ::pmNone;  ProxyHost = L"proxy";  ProxyPort = ProxyPortNumber;  ProxyUsername = L"";  ProxyPassword = L"";  ProxyTelnetCommand = L"connect %host %port\\n";  ProxyLocalCommand = L"";  ProxyDNS = asAuto;  ProxyLocalhost = false;  for (unsigned int Index = 0; Index < LENOF(FBugs); Index++)  {    Bug[(TSshBug)Index] = asAuto;  }  Special = false;  FSProtocol = fsSFTP;  AddressFamily = afAuto;  RekeyData = L"1G";  RekeyTime = MinsPerHour;  // FS common  LocalDirectory = L"";  RemoteDirectory = L"";  SynchronizeBrowsing = false;  UpdateDirectories = true;  CacheDirectories = true;  CacheDirectoryChanges = true;  PreserveDirectoryChanges = true;  LockInHome = false;  ResolveSymlinks = true;  FollowDirectorySymlinks = false;  DSTMode = dstmUnix;  DeleteToRecycleBin = false;  OverwrittenToRecycleBin = false;  RecycleBinPath = L"";  Color = 0;  PostLoginCommands = L"";  // SCP  ReturnVar = L"";  LookupUserGroups = asAuto;  EOLType = eolLF;  TrimVMSVersions = false;  Shell = L""; //default shell  ReturnVar = L"";  ClearAliases = true;  UnsetNationalVars = true;  ListingCommand = L"ls -la";  IgnoreLsWarnings = true;  Scp1Compatibility = false;  TimeDifference = 0;  TimeDifferenceAuto = true;  SCPLsFullTime = asAuto;  NotUtf = asAuto;  // SFTP  SftpServer = L"";  SFTPDownloadQueue = 32;  SFTPUploadQueue = 32;  SFTPListingQueue = 2;  SFTPMaxVersion = ::SFTPMaxVersion;  SFTPMaxPacketSize = 0;  for (unsigned int Index = 0; Index < LENOF(FSFTPBugs); Index++)  {    SFTPBug[(TSftpBug)Index] = asAuto;  }  Tunnel = false;  TunnelHostName = L"";  TunnelPortNumber = SshPortNumber;  TunnelUserName = L"";  TunnelPassword = L"";  TunnelPublicKeyFile = L"";  TunnelLocalPortNumber = 0;  TunnelPortFwd = L"";  TunnelHostKey = L"";  // FTP  FtpPasvMode = true;  FtpForcePasvIp = asAuto;  FtpUseMlsd = asAuto;  FtpAccount = L"";  FtpPingInterval = 30;  FtpPingType = ptDummyCommand;  FtpTransferActiveImmediately = asAuto;  Ftps = ftpsNone;  MinTlsVersion = tls10;  MaxTlsVersion = tls12;  FtpListAll = asAuto;  FtpHost = asAuto;  SslSessionReuse = true;  TlsCertificateFile = L"";  FtpProxyLogonType = 0; // none  CustomParam1 = L"";  CustomParam2 = L"";  IsWorkspace = false;  Link = L"";  Selected = false;  FModified = false;  FSource = ::ssNone;  FSaveOnly = false;  // add also to TSessionLog::AddStartupInfo()}//---------------------------------------------------------------------void __fastcall TSessionData::NonPersistant(){  UpdateDirectories = false;  PreserveDirectoryChanges = false;}//---------------------------------------------------------------------#define BASE_PROPERTIES \  PROPERTY(HostName); \  PROPERTY(PortNumber); \  PROPERTY(UserName); \  PROPERTY(Password); \  PROPERTY(PublicKeyFile); \  PROPERTY(Passphrase); \  PROPERTY(FSProtocol); \  PROPERTY(Ftps); \  PROPERTY(LocalDirectory); \  PROPERTY(RemoteDirectory); \  PROPERTY(Color); \  PROPERTY(SynchronizeBrowsing); \  PROPERTY(Note);//---------------------------------------------------------------------#define ADVANCED_PROPERTIES \  PROPERTY(PingInterval); \  PROPERTY(PingType); \  PROPERTY(Timeout); \  PROPERTY(TryAgent); \  PROPERTY(AgentFwd); \  PROPERTY(AuthTIS); \  PROPERTY(ChangeUsername); \  PROPERTY(Compression); \  PROPERTY(SshProt); \  PROPERTY(Ssh2DES); \  PROPERTY(SshNoUserAuth); \  PROPERTY(CipherList); \  PROPERTY(KexList); \  PROPERTY(AddressFamily); \  PROPERTY(RekeyData); \  PROPERTY(RekeyTime); \  PROPERTY(HostKey); \  PROPERTY(FingerprintScan); \  \  PROPERTY(UpdateDirectories); \  PROPERTY(CacheDirectories); \  PROPERTY(CacheDirectoryChanges); \  PROPERTY(PreserveDirectoryChanges); \  \  PROPERTY(ResolveSymlinks); \  PROPERTY(FollowDirectorySymlinks); \  PROPERTY(DSTMode); \  PROPERTY(LockInHome); \  PROPERTY(Special); \  PROPERTY(Selected); \  PROPERTY(ReturnVar); \  PROPERTY(LookupUserGroups); \  PROPERTY(EOLType); \  PROPERTY(TrimVMSVersions); \  PROPERTY(Shell); \  PROPERTY(ClearAliases); \  PROPERTY(Scp1Compatibility); \  PROPERTY(UnsetNationalVars); \  PROPERTY(ListingCommand); \  PROPERTY(IgnoreLsWarnings); \  PROPERTY(SCPLsFullTime); \  \  PROPERTY(TimeDifference); \  PROPERTY(TimeDifferenceAuto); \  PROPERTY(TcpNoDelay); \  PROPERTY(SendBuf); \  PROPERTY(SshSimple); \  PROPERTY(AuthKI); \  PROPERTY(AuthKIPassword); \  PROPERTY(AuthGSSAPI); \  PROPERTY(GSSAPIFwdTGT); \  PROPERTY(DeleteToRecycleBin); \  PROPERTY(OverwrittenToRecycleBin); \  PROPERTY(RecycleBinPath); \  PROPERTY(NotUtf); \  PROPERTY(PostLoginCommands); \  \  PROPERTY(ProxyMethod); \  PROPERTY(ProxyHost); \  PROPERTY(ProxyPort); \  PROPERTY(ProxyUsername); \  PROPERTY(ProxyPassword); \  PROPERTY(ProxyTelnetCommand); \  PROPERTY(ProxyLocalCommand); \  PROPERTY(ProxyDNS); \  PROPERTY(ProxyLocalhost); \  \  for (unsigned int Index = 0; Index < LENOF(FBugs); Index++) \  { \    PROPERTY(Bug[(TSshBug)Index]); \  } \  \  PROPERTY(SftpServer); \  PROPERTY(SFTPDownloadQueue); \  PROPERTY(SFTPUploadQueue); \  PROPERTY(SFTPListingQueue); \  PROPERTY(SFTPMaxVersion); \  PROPERTY(SFTPMaxPacketSize); \  \  for (unsigned int Index = 0; Index < LENOF(FSFTPBugs); Index++) \  { \    PROPERTY(SFTPBug[(TSftpBug)Index]); \  } \  \  PROPERTY(Tunnel); \  PROPERTY(TunnelHostName); \  PROPERTY(TunnelPortNumber); \  PROPERTY(TunnelUserName); \  PROPERTY(TunnelPassword); \  PROPERTY(TunnelPublicKeyFile); \  PROPERTY(TunnelLocalPortNumber); \  PROPERTY(TunnelPortFwd); \  PROPERTY(TunnelHostKey); \  \  PROPERTY(FtpPasvMode); \  PROPERTY(FtpForcePasvIp); \  PROPERTY(FtpUseMlsd); \  PROPERTY(FtpAccount); \  PROPERTY(FtpPingInterval); \  PROPERTY(FtpPingType); \  PROPERTY(FtpTransferActiveImmediately); \  PROPERTY(FtpListAll); \  PROPERTY(FtpHost); \  PROPERTY(SslSessionReuse); \  PROPERTY(TlsCertificateFile); \  \  PROPERTY(FtpProxyLogonType); \  \  PROPERTY(MinTlsVersion); \  PROPERTY(MaxTlsVersion); \  \  PROPERTY(CustomParam1); \  PROPERTY(CustomParam2);#define META_PROPERTIES \  PROPERTY(IsWorkspace); \  PROPERTY(Link);//---------------------------------------------------------------------void __fastcall TSessionData::Assign(TPersistent * Source){  if (Source && Source->InheritsFrom(__classid(TSessionData)))  {    TSessionData * SourceData = (TSessionData *)Source;    CopyData(SourceData);    FSource = SourceData->FSource;  }  else  {    TNamedObject::Assign(Source);  }}//---------------------------------------------------------------------void __fastcall TSessionData::CopyData(TSessionData * SourceData){  #define PROPERTY(P) P = SourceData->P  PROPERTY(Name);  BASE_PROPERTIES;  ADVANCED_PROPERTIES;  META_PROPERTIES;  #undef PROPERTY  FOverrideCachedHostKey = SourceData->FOverrideCachedHostKey;  FModified = SourceData->Modified;  FSaveOnly = SourceData->FSaveOnly;}//---------------------------------------------------------------------void __fastcall TSessionData::CopyDirectoriesStateData(TSessionData * SourceData){  RemoteDirectory = SourceData->RemoteDirectory;  LocalDirectory = SourceData->LocalDirectory;  SynchronizeBrowsing = SourceData->SynchronizeBrowsing;}//---------------------------------------------------------------------bool __fastcall TSessionData::HasStateData(){  return    !RemoteDirectory.IsEmpty() ||    !LocalDirectory.IsEmpty() ||    (Color != 0);}//---------------------------------------------------------------------void __fastcall TSessionData::CopyStateData(TSessionData * SourceData){  // Keep in sync with TCustomScpExplorerForm::UpdateSessionData.  CopyDirectoriesStateData(SourceData);  Color = SourceData->Color;}//---------------------------------------------------------------------void __fastcall TSessionData::CopyNonCoreData(TSessionData * SourceData){  CopyStateData(SourceData);  UpdateDirectories = SourceData->UpdateDirectories;  Note = SourceData->Note;}//---------------------------------------------------------------------bool __fastcall TSessionData::IsSame(const TSessionData * Default, bool AdvancedOnly, TStrings * DifferentProperties){  bool Result = true;  #define PROPERTY(P) \    if (P != Default->P) \    { \      if (DifferentProperties != NULL) \      { \        DifferentProperties->Add(#P); \      } \      Result = false; \    }  if (!AdvancedOnly)  {    BASE_PROPERTIES;    META_PROPERTIES;  }  ADVANCED_PROPERTIES;  #undef PROPERTY  return Result;}//---------------------------------------------------------------------bool __fastcall TSessionData::IsSame(const TSessionData * Default, bool AdvancedOnly){  return IsSame(Default, AdvancedOnly, NULL);}//---------------------------------------------------------------------bool __fastcall TSessionData::IsSameSite(const TSessionData * Other){  return      (FSProtocol == Other->FSProtocol) &&      (HostName == Other->HostName) &&      (PortNumber == Other->PortNumber) &&      (UserName == Other->UserName);}//---------------------------------------------------------------------bool __fastcall TSessionData::IsInFolderOrWorkspace(UnicodeString AFolder){  return StartsText(UnixIncludeTrailingBackslash(AFolder), Name);}//---------------------------------------------------------------------void __fastcall TSessionData::DoLoad(THierarchicalStorage * Storage, bool & RewritePassword){  // Make sure we only ever use methods supported by TOptionsStorage  // (implemented by TOptionsIniFile)  PortNumber = Storage->ReadInteger(L"PortNumber", PortNumber);  UserName = Storage->ReadString(L"UserName", UserName);  // must be loaded after UserName, because HostName may be in format user@host  HostName = Storage->ReadString(L"HostName", HostName);  if (!Configuration->DisablePasswordStoring)  {    if (Storage->ValueExists(L"PasswordPlain"))    {      Password = Storage->ReadString(L"PasswordPlain", Password);      RewritePassword = true;    }    else    {      FPassword = Storage->ReadStringAsBinaryData(L"Password", FPassword);    }  }  HostKey = Storage->ReadString(L"HostKey", HostKey);  Note = Storage->ReadString(L"Note", Note);  // Putty uses PingIntervalSecs  int PingIntervalSecs = Storage->ReadInteger(L"PingIntervalSecs", -1);  if (PingIntervalSecs < 0)  {    PingIntervalSecs = Storage->ReadInteger(L"PingIntervalSec", PingInterval%SecsPerMin);  }  PingInterval =    Storage->ReadInteger(L"PingInterval", PingInterval/SecsPerMin)*SecsPerMin +    PingIntervalSecs;  if (PingInterval == 0)  {    PingInterval = 30;  }  PingType = static_cast<TPingType>(Storage->ReadInteger(L"PingType", PingType));  Timeout = Storage->ReadInteger(L"Timeout", Timeout);  TryAgent = Storage->ReadBool(L"TryAgent", TryAgent);  AgentFwd = Storage->ReadBool(L"AgentFwd", AgentFwd);  AuthTIS = Storage->ReadBool(L"AuthTIS", AuthTIS);  AuthKI = Storage->ReadBool(L"AuthKI", AuthKI);  AuthKIPassword = Storage->ReadBool(L"AuthKIPassword", AuthKIPassword);  // Continue to use setting keys of previous kerberos implementation (vaclav tomec),  // but fallback to keys of other implementations (official putty and vintela quest putty),  // to allow imports from all putty versions.  // Both vaclav tomec and official putty use AuthGSSAPI  AuthGSSAPI = Storage->ReadBool(L"AuthGSSAPI", Storage->ReadBool(L"AuthSSPI", AuthGSSAPI));  GSSAPIFwdTGT = Storage->ReadBool(L"GSSAPIFwdTGT", Storage->ReadBool(L"GssapiFwd", Storage->ReadBool(L"SSPIFwdTGT", GSSAPIFwdTGT)));  ChangeUsername = Storage->ReadBool(L"ChangeUsername", ChangeUsername);  Compression = Storage->ReadBool(L"Compression", Compression);  TSshProt ASshProt = (TSshProt)Storage->ReadInteger(L"SshProt", SshProt);  // Old sessions may contain the values correponding to the fallbacks we used to allow; migrate them  if (ASshProt == ssh2deprecated)  {    ASshProt = ssh2only;  }  else if (ASshProt == ssh1deprecated)  {    ASshProt = ssh1only;  }  SshProt = ASshProt;  Ssh2DES = Storage->ReadBool(L"Ssh2DES", Ssh2DES);  SshNoUserAuth = Storage->ReadBool(L"SshNoUserAuth", SshNoUserAuth);  CipherList = Storage->ReadString(L"Cipher", CipherList);  KexList = Storage->ReadString(L"KEX", KexList);  PublicKeyFile = Storage->ReadString(L"PublicKeyFile", PublicKeyFile);  AddressFamily = static_cast<TAddressFamily>    (Storage->ReadInteger(L"AddressFamily", AddressFamily));  RekeyData = Storage->ReadString(L"RekeyBytes", RekeyData);  RekeyTime = Storage->ReadInteger(L"RekeyTime", RekeyTime);  FSProtocol = (TFSProtocol)Storage->ReadInteger(L"FSProtocol", FSProtocol);  LocalDirectory = Storage->ReadString(L"LocalDirectory", LocalDirectory);  RemoteDirectory = Storage->ReadString(L"RemoteDirectory", RemoteDirectory);  SynchronizeBrowsing = Storage->ReadBool(L"SynchronizeBrowsing", SynchronizeBrowsing);  UpdateDirectories = Storage->ReadBool(L"UpdateDirectories", UpdateDirectories);  CacheDirectories = Storage->ReadBool(L"CacheDirectories", CacheDirectories);  CacheDirectoryChanges = Storage->ReadBool(L"CacheDirectoryChanges", CacheDirectoryChanges);  PreserveDirectoryChanges = Storage->ReadBool(L"PreserveDirectoryChanges", PreserveDirectoryChanges);  ResolveSymlinks = Storage->ReadBool(L"ResolveSymlinks", ResolveSymlinks);  FollowDirectorySymlinks = Storage->ReadBool(L"FollowDirectorySymlinks", FollowDirectorySymlinks);  DSTMode = (TDSTMode)Storage->ReadInteger(L"ConsiderDST", DSTMode);  LockInHome = Storage->ReadBool(L"LockInHome", LockInHome);  Special = Storage->ReadBool(L"Special", Special);  Shell = Storage->ReadString(L"Shell", Shell);  ClearAliases = Storage->ReadBool(L"ClearAliases", ClearAliases);  UnsetNationalVars = Storage->ReadBool(L"UnsetNationalVars", UnsetNationalVars);  ListingCommand = Storage->ReadString(L"ListingCommand",    Storage->ReadBool(L"AliasGroupList", false) ? UnicodeString(L"ls -gla") : ListingCommand);  IgnoreLsWarnings = Storage->ReadBool(L"IgnoreLsWarnings", IgnoreLsWarnings);  SCPLsFullTime = TAutoSwitch(Storage->ReadInteger(L"SCPLsFullTime", SCPLsFullTime));  Scp1Compatibility = Storage->ReadBool(L"Scp1Compatibility", Scp1Compatibility);  TimeDifference = Storage->ReadFloat(L"TimeDifference", TimeDifference);  TimeDifferenceAuto = Storage->ReadBool(L"TimeDifferenceAuto", (TimeDifference == TDateTime()));  DeleteToRecycleBin = Storage->ReadBool(L"DeleteToRecycleBin", DeleteToRecycleBin);  OverwrittenToRecycleBin = Storage->ReadBool(L"OverwrittenToRecycleBin", OverwrittenToRecycleBin);  RecycleBinPath = Storage->ReadString(L"RecycleBinPath", RecycleBinPath);  PostLoginCommands = Storage->ReadString(L"PostLoginCommands", PostLoginCommands);  ReturnVar = Storage->ReadString(L"ReturnVar", ReturnVar);  LookupUserGroups = TAutoSwitch(Storage->ReadInteger(L"LookupUserGroups2", LookupUserGroups));  EOLType = (TEOLType)Storage->ReadInteger(L"EOLType", EOLType);  TrimVMSVersions = Storage->ReadBool(L"TrimVMSVersions", TrimVMSVersions);  NotUtf = TAutoSwitch(Storage->ReadInteger(L"Utf", Storage->ReadInteger(L"SFTPUtfBug", NotUtf)));  TcpNoDelay = Storage->ReadBool(L"TcpNoDelay", TcpNoDelay);  SendBuf = Storage->ReadInteger(L"SendBuf", Storage->ReadInteger("SshSendBuf", SendBuf));  SshSimple = Storage->ReadBool(L"SshSimple", SshSimple);  ProxyMethod = (TProxyMethod)Storage->ReadInteger(L"ProxyMethod", ProxyMethod);  ProxyHost = Storage->ReadString(L"ProxyHost", ProxyHost);  ProxyPort = Storage->ReadInteger(L"ProxyPort", ProxyPort);  ProxyUsername = Storage->ReadString(L"ProxyUsername", ProxyUsername);  if (Storage->ValueExists(L"ProxyPassword"))  {    // encrypt unencrypted password    ProxyPassword = Storage->ReadString(L"ProxyPassword", L"");  }  else  {    // load encrypted password    FProxyPassword = Storage->ReadStringAsBinaryData(L"ProxyPasswordEnc", FProxyPassword);  }  if (ProxyMethod == pmCmd)  {    ProxyLocalCommand = Storage->ReadStringRaw(L"ProxyTelnetCommand", ProxyLocalCommand);  }  else  {    ProxyTelnetCommand = Storage->ReadStringRaw(L"ProxyTelnetCommand", ProxyTelnetCommand);  }  ProxyDNS = TAutoSwitch((Storage->ReadInteger(L"ProxyDNS", (ProxyDNS + 2) % 3) + 1) % 3);  ProxyLocalhost = Storage->ReadBool(L"ProxyLocalhost", ProxyLocalhost);  #define READ_BUG(BUG) \    Bug[sb##BUG] = TAutoSwitch(2 - Storage->ReadInteger(L"Bug"#BUG, \      2 - Bug[sb##BUG]));  READ_BUG(Ignore1);  READ_BUG(PlainPW1);  READ_BUG(RSA1);  READ_BUG(HMAC2);  READ_BUG(DeriveKey2);  READ_BUG(RSAPad2);  READ_BUG(PKSessID2);  READ_BUG(Rekey2);  READ_BUG(MaxPkt2);  READ_BUG(Ignore2);  READ_BUG(OldGex2);  READ_BUG(WinAdj);  #undef READ_BUG  if ((Bug[sbHMAC2] == asAuto) &&      Storage->ReadBool(L"BuggyMAC", false))  {      Bug[sbHMAC2] = asOn;  }  SftpServer = Storage->ReadString(L"SftpServer", SftpServer);  #define READ_SFTP_BUG(BUG) \    SFTPBug[sb##BUG] = TAutoSwitch(Storage->ReadInteger(L"SFTP" #BUG "Bug", SFTPBug[sb##BUG]));  READ_SFTP_BUG(Symlink);  READ_SFTP_BUG(SignedTS);  #undef READ_SFTP_BUG  SFTPMaxVersion = Storage->ReadInteger(L"SFTPMaxVersion", SFTPMaxVersion);  SFTPMaxPacketSize = Storage->ReadInteger(L"SFTPMaxPacketSize", SFTPMaxPacketSize);  SFTPDownloadQueue = Storage->ReadInteger(L"SFTPDownloadQueue", SFTPDownloadQueue);  SFTPUploadQueue = Storage->ReadInteger(L"SFTPUploadQueue", SFTPUploadQueue);  SFTPListingQueue = Storage->ReadInteger(L"SFTPListingQueue", SFTPListingQueue);  Color = Storage->ReadInteger(L"Color", Color);  PuttyProtocol = Storage->ReadString(L"Protocol", PuttyProtocol);  Tunnel = Storage->ReadBool(L"Tunnel", Tunnel);  TunnelPortNumber = Storage->ReadInteger(L"TunnelPortNumber", TunnelPortNumber);  TunnelUserName = Storage->ReadString(L"TunnelUserName", TunnelUserName);  // must be loaded after TunnelUserName,  // because TunnelHostName may be in format user@host  TunnelHostName = Storage->ReadString(L"TunnelHostName", TunnelHostName);  if (!Configuration->DisablePasswordStoring)  {    if (Storage->ValueExists(L"TunnelPasswordPlain"))    {      TunnelPassword = Storage->ReadString(L"TunnelPasswordPlain", TunnelPassword);      RewritePassword = true;    }    else    {      FTunnelPassword = Storage->ReadStringAsBinaryData(L"TunnelPassword", FTunnelPassword);    }  }  TunnelPublicKeyFile = Storage->ReadString(L"TunnelPublicKeyFile", TunnelPublicKeyFile);  TunnelLocalPortNumber = Storage->ReadInteger(L"TunnelLocalPortNumber", TunnelLocalPortNumber);  TunnelHostKey = Storage->ReadString(L"TunnelHostKey", TunnelHostKey);  // Ftp prefix  FtpPasvMode = Storage->ReadBool(L"FtpPasvMode", FtpPasvMode);  FtpForcePasvIp = TAutoSwitch(Storage->ReadInteger(L"FtpForcePasvIp2", FtpForcePasvIp));  FtpUseMlsd = TAutoSwitch(Storage->ReadInteger(L"FtpUseMlsd", FtpUseMlsd));  FtpAccount = Storage->ReadString(L"FtpAccount", FtpAccount);  FtpPingInterval = Storage->ReadInteger(L"FtpPingInterval", FtpPingInterval);  FtpPingType = static_cast<TPingType>(Storage->ReadInteger(L"FtpPingType", FtpPingType));  FtpTransferActiveImmediately = static_cast<TAutoSwitch>(Storage->ReadInteger(L"FtpTransferActiveImmediately2", FtpTransferActiveImmediately));  Ftps = static_cast<TFtps>(Storage->ReadInteger(L"Ftps", Ftps));  FtpListAll = TAutoSwitch(Storage->ReadInteger(L"FtpListAll", FtpListAll));  FtpHost = TAutoSwitch(Storage->ReadInteger(L"FtpHost", FtpHost));  SslSessionReuse = Storage->ReadBool(L"SslSessionReuse", SslSessionReuse);  TlsCertificateFile = Storage->ReadString(L"TlsCertificateFile", TlsCertificateFile);  FtpProxyLogonType = Storage->ReadInteger(L"FtpProxyLogonType", FtpProxyLogonType);  MinTlsVersion = static_cast<TTlsVersion>(Storage->ReadInteger(L"MinTlsVersion", MinTlsVersion));  MaxTlsVersion = static_cast<TTlsVersion>(Storage->ReadInteger(L"MaxTlsVersion", MaxTlsVersion));  IsWorkspace = Storage->ReadBool(L"IsWorkspace", IsWorkspace);  Link = Storage->ReadString(L"Link", Link);  CustomParam1 = Storage->ReadString(L"CustomParam1", CustomParam1);  CustomParam2 = Storage->ReadString(L"CustomParam2", CustomParam2);#ifdef TEST  #define KEX_TEST(VALUE, EXPECTED) KexList = VALUE; DebugAssert(KexList == EXPECTED);  #define KEX_DEFAULT L"ecdh,dh-gex-sha1,dh-group14-sha1,rsa,WARN,dh-group1-sha1"  // Empty source should result in default list  KEX_TEST(L"", KEX_DEFAULT);  // Default of pre 5.8.1  KEX_TEST(L"dh-gex-sha1,dh-group14-sha1,dh-group1-sha1,rsa,WARN", L"ecdh,dh-gex-sha1,dh-group14-sha1,dh-group1-sha1,rsa,WARN");  // Missing first two priority algos, and last non-priority algo  KEX_TEST(L"dh-group14-sha1,dh-group1-sha1,WARN", L"ecdh,dh-gex-sha1,dh-group14-sha1,dh-group1-sha1,rsa,WARN");  // Missing first two priority algos, last non-priority algo and WARN  KEX_TEST(L"dh-group14-sha1,dh-group1-sha1", L"ecdh,dh-gex-sha1,dh-group14-sha1,dh-group1-sha1,rsa,WARN");  // Old algos, with all but the first below WARN  KEX_TEST(L"dh-gex-sha1,WARN,dh-group14-sha1,dh-group1-sha1,rsa", L"ecdh,dh-gex-sha1,WARN,dh-group14-sha1,dh-group1-sha1,rsa");  // Unknown algo at front  KEX_TEST(L"unknown,ecdh,dh-gex-sha1,dh-group14-sha1,rsa,WARN,dh-group1-sha1", KEX_DEFAULT);  // Unknown algo at back  KEX_TEST(L"ecdh,dh-gex-sha1,dh-group14-sha1,rsa,WARN,dh-group1-sha1,unknown", KEX_DEFAULT);  // Unknown algo in the middle  KEX_TEST(L"ecdh,dh-gex-sha1,dh-group14-sha1,unknown,rsa,WARN,dh-group1-sha1", KEX_DEFAULT);  #undef KEX_DEFAULT  #undef KEX_TEST  #define CIPHER_TEST(VALUE, EXPECTED) CipherList = VALUE; DebugAssert(CipherList == EXPECTED);  #define CIPHER_DEFAULT L"aes,chacha20,blowfish,3des,WARN,arcfour,des"  // Empty source should result in default list  CIPHER_TEST(L"", CIPHER_DEFAULT);  // Default of pre 5.8.1  CIPHER_TEST(L"aes,blowfish,3des,WARN,arcfour,des", L"aes,blowfish,3des,chacha20,WARN,arcfour,des");  // Missing priority algo  CIPHER_TEST(L"chacha20,blowfish,3des,WARN,arcfour,des", CIPHER_DEFAULT);  // Missing non-priority algo  CIPHER_TEST(L"aes,chacha20,3des,WARN,arcfour,des", L"aes,chacha20,3des,blowfish,WARN,arcfour,des");  // Missing last warn algo  CIPHER_TEST(L"aes,blowfish,chacha20,3des,WARN,arcfour", L"aes,blowfish,chacha20,3des,WARN,arcfour,des");  // Missing first warn algo  CIPHER_TEST(L"aes,blowfish,chacha20,3des,WARN,des", L"aes,blowfish,chacha20,3des,WARN,des,arcfour");  #undef CIPHER_DEFAULT  #undef CIPHER_TEST#endif}//---------------------------------------------------------------------void __fastcall TSessionData::Load(THierarchicalStorage * Storage){  bool RewritePassword = false;  if (Storage->OpenSubKey(InternalStorageKey, False))  {    // In case we are re-loading, reset passwords, to avoid pointless    // re-cryption, while loading username/hostname. And moreover, when    // the password is wrongly encrypted (using a different master password),    // this breaks sites reload and consequently an overal operation,    // such as opening Sites menu    ClearSessionPasswords();    FProxyPassword = L"";    DoLoad(Storage, RewritePassword);    Storage->CloseSubKey();  }  if (RewritePassword)  {    TStorageAccessMode AccessMode = Storage->AccessMode;    Storage->AccessMode = smReadWrite;    try    {      if (Storage->OpenSubKey(InternalStorageKey, true))      {        Storage->DeleteValue(L"PasswordPlain");        if (!Password.IsEmpty())        {          Storage->WriteBinaryDataAsString(L"Password", FPassword);        }        Storage->DeleteValue(L"TunnelPasswordPlain");        if (!TunnelPassword.IsEmpty())        {          Storage->WriteBinaryDataAsString(L"TunnelPassword", FTunnelPassword);        }        Storage->CloseSubKey();      }    }    catch(...)    {      // ignore errors (like read-only INI file)    }    Storage->AccessMode = AccessMode;  }  FModified = false;  FSource = ssStored;}//---------------------------------------------------------------------void __fastcall TSessionData::DoSave(THierarchicalStorage * Storage,  bool PuttyExport, const TSessionData * Default, bool DoNotEncryptPasswords){  #define WRITE_DATA_EX(TYPE, NAME, PROPERTY, CONV) \    if ((Default != NULL) && (CONV(Default->PROPERTY) == CONV(PROPERTY))) \    { \      Storage->DeleteValue(NAME); \    } \    else \    { \      Storage->Write ## TYPE(NAME, CONV(PROPERTY)); \    }  #define WRITE_DATA_CONV(TYPE, NAME, PROPERTY) WRITE_DATA_EX(TYPE, NAME, PROPERTY, WRITE_DATA_CONV_FUNC)  #define WRITE_DATA(TYPE, PROPERTY) WRITE_DATA_EX(TYPE, TEXT(#PROPERTY), PROPERTY, )  WRITE_DATA(String, HostName);  WRITE_DATA(Integer, PortNumber);  WRITE_DATA_EX(Integer, L"PingInterval", PingInterval / SecsPerMin, );  WRITE_DATA_EX(Integer, L"PingIntervalSecs", PingInterval % SecsPerMin, );  Storage->DeleteValue(L"PingIntervalSec"); // obsolete  WRITE_DATA(Integer, PingType);  WRITE_DATA(Integer, Timeout);  WRITE_DATA(Bool, TryAgent);  WRITE_DATA(Bool, AgentFwd);  WRITE_DATA(Bool, AuthTIS);  WRITE_DATA(Bool, AuthKI);  WRITE_DATA(Bool, AuthKIPassword);  WRITE_DATA(String, Note);  WRITE_DATA(Bool, AuthGSSAPI);  WRITE_DATA(Bool, GSSAPIFwdTGT);  Storage->DeleteValue(L"TryGSSKEX");  Storage->DeleteValue(L"UserNameFromEnvironment");  Storage->DeleteValue("GSSAPIServerChoosesUserName");  Storage->DeleteValue(L"GSSAPITrustDNS");  if (PuttyExport)  {    // duplicate kerberos setting with keys of the vintela quest putty    WRITE_DATA_EX(Bool, L"AuthSSPI", AuthGSSAPI, );    WRITE_DATA_EX(Bool, L"SSPIFwdTGT", GSSAPIFwdTGT, );    // duplicate kerberos setting with keys of the official putty    WRITE_DATA_EX(Bool, L"GssapiFwd", GSSAPIFwdTGT, );  }  WRITE_DATA(Bool, ChangeUsername);  WRITE_DATA(Bool, Compression);  WRITE_DATA(Integer, SshProt);  WRITE_DATA(Bool, Ssh2DES);  WRITE_DATA(Bool, SshNoUserAuth);  WRITE_DATA_EX(String, L"Cipher", CipherList, );  WRITE_DATA_EX(String, L"KEX", KexList, );  WRITE_DATA(Integer, AddressFamily);  WRITE_DATA_EX(String, L"RekeyBytes", RekeyData, );  WRITE_DATA(Integer, RekeyTime);  WRITE_DATA(Bool, TcpNoDelay);  if (PuttyExport)  {    WRITE_DATA(StringRaw, UserName);    // PuTTY is started in its binary directory to allow relative paths when opening PuTTY's own stored session.    // To allow relative paths in our sessions, we have to expand them for PuTTY.    WRITE_DATA_EX(StringRaw, L"PublicKeyFile", PublicKeyFile, ExpandFileName);  }  else  {    WRITE_DATA(String, UserName);    WRITE_DATA(String, PublicKeyFile);    WRITE_DATA(Integer, FSProtocol);    WRITE_DATA(String, LocalDirectory);    WRITE_DATA(String, RemoteDirectory);    WRITE_DATA(Bool, SynchronizeBrowsing);    WRITE_DATA(Bool, UpdateDirectories);    WRITE_DATA(Bool, CacheDirectories);    WRITE_DATA(Bool, CacheDirectoryChanges);    WRITE_DATA(Bool, PreserveDirectoryChanges);    WRITE_DATA(Bool, ResolveSymlinks);    WRITE_DATA(Bool, FollowDirectorySymlinks);    WRITE_DATA_EX(Integer, L"ConsiderDST", DSTMode, );    WRITE_DATA(Bool, LockInHome);    // Special is never stored (if it would, login dialog must be modified not to    // duplicate Special parameter when Special session is loaded and then stored    // under different name)    // WRITE_DATA(Bool, Special);    WRITE_DATA(String, Shell);    WRITE_DATA(Bool, ClearAliases);    WRITE_DATA(Bool, UnsetNationalVars);    WRITE_DATA(String, ListingCommand);    WRITE_DATA(Bool, IgnoreLsWarnings);    WRITE_DATA(Integer, SCPLsFullTime);    WRITE_DATA(Bool, Scp1Compatibility);    // TimeDifferenceAuto is valid for FTP protocol only.    // For other protocols it's typically true (default value),    // but ignored so TimeDifference is still taken into account (SCP only actually)    if (TimeDifferenceAuto && (FSProtocol == fsFTP))    {      // Have to delete it as TimeDifferenceAuto is not saved when enabled,      // but the default is derived from value of TimeDifference.      Storage->DeleteValue(L"TimeDifference");    }    else    {      WRITE_DATA(Float, TimeDifference);    }    WRITE_DATA(Bool, TimeDifferenceAuto);    WRITE_DATA(Bool, DeleteToRecycleBin);    WRITE_DATA(Bool, OverwrittenToRecycleBin);    WRITE_DATA(String, RecycleBinPath);    WRITE_DATA(String, PostLoginCommands);    WRITE_DATA(String, ReturnVar);    WRITE_DATA_EX(Integer, L"LookupUserGroups2", LookupUserGroups, );    WRITE_DATA(Integer, EOLType);    WRITE_DATA(Bool, TrimVMSVersions);    Storage->DeleteValue(L"SFTPUtfBug");    WRITE_DATA_EX(Integer, L"Utf", NotUtf, );    WRITE_DATA(Integer, SendBuf);    WRITE_DATA(Bool, SshSimple);  }  WRITE_DATA(Integer, ProxyMethod);  WRITE_DATA(String, ProxyHost);  WRITE_DATA(Integer, ProxyPort);  WRITE_DATA(String, ProxyUsername);  if (ProxyMethod == pmCmd)  {    WRITE_DATA_EX(StringRaw, L"ProxyTelnetCommand", ProxyLocalCommand, );  }  else  {    WRITE_DATA(StringRaw, ProxyTelnetCommand);  }  #define WRITE_DATA_CONV_FUNC(X) (((X) + 2) % 3)  WRITE_DATA_CONV(Integer, L"ProxyDNS", ProxyDNS);  #undef WRITE_DATA_CONV_FUNC  WRITE_DATA(Bool, ProxyLocalhost);  #define WRITE_DATA_CONV_FUNC(X) (2 - (X))  #define WRITE_BUG(BUG) WRITE_DATA_CONV(Integer, L"Bug" #BUG, Bug[sb##BUG]);  WRITE_BUG(Ignore1);  WRITE_BUG(PlainPW1);  WRITE_BUG(RSA1);  WRITE_BUG(HMAC2);  WRITE_BUG(DeriveKey2);  WRITE_BUG(RSAPad2);  WRITE_BUG(PKSessID2);  WRITE_BUG(Rekey2);  WRITE_BUG(MaxPkt2);  WRITE_BUG(Ignore2);  WRITE_BUG(OldGex2);  WRITE_BUG(WinAdj);  #undef WRITE_BUG  #undef WRITE_DATA_CONV_FUNC  Storage->DeleteValue(L"BuggyMAC");  Storage->DeleteValue(L"AliasGroupList");  if (PuttyExport)  {    WRITE_DATA_EX(String, L"Protocol", GetNormalizedPuttyProtocol(), );  }  if (!PuttyExport)  {    WRITE_DATA(String, SftpServer);    #define WRITE_SFTP_BUG(BUG) WRITE_DATA_EX(Integer, L"SFTP" #BUG "Bug", SFTPBug[sb##BUG], );    WRITE_SFTP_BUG(Symlink);    WRITE_SFTP_BUG(SignedTS);    #undef WRITE_SFTP_BUG    WRITE_DATA(Integer, SFTPMaxVersion);    WRITE_DATA(Integer, SFTPMaxPacketSize);    WRITE_DATA(Integer, SFTPDownloadQueue);    WRITE_DATA(Integer, SFTPUploadQueue);    WRITE_DATA(Integer, SFTPListingQueue);    WRITE_DATA(Integer, Color);    WRITE_DATA(Bool, Tunnel);    WRITE_DATA(String, TunnelHostName);    WRITE_DATA(Integer, TunnelPortNumber);    WRITE_DATA(String, TunnelUserName);    WRITE_DATA(String, TunnelPublicKeyFile);    WRITE_DATA(Integer, TunnelLocalPortNumber);    WRITE_DATA(Bool, FtpPasvMode);    WRITE_DATA_EX(Integer, L"FtpForcePasvIp2", FtpForcePasvIp, );    WRITE_DATA(Integer, FtpUseMlsd);    WRITE_DATA(String, FtpAccount);    WRITE_DATA(Integer, FtpPingInterval);    WRITE_DATA(Integer, FtpPingType);    WRITE_DATA_EX(Integer, L"FtpTransferActiveImmediately2", FtpTransferActiveImmediately, );    WRITE_DATA(Integer, Ftps);    WRITE_DATA(Integer, FtpListAll);    WRITE_DATA(Integer, FtpHost);    WRITE_DATA(Bool, SslSessionReuse);    WRITE_DATA(String, TlsCertificateFile);    WRITE_DATA(Integer, FtpProxyLogonType);    WRITE_DATA(Integer, MinTlsVersion);    WRITE_DATA(Integer, MaxTlsVersion);    WRITE_DATA(Bool, IsWorkspace);    WRITE_DATA(String, Link);    WRITE_DATA(String, CustomParam1);    WRITE_DATA(String, CustomParam2);  }  SavePasswords(Storage, PuttyExport, DoNotEncryptPasswords);}//---------------------------------------------------------------------TStrings * __fastcall TSessionData::SaveToOptions(const TSessionData * Default){  std::unique_ptr<TStringList> Options(new TStringList());  std::unique_ptr<TOptionsStorage> OptionsStorage(new TOptionsStorage(Options.get(), true));  DoSave(OptionsStorage.get(), false, Default, true);  return Options.release();}//---------------------------------------------------------------------void __fastcall TSessionData::Save(THierarchicalStorage * Storage,  bool PuttyExport, const TSessionData * Default){  if (Storage->OpenSubKey(InternalStorageKey, true))  {    DoSave(Storage, PuttyExport, Default, false);    Storage->CloseSubKey();  }}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::ReadXmlNode(_di_IXMLNode Node, const UnicodeString & Name, const UnicodeString & Default){  _di_IXMLNode TheNode = Node->ChildNodes->FindNode(Name);  UnicodeString Result;  if (TheNode != NULL)  {    Result = TheNode->Text.Trim();  }  if (Result.IsEmpty())  {    Result = Default;  }  return Result;}//---------------------------------------------------------------------int __fastcall TSessionData::ReadXmlNode(_di_IXMLNode Node, const UnicodeString & Name, int Default){  _di_IXMLNode TheNode = Node->ChildNodes->FindNode(Name);  int Result;  if (TheNode != NULL)  {    Result = StrToIntDef(TheNode->Text.Trim(), Default);  }  else  {    Result = Default;  }  return Result;}//---------------------------------------------------------------------void __fastcall TSessionData::ImportFromFilezilla(_di_IXMLNode Node, const UnicodeString & Path){  Name = UnixIncludeTrailingBackslash(Path) + MakeValidName(ReadXmlNode(Node, L"Name", Name));  HostName = ReadXmlNode(Node, L"Host", HostName);  PortNumber = ReadXmlNode(Node, L"Port", PortNumber);  int AProtocol = ReadXmlNode(Node, L"Protocol", 0);  // ServerProtocol enum  switch (AProtocol)  {    case 0: // FTP    default: // UNKNOWN, HTTP, HTTPS, INSECURE_FTP      FSProtocol = fsFTP;      break;    case 1: // SFTP      FSProtocol = fsSFTP;      break;    case 3: // FTPS      FSProtocol = fsFTP;      Ftps = ftpsImplicit;      break;    case 4: // FTPES      FSProtocol = fsFTP;      Ftps = ftpsExplicitTls;      break;  }  // LogonType enum  int LogonType = ReadXmlNode(Node, L"Logontype", 0);  if (LogonType == 0) // ANONYMOUS  {    UserName = AnonymousUserName;    Password = AnonymousPassword;  }  else  {    UserName = ReadXmlNode(Node, L"User", UserName);    FtpAccount = ReadXmlNode(Node, L"Account", FtpAccount);    _di_IXMLNode PassNode = Node->ChildNodes->FindNode(L"Pass");    if (PassNode != NULL)    {      UnicodeString APassword = PassNode->Text.Trim();      OleVariant EncodingValue = PassNode->GetAttribute(L"encoding");      if (!EncodingValue.IsNull())      {        UnicodeString EncodingValueStr = EncodingValue;        if (SameText(EncodingValueStr, L"base64"))        {          TBytes Bytes = DecodeBase64(APassword);          APassword = TEncoding::UTF8->GetString(Bytes);        }      }      Password = APassword;    }  }  int DefaultTimeDifference = TimeToSeconds(TimeDifference);  TimeDifference =    (double(ReadXmlNode(Node, L"TimezoneOffset", DefaultTimeDifference) / SecsPerDay));  TimeDifferenceAuto = (TimeDifference == TDateTime());  UnicodeString PasvMode = ReadXmlNode(Node, L"PasvMode", L"");  if (SameText(PasvMode, L"MODE_PASSIVE"))  {    FtpPasvMode = true;  }  else if (SameText(PasvMode, L"MODE_ACTIVE"))  {    FtpPasvMode = false;  }  UnicodeString EncodingType = ReadXmlNode(Node, L"EncodingType", L"");  if (SameText(EncodingType, L"Auto"))  {    NotUtf = asAuto;  }  else if (SameText(EncodingType, L"UTF-8"))  {    NotUtf = asOff;  }  // todo PostLoginCommands  Note = ReadXmlNode(Node, L"Comments", Note);  LocalDirectory = ReadXmlNode(Node, L"LocalDir", LocalDirectory);  UnicodeString RemoteDir = ReadXmlNode(Node, L"RemoteDir", L"");  if (!RemoteDir.IsEmpty())  {    CutToChar(RemoteDir, L' ', false); // type    int PrefixSize = StrToIntDef(CutToChar(RemoteDir, L' ', false), 0); // prefix size    if (PrefixSize > 0)    {      RemoteDir.Delete(1, PrefixSize);    }    RemoteDirectory = L"/";    while (!RemoteDir.IsEmpty())    {      int SegmentSize = StrToIntDef(CutToChar(RemoteDir, L' ', false), 0);      UnicodeString Segment = RemoteDir.SubString(1, SegmentSize);      RemoteDirectory = UnixIncludeTrailingBackslash(RemoteDirectory) + Segment;      RemoteDir.Delete(1, SegmentSize + 1);    }  }  SynchronizeBrowsing = (ReadXmlNode(Node, L"SyncBrowsing", SynchronizeBrowsing ? 1 : 0) != 0);}//---------------------------------------------------------------------void __fastcall TSessionData::SavePasswords(THierarchicalStorage * Storage, bool PuttyExport, bool DoNotEncryptPasswords){  if (!Configuration->DisablePasswordStoring && !PuttyExport && !FPassword.IsEmpty())  {    // DoNotEncryptPasswords is set when called from GenerateOpenCommandArgs only    // and it never saves session password    DebugAssert(!DoNotEncryptPasswords);    Storage->WriteBinaryDataAsString(L"Password", StronglyRecryptPassword(FPassword, UserName+HostName));  }  else  {    Storage->DeleteValue(L"Password");  }  Storage->DeleteValue(L"PasswordPlain");  if (PuttyExport)  {    // save password unencrypted    Storage->WriteString(L"ProxyPassword", ProxyPassword);  }  else  {    if (DoNotEncryptPasswords)    {      if (!FProxyPassword.IsEmpty())      {        Storage->WriteString(L"ProxyPassword", ProxyPassword);      }      else      {        Storage->DeleteValue(L"ProxyPassword");      }      Storage->DeleteValue(L"ProxyPasswordEnc");    }    else    {      // save password encrypted      if (!FProxyPassword.IsEmpty())      {        Storage->WriteBinaryDataAsString(L"ProxyPasswordEnc", StronglyRecryptPassword(FProxyPassword, ProxyUsername+ProxyHost));      }      else      {        Storage->DeleteValue(L"ProxyPasswordEnc");      }      Storage->DeleteValue(L"ProxyPassword");    }    if (DoNotEncryptPasswords)    {      if (!FTunnelPassword.IsEmpty())      {        Storage->WriteString(L"TunnelPasswordPlain", TunnelPassword);      }      else      {        Storage->DeleteValue(L"TunnelPasswordPlain");      }    }    else    {      if (!Configuration->DisablePasswordStoring && !FTunnelPassword.IsEmpty())      {        Storage->WriteBinaryDataAsString(L"TunnelPassword", StronglyRecryptPassword(FTunnelPassword, TunnelUserName+TunnelHostName));      }      else      {        Storage->DeleteValue(L"TunnelPassword");      }    }  }}//---------------------------------------------------------------------void __fastcall TSessionData::RecryptPasswords(){  Password = Password;  ProxyPassword = ProxyPassword;  TunnelPassword = TunnelPassword;  Passphrase = Passphrase;}//---------------------------------------------------------------------bool __fastcall TSessionData::HasPassword(){  return !FPassword.IsEmpty();}//---------------------------------------------------------------------bool __fastcall TSessionData::HasAnySessionPassword(){  return HasPassword() || !FTunnelPassword.IsEmpty();}//---------------------------------------------------------------------bool __fastcall TSessionData::HasAnyPassword(){  return HasAnySessionPassword() || !FProxyPassword.IsEmpty();}//---------------------------------------------------------------------void __fastcall TSessionData::ClearSessionPasswords(){  FPassword = L"";  FTunnelPassword = L"";}//---------------------------------------------------------------------void __fastcall TSessionData::Modify(){  FModified = true;  if (FSource == ssStored)  {    FSource = ssStoredModified;  }}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetSource(){  switch (FSource)  {    case ::ssNone:      return L"Ad-Hoc site";    case ssStored:      return L"Site";    case ssStoredModified:      return L"Modified site";    default:      DebugFail();      return L"";  }}//---------------------------------------------------------------------void __fastcall TSessionData::SaveRecryptedPasswords(THierarchicalStorage * Storage){  if (Storage->OpenSubKey(InternalStorageKey, true))  {    try    {      RecryptPasswords();      SavePasswords(Storage, false, false);    }    __finally    {      Storage->CloseSubKey();    }  }}//---------------------------------------------------------------------void __fastcall TSessionData::Remove(){  bool SessionList = true;  THierarchicalStorage * Storage = Configuration->CreateScpStorage(SessionList);  try  {    Storage->Explicit = true;    if (Storage->OpenSubKey(Configuration->StoredSessionsSubKey, false))    {      Storage->RecursiveDeleteSubKey(InternalStorageKey);    }  }  __finally  {    delete Storage;  }}//---------------------------------------------------------------------void __fastcall TSessionData::CacheHostKeyIfNotCached(){  UnicodeString KeyType = KeyTypeFromFingerprint(HostKey);  UnicodeString TargetKey = Configuration->RegistryStorageKey + L"\\" + Configuration->SshHostKeysSubKey;  std::unique_ptr<TRegistryStorage> Storage(new TRegistryStorage(TargetKey));  Storage->AccessMode = smReadWrite;  if (Storage->OpenRootKey(true))  {    UnicodeString HostKeyName = PuttyMungeStr(FORMAT(L"%s@%d:%s", (KeyType, PortNumber, HostName)));    if (!Storage->ValueExists(HostKeyName))    {      // fingerprint is MD5 of host key, so it cannot be translated back to host key,      // so we store fingerprint and TSecureShell::VerifyHostKey was      // modified to accept also fingerprint      Storage->WriteString(HostKeyName, HostKey);    }  }}//---------------------------------------------------------------------inline void __fastcall MoveStr(UnicodeString & Source, UnicodeString * Dest, int Count){  if (Dest != NULL)  {    (*Dest) += Source.SubString(1, Count);  }  Source.Delete(1, Count);}//---------------------------------------------------------------------bool __fastcall TSessionData::DoIsProtocolUrl(  const UnicodeString & Url, const UnicodeString & Protocol, int & ProtocolLen){  bool Result = SameText(Url.SubString(1, Protocol.Length() + 1), Protocol + L":");  if (Result)  {    ProtocolLen = Protocol.Length() + 1;  }  return Result;}//---------------------------------------------------------------------bool __fastcall TSessionData::IsProtocolUrl(  const UnicodeString & Url, const UnicodeString & Protocol, int & ProtocolLen){  return    DoIsProtocolUrl(Url, Protocol, ProtocolLen) ||    DoIsProtocolUrl(Url, WinSCPProtocolPrefix + Protocol, ProtocolLen);}//---------------------------------------------------------------------bool __fastcall TSessionData::IsSensitiveOption(const UnicodeString & Option){  return SameText(Option, PassphraseOption);}//---------------------------------------------------------------------bool __fastcall TSessionData::ParseUrl(UnicodeString Url, TOptions * Options,  TStoredSessionList * StoredSessions, bool & DefaultsOnly, UnicodeString * FileName,  bool * AProtocolDefined, UnicodeString * MaskedUrl){  bool ProtocolDefined = false;  bool PortNumberDefined = false;  TFSProtocol AFSProtocol;  int APortNumber;  TFtps AFtps = ftpsNone;  int ProtocolLen = 0;  if (IsProtocolUrl(Url, ScpProtocol, ProtocolLen))  {    AFSProtocol = fsSCPonly;    APortNumber = SshPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  else if (IsProtocolUrl(Url, SftpProtocol, ProtocolLen))  {    AFSProtocol = fsSFTPonly;    APortNumber = SshPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  else if (IsProtocolUrl(Url, FtpProtocol, ProtocolLen))  {    AFSProtocol = fsFTP;    Ftps = ftpsNone;    APortNumber = FtpPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  else if (IsProtocolUrl(Url, FtpsProtocol, ProtocolLen))  {    AFSProtocol = fsFTP;    AFtps = ftpsImplicit;    APortNumber = FtpsImplicitPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  else if (IsProtocolUrl(Url, FtpesProtocol, ProtocolLen))  {    AFSProtocol = fsFTP;    AFtps = ftpsExplicitTls;    APortNumber = FtpPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  else if (IsProtocolUrl(Url, WebDAVProtocol, ProtocolLen))  {    AFSProtocol = fsWebDAV;    AFtps = ftpsNone;    APortNumber = HTTPPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  else if (IsProtocolUrl(Url, WebDAVSProtocol, ProtocolLen))  {    AFSProtocol = fsWebDAV;    AFtps = ftpsImplicit;    APortNumber = HTTPSPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  else if (IsProtocolUrl(Url, SshProtocol, ProtocolLen))  {    // For most uses, handling ssh:// the same way as sftp://    // The only place where a difference is made is GetLoginData() in WinMain.cpp    AFSProtocol = fsSFTPonly;    PuttyProtocol = PuttySshProtocol;    APortNumber = SshPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  if (ProtocolDefined && (Url.SubString(1, 2) == L"//"))  {    MoveStr(Url, MaskedUrl, 2);  }  if (AProtocolDefined != NULL)  {    *AProtocolDefined = ProtocolDefined;  }  if (!Url.IsEmpty())  {    UnicodeString DecodedUrl = DecodeUrlChars(Url);    // lookup stored session even if protocol was defined    // (this allows setting for example default username for host    // by creating stored session named by host)    TSessionData * Data = NULL;    // When using to paste URL on Login dialog, we do not want to lookup the stored sites    if (StoredSessions != NULL)    {      // this can be optimized as the list is sorted      for (Integer Index = 0; Index < StoredSessions->CountIncludingHidden; Index++)      {        TSessionData * AData = (TSessionData *)StoredSessions->Items[Index];        if (!AData->IsWorkspace)        {          bool Match = false;          // Comparison optimizations as this is called many times          // e.g. when updating jumplist          if ((AData->Name.Length() == DecodedUrl.Length()) &&              SameText(AData->Name, DecodedUrl))          {            Match = true;          }          else if ((AData->Name.Length() < DecodedUrl.Length()) &&                   (DecodedUrl[AData->Name.Length() + 1] == L'/') &&                   // StrLIComp is an equivalent of SameText                   (StrLIComp(AData->Name.c_str(), DecodedUrl.c_str(), AData->Name.Length()) == 0))          {            Match = true;          }          if (Match)          {            Data = AData;            break;          }        }      }    }    UnicodeString ARemoteDirectory;    if (Data != NULL)    {      Assign(Data);      int P = 1;      while (!AnsiSameText(DecodeUrlChars(Url.SubString(1, P)), Data->Name))      {        P++;        DebugAssert(P <= Url.Length());      }      ARemoteDirectory = Url.SubString(P + 1, Url.Length() - P);      if (Data->Hidden)      {        Data->Remove();        StoredSessions->Remove(Data);        // only modified, implicit        StoredSessions->Save(false, false);      }      if (MaskedUrl != NULL)      {        (*MaskedUrl) += Url;      }    }    else    {      // This happens when pasting URL on Login dialog      if (StoredSessions != NULL)      {        CopyData(StoredSessions->DefaultSettings);      }      Name = L"";      int PSlash = Url.Pos(L"/");      if (PSlash == 0)      {        PSlash = Url.Length() + 1;      }      UnicodeString ConnectInfo = Url.SubString(1, PSlash - 1);      int P = ConnectInfo.LastDelimiter(L"@");      UnicodeString UserInfo;      UnicodeString HostInfo;      if (P > 0)      {        UserInfo = ConnectInfo.SubString(1, P - 1);        HostInfo = ConnectInfo.SubString(P + 1, ConnectInfo.Length() - P);      }      else      {        HostInfo = ConnectInfo;      }      UnicodeString OrigHostInfo = HostInfo;      if ((HostInfo.Length() >= 2) && (HostInfo[1] == L'[') && ((P = HostInfo.Pos(L"]")) > 0))      {        HostName = HostInfo.SubString(2, P - 2);        HostInfo.Delete(1, P);        if (!HostInfo.IsEmpty() && (HostInfo[1] == L':'))        {          HostInfo.Delete(1, 1);        }      }      else      {        HostName = DecodeUrlChars(CutToChar(HostInfo, L':', true));      }      // expanded from ?: operator, as it caused strange "access violation" errors      if (!HostInfo.IsEmpty())      {        PortNumber = StrToIntDef(DecodeUrlChars(HostInfo), -1);        PortNumberDefined = true;      }      else if (ProtocolDefined)      {        PortNumber = APortNumber;      }      if (ProtocolDefined)      {        Ftps = AFtps;      }      UnicodeString UserInfoWithoutConnectionParams = CutToChar(UserInfo, UrlParamSeparator, false);      UnicodeString ConnectionParams = UserInfo;      UserInfo = UserInfoWithoutConnectionParams;      while (!ConnectionParams.IsEmpty())      {        UnicodeString ConnectionParam = CutToChar(ConnectionParams, UrlParamSeparator, false);        UnicodeString ConnectionParamName = CutToChar(ConnectionParam, UrlParamValueSeparator, false);        if (SameText(ConnectionParamName, UrlHostKeyParamName))        {          HostKey = ConnectionParam;          FOverrideCachedHostKey = false;        }      }      UnicodeString RawUserName = CutToChar(UserInfo, L':', false);      UserName = DecodeUrlChars(RawUserName);      Password = DecodeUrlChars(UserInfo);      UnicodeString RemoteDirectoryWithSessionParams = Url.SubString(PSlash, Url.Length() - PSlash + 1);      ARemoteDirectory = CutToChar(RemoteDirectoryWithSessionParams, UrlParamSeparator, false);      UnicodeString SessionParams = RemoteDirectoryWithSessionParams;      // We should handle session params in "stored session" branch too.      // And particularly if there's a "save" param, we should actually not try to match the      // URL against site names      while (!SessionParams.IsEmpty())      {        UnicodeString SessionParam = CutToChar(SessionParams, UrlParamSeparator, false);        UnicodeString SessionParamName = CutToChar(SessionParam, UrlParamValueSeparator, false);        if (SameText(SessionParamName, UrlSaveParamName))        {          FSaveOnly = (StrToIntDef(SessionParam, 1) != 0);        }      }      if (MaskedUrl != NULL)      {        (*MaskedUrl) += RawUserName;        if (!UserInfo.IsEmpty())        {          (*MaskedUrl) += L":" + PasswordMask;        }        if (!RawUserName.IsEmpty() || !UserInfo.IsEmpty())        {          (*MaskedUrl) += L"@";        }        (*MaskedUrl) += OrigHostInfo + ARemoteDirectory;      }    }    if (!ARemoteDirectory.IsEmpty() && (ARemoteDirectory != L"/"))    {      if ((ARemoteDirectory[ARemoteDirectory.Length()] != L'/') &&          (FileName != NULL))      {        *FileName = DecodeUrlChars(UnixExtractFileName(ARemoteDirectory));        ARemoteDirectory = UnixExtractFilePath(ARemoteDirectory);      }      RemoteDirectory = DecodeUrlChars(ARemoteDirectory);    }    DefaultsOnly = false;  }  else  {    // This happens when pasting URL on Login dialog    if (StoredSessions != NULL)    {      CopyData(StoredSessions->DefaultSettings);    }    DefaultsOnly = true;  }  if (ProtocolDefined)  {    FSProtocol = AFSProtocol;  }  if (Options != NULL)  {    // we deliberately do keep defaultonly to false, in presence of any option,    // as the option should not make session "connectable"    UnicodeString Value;    if (Options->FindSwitch(SESSIONNAME_SWICH, Value))    {      Name = Value;    }    if (Options->FindSwitch(L"privatekey", Value))    {      PublicKeyFile = Value;    }    if (Options->FindSwitch(L"clientcert", Value))    {      TlsCertificateFile = Value;    }    if (Options->FindSwitch(PassphraseOption, Value))    {      Passphrase = Value;    }    if (Options->FindSwitch(L"timeout", Value))    {      Timeout = StrToInt(Value);    }    if (Options->FindSwitch(L"hostkey", Value) ||        Options->FindSwitch(L"certificate", Value))    {      HostKey = Value;      FOverrideCachedHostKey = true;    }    FtpPasvMode = Options->SwitchValue(L"passive", FtpPasvMode);    if (Options->FindSwitch(L"implicit"))    {      bool Enabled = Options->SwitchValue(L"implicit", true);      Ftps = Enabled ? ftpsImplicit : ftpsNone;      if (!PortNumberDefined && Enabled)      {        PortNumber = FtpsImplicitPortNumber;      }    }    // BACKWARD COMPATIBILITY with 5.5.x    if (Options->FindSwitch(L"explicitssl"))    {      bool Enabled = Options->SwitchValue(L"explicitssl", true);      Ftps = Enabled ? ftpsExplicitSsl : ftpsNone;      if (!PortNumberDefined && Enabled)      {        PortNumber = FtpPortNumber;      }    }    if (Options->FindSwitch(L"explicit") ||        // BACKWARD COMPATIBILITY with 5.5.x        Options->FindSwitch(L"explicittls"))    {      UnicodeString SwitchName =        Options->FindSwitch(L"explicit") ? L"explicit" : L"explicittls";      bool Enabled = Options->SwitchValue(SwitchName, true);      Ftps = Enabled ? ftpsExplicitTls : ftpsNone;      if (!PortNumberDefined && Enabled)      {        PortNumber = FtpPortNumber;      }    }    if (Options->FindSwitch(L"rawsettings"))    {      TStrings * RawSettings = NULL;      TOptionsStorage * OptionsStorage = NULL;      try      {        RawSettings = new TStringList();        if (Options->FindSwitch(L"rawsettings", RawSettings))        {          OptionsStorage = new TOptionsStorage(RawSettings, false);          ApplyRawSettings(OptionsStorage);        }      }      __finally      {        delete RawSettings;        delete OptionsStorage;      }    }  }  return true;}//---------------------------------------------------------------------void __fastcall TSessionData::ApplyRawSettings(THierarchicalStorage * Storage){  bool Dummy;  DoLoad(Storage, Dummy);}//---------------------------------------------------------------------void __fastcall TSessionData::ConfigureTunnel(int APortNumber){  FOrigHostName = HostName;  FOrigPortNumber = PortNumber;  FOrigProxyMethod = ProxyMethod;  HostName = L"127.0.0.1";  PortNumber = APortNumber;  // proxy settings is used for tunnel  ProxyMethod = ::pmNone;  FLogicalHostName = FOrigHostName;}//---------------------------------------------------------------------void __fastcall TSessionData::RollbackTunnel(){  HostName = FOrigHostName;  PortNumber = FOrigPortNumber;  ProxyMethod = FOrigProxyMethod;  FLogicalHostName = L"";}//---------------------------------------------------------------------void __fastcall TSessionData::ExpandEnvironmentVariables(){  HostName = HostNameExpanded;  UserName = UserNameExpanded;  PublicKeyFile = ::ExpandEnvironmentVariables(PublicKeyFile);}//---------------------------------------------------------------------void __fastcall TSessionData::ValidatePath(const UnicodeString Path){  // noop}//---------------------------------------------------------------------void __fastcall TSessionData::ValidateName(const UnicodeString Name){  // keep consistent with MakeValidName  if (Name.LastDelimiter(L"/") > 0)  {    throw Exception(FMTLOAD(ITEM_NAME_INVALID, (Name, L"/")));  }}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::MakeValidName(const UnicodeString & Name){  // keep consistent with ValidateName  return ReplaceStr(Name, L"/", L"\\");}//---------------------------------------------------------------------RawByteString __fastcall TSessionData::EncryptPassword(const UnicodeString & Password, UnicodeString Key){  return Configuration->EncryptPassword(Password, Key);}//---------------------------------------------------------------------RawByteString __fastcall TSessionData::StronglyRecryptPassword(const RawByteString & Password, UnicodeString Key){  return Configuration->StronglyRecryptPassword(Password, Key);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::DecryptPassword(const RawByteString & Password, UnicodeString Key){  UnicodeString Result;  try  {    Result = Configuration->DecryptPassword(Password, Key);  }  catch(EAbort &)  {    // silently ignore aborted prompts for master password and return empty password  }  return Result;}//---------------------------------------------------------------------bool __fastcall TSessionData::GetCanLogin(){  return !FHostName.IsEmpty();}//---------------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetSessionKey(){  UnicodeString Result = FORMAT(L"%s@%s", (UserName, HostName));  if (PortNumber != DefaultPort(FSProtocol, Ftps))  {    Result += FORMAT(L":%d", (PortNumber));  }  return Result;}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetInternalStorageKey(){  // This is probably useless remnant of previous use of this method from OpenSessionInPutty  // that needs the method to return something even for ad-hoc sessions  if (Name.IsEmpty())  {    return SessionKey;  }  else  {    return Name;  }}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetStorageKey(){  return SessionName;}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::FormatSiteKey(const UnicodeString & HostName, int PortNumber){  return FORMAT(L"%s:%d", (HostName, PortNumber));}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetSiteKey(){  return FormatSiteKey(HostNameExpanded, PortNumber);}//---------------------------------------------------------------------void __fastcall TSessionData::SetHostName(UnicodeString value){  if (FHostName != value)  {    // HostName is key for password encryption    UnicodeString XPassword = Password;    // This is now hardly used as hostname is parsed directly on login dialog.    // But can be used when importing sites from PuTTY, as it allows same format too.    int P = value.LastDelimiter(L"@");    if (P > 0)    {      UserName = value.SubString(1, P - 1);      value = value.SubString(P + 1, value.Length() - P);    }    FHostName = value;    Modify();    Password = XPassword;    Shred(XPassword);  }}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetHostNameExpanded(){  return ::ExpandEnvironmentVariables(HostName);}//---------------------------------------------------------------------void __fastcall TSessionData::SetPortNumber(int value){  SET_SESSION_PROPERTY(PortNumber);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetShell(UnicodeString value){  SET_SESSION_PROPERTY(Shell);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetSftpServer(UnicodeString value){  SET_SESSION_PROPERTY(SftpServer);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetClearAliases(bool value){  SET_SESSION_PROPERTY(ClearAliases);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetListingCommand(UnicodeString value){  SET_SESSION_PROPERTY(ListingCommand);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetIgnoreLsWarnings(bool value){  SET_SESSION_PROPERTY(IgnoreLsWarnings);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetUnsetNationalVars(bool value){  SET_SESSION_PROPERTY(UnsetNationalVars);}//---------------------------------------------------------------------void __fastcall TSessionData::SetUserName(UnicodeString value){  // Avoid password recryption (what may popup master password prompt)  if (FUserName != value)  {    // UserName is key for password encryption    UnicodeString XPassword = Password;    SET_SESSION_PROPERTY(UserName);    Password = XPassword;    Shred(XPassword);  }}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetUserNameExpanded(){  return ::ExpandEnvironmentVariables(UserName);}//---------------------------------------------------------------------void __fastcall TSessionData::SetPassword(UnicodeString avalue){  RawByteString value = EncryptPassword(avalue, UserName+HostName);  SET_SESSION_PROPERTY(Password);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetPassword() const{  return DecryptPassword(FPassword, UserName+HostName);}//---------------------------------------------------------------------void __fastcall TSessionData::SetPingInterval(int value){  SET_SESSION_PROPERTY(PingInterval);}//---------------------------------------------------------------------void __fastcall TSessionData::SetTryAgent(bool value){  SET_SESSION_PROPERTY(TryAgent);}//---------------------------------------------------------------------void __fastcall TSessionData::SetAgentFwd(bool value){  SET_SESSION_PROPERTY(AgentFwd);}//---------------------------------------------------------------------void __fastcall TSessionData::SetAuthTIS(bool value){  SET_SESSION_PROPERTY(AuthTIS);}//---------------------------------------------------------------------void __fastcall TSessionData::SetAuthKI(bool value){  SET_SESSION_PROPERTY(AuthKI);}//---------------------------------------------------------------------void __fastcall TSessionData::SetAuthKIPassword(bool value){  SET_SESSION_PROPERTY(AuthKIPassword);}//---------------------------------------------------------------------void __fastcall TSessionData::SetAuthGSSAPI(bool value){  SET_SESSION_PROPERTY(AuthGSSAPI);}//---------------------------------------------------------------------void __fastcall TSessionData::SetGSSAPIFwdTGT(bool value){  SET_SESSION_PROPERTY(GSSAPIFwdTGT);}//---------------------------------------------------------------------void __fastcall TSessionData::SetChangeUsername(bool value){  SET_SESSION_PROPERTY(ChangeUsername);}//---------------------------------------------------------------------void __fastcall TSessionData::SetCompression(bool value){  SET_SESSION_PROPERTY(Compression);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSshProt(TSshProt value){  SET_SESSION_PROPERTY(SshProt);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSsh2DES(bool value){  SET_SESSION_PROPERTY(Ssh2DES);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSshNoUserAuth(bool value){  SET_SESSION_PROPERTY(SshNoUserAuth);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetSshProtStr(){  return SshProtList[FSshProt];}//---------------------------------------------------------------------bool __fastcall TSessionData::GetUsesSsh(){  return IsSshProtocol(FSProtocol);}//---------------------------------------------------------------------void __fastcall TSessionData::SetCipher(int Index, TCipher value){  DebugAssert(Index >= 0 && Index < CIPHER_COUNT);  SET_SESSION_PROPERTY(Ciphers[Index]);}//---------------------------------------------------------------------TCipher __fastcall TSessionData::GetCipher(int Index) const{  DebugAssert(Index >= 0 && Index < CIPHER_COUNT);  return FCiphers[Index];}//---------------------------------------------------------------------template<class AlgoT>void __fastcall TSessionData::SetAlgoList(AlgoT * List, const AlgoT * DefaultList, const UnicodeString * Names,  int Count, AlgoT WarnAlgo, UnicodeString value){  std::vector<bool> Used(Count); // initialized to false  std::vector<AlgoT> NewList(Count);  const AlgoT * WarnPtr = std::find(DefaultList, DefaultList + Count, WarnAlgo);  DebugAssert(WarnPtr != NULL);  int WarnDefaultIndex = (WarnPtr - DefaultList);  int Index = 0;  while (!value.IsEmpty())  {    UnicodeString AlgoStr = CutToChar(value, L',', true);    for (int Algo = 0; Algo < Count; Algo++)    {      if (!AlgoStr.CompareIC(Names[Algo]) &&          !Used[Algo] && DebugAlwaysTrue(Index < Count))      {        NewList[Index] = (AlgoT)Algo;        Used[Algo] = true;        Index++;        break;      }    }  }  if (!Used[WarnAlgo] && DebugAlwaysTrue(Index < Count))  {    NewList[Index] = WarnAlgo;    Used[WarnAlgo] = true;    Index++;  }  int WarnIndex = std::find(NewList.begin(), NewList.end(), WarnAlgo) - NewList.begin();  bool Priority = true;  for (int DefaultIndex = 0; (DefaultIndex < Count); DefaultIndex++)  {    AlgoT DefaultAlgo = DefaultList[DefaultIndex];    if (!Used[DefaultAlgo] && DebugAlwaysTrue(Index < Count))    {      int TargetIndex;      // Unused algs that are prioritized in the default list,      // should be merged before the existing custom list      if (Priority)      {        TargetIndex = DefaultIndex;      }      else      {        if (DefaultIndex < WarnDefaultIndex)        {          TargetIndex = WarnIndex;        }        else        {          TargetIndex = Index;        }      }      NewList.insert(NewList.begin() + TargetIndex, DefaultAlgo);      DebugAssert(NewList.back() == AlgoT());      NewList.pop_back();      if (TargetIndex <= WarnIndex)      {        WarnIndex++;      }      Index++;    }    else    {      Priority = false;    }  }  if (!std::equal(NewList.begin(), NewList.end(), List))  {    std::copy(NewList.begin(), NewList.end(), List);    Modify();  }}//---------------------------------------------------------------------void __fastcall TSessionData::SetCipherList(UnicodeString value){  SetAlgoList(FCiphers, DefaultCipherList, CipherNames, CIPHER_COUNT, cipWarn, value);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetCipherList() const{  UnicodeString Result;  for (int Index = 0; Index < CIPHER_COUNT; Index++)  {    Result += UnicodeString(Index ? L"," : L"") + CipherNames[Cipher[Index]];  }  return Result;}//---------------------------------------------------------------------void __fastcall TSessionData::SetKex(int Index, TKex value){  DebugAssert(Index >= 0 && Index < KEX_COUNT);  SET_SESSION_PROPERTY(Kex[Index]);}//---------------------------------------------------------------------TKex __fastcall TSessionData::GetKex(int Index) const{  DebugAssert(Index >= 0 && Index < KEX_COUNT);  return FKex[Index];}//---------------------------------------------------------------------void __fastcall TSessionData::SetKexList(UnicodeString value){  SetAlgoList(FKex, DefaultKexList, KexNames, KEX_COUNT, kexWarn, value);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetKexList() const{  UnicodeString Result;  for (int Index = 0; Index < KEX_COUNT; Index++)  {    Result += UnicodeString(Index ? L"," : L"") + KexNames[Kex[Index]];  }  return Result;}//---------------------------------------------------------------------void __fastcall TSessionData::SetPublicKeyFile(UnicodeString value){  if (FPublicKeyFile != value)  {    // PublicKeyFile is key for Passphrase encryption    UnicodeString XPassphrase = Passphrase;    // StripPathQuotes should not be needed as we do not feed quotes anymore    FPublicKeyFile = StripPathQuotes(value);    Modify();    Passphrase = XPassphrase;    Shred(XPassphrase);  }}//---------------------------------------------------------------------void __fastcall TSessionData::SetPassphrase(UnicodeString avalue){  RawByteString value = EncryptPassword(avalue, PublicKeyFile);  SET_SESSION_PROPERTY(Passphrase);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetPassphrase() const{  return DecryptPassword(FPassphrase, PublicKeyFile);}//---------------------------------------------------------------------void __fastcall TSessionData::SetReturnVar(UnicodeString value){  SET_SESSION_PROPERTY(ReturnVar);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetLookupUserGroups(TAutoSwitch value){  SET_SESSION_PROPERTY(LookupUserGroups);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetEOLType(TEOLType value){  SET_SESSION_PROPERTY(EOLType);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetTrimVMSVersions(bool value){  SET_SESSION_PROPERTY(TrimVMSVersions);}//---------------------------------------------------------------------------TDateTime __fastcall TSessionData::GetTimeoutDT(){  return SecToDateTime(Timeout);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetTimeout(int value){  SET_SESSION_PROPERTY(Timeout);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetFSProtocol(TFSProtocol value){  SET_SESSION_PROPERTY(FSProtocol);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetFSProtocolStr(){  DebugAssert(FSProtocol >= 0 && FSProtocol < FSPROTOCOL_COUNT);  return FSProtocolNames[FSProtocol];}//---------------------------------------------------------------------------void __fastcall TSessionData::SetDetectReturnVar(bool value){  if (value != DetectReturnVar)  {    ReturnVar = value ? L"" : L"$?";  }}//---------------------------------------------------------------------------bool __fastcall TSessionData::GetDetectReturnVar(){  return ReturnVar.IsEmpty();}//---------------------------------------------------------------------------void __fastcall TSessionData::SetDefaultShell(bool value){  if (value != DefaultShell)  {    Shell = value ? L"" : L"/bin/bash";  }}//---------------------------------------------------------------------------bool __fastcall TSessionData::GetDefaultShell(){  return Shell.IsEmpty();}//---------------------------------------------------------------------------void __fastcall TSessionData::SetPuttyProtocol(UnicodeString value){  SET_SESSION_PROPERTY(PuttyProtocol);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetNormalizedPuttyProtocol() const{  return DefaultStr(PuttyProtocol, PuttySshProtocol);}//---------------------------------------------------------------------void __fastcall TSessionData::SetPingIntervalDT(TDateTime value){  unsigned short hour, min, sec, msec;  value.DecodeTime(&hour, &min, &sec, &msec);  PingInterval = ((int)hour)*SecsPerHour + ((int)min)*SecsPerMin + sec;}//---------------------------------------------------------------------------TDateTime __fastcall TSessionData::GetPingIntervalDT(){  return SecToDateTime(PingInterval);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetPingType(TPingType value){  SET_SESSION_PROPERTY(PingType);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetAddressFamily(TAddressFamily value){  SET_SESSION_PROPERTY(AddressFamily);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetRekeyData(UnicodeString value){  SET_SESSION_PROPERTY(RekeyData);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetRekeyTime(unsigned int value){  SET_SESSION_PROPERTY(RekeyTime);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetDefaultSessionName(){  if (!HostName.IsEmpty() && !UserName.IsEmpty())  {    // If we ever choose to include port number,    // we have to escape IPv6 literals in HostName    return FORMAT(L"%s@%s", (UserName, HostName));  }  else if (!HostName.IsEmpty())  {    return HostName;  }  else  {    return L"session";  }}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetNameWithoutHiddenPrefix(){  UnicodeString Result = Name;  if (Hidden)  {    Result = Result.SubString(TNamedObjectList::HiddenPrefix.Length() + 1, Result.Length() - TNamedObjectList::HiddenPrefix.Length());  }  return Result;}//---------------------------------------------------------------------bool __fastcall TSessionData::HasSessionName(){  return (!GetNameWithoutHiddenPrefix().IsEmpty() && (Name != DefaultName) && !IsWorkspace);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetSessionName(){  UnicodeString Result;  if (HasSessionName())  {    Result = GetNameWithoutHiddenPrefix();  }  else  {    Result = DefaultSessionName;  }  return Result;}//---------------------------------------------------------------------bool __fastcall TSessionData::IsSecure(){  bool Result;  switch (FSProtocol)  {    case fsSCPonly:    case fsSFTP:    case fsSFTPonly:      Result = true;      break;    case fsFTP:    case fsWebDAV:      Result = (Ftps != ftpsNone);      break;    default:      DebugFail();      break;  }  return Result;}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetProtocolUrl(){  UnicodeString Url;  switch (FSProtocol)  {    case fsSCPonly:      Url = ScpProtocol;      break;    default:      DebugFail();      // fallback    case fsSFTP:    case fsSFTPonly:      Url = SftpProtocol;      break;    case fsFTP:      if (Ftps == ftpsImplicit)      {        Url = FtpsProtocol;      }      else if ((Ftps == ftpsExplicitTls) || (Ftps == ftpsExplicitSsl))      {        Url = FtpesProtocol;      }      else      {        Url = FtpProtocol;      }      break;    case fsWebDAV:      if (Ftps == ftpsImplicit)      {        Url = WebDAVSProtocol;      }      else      {        Url = WebDAVProtocol;      }      break;  }  Url += ProtocolSeparator;  return Url;}//---------------------------------------------------------------------static bool __fastcall IsIPv6Literal(const UnicodeString & HostName){  bool Result = (HostName.Pos(L":") > 0);  if (Result)  {    for (int Index = 1; Result && (Index <= HostName.Length()); Index++)    {      wchar_t C = HostName[Index];      Result = IsHex(C) || (C == L':');    }  }  return Result;}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GenerateSessionUrl(unsigned int Flags){  UnicodeString Url;  if (FLAGSET(Flags, sufSpecific))  {    Url += WinSCPProtocolPrefix;  }  Url += GetProtocolUrl();  if (FLAGSET(Flags, sufUserName) && !UserNameExpanded.IsEmpty())  {    Url += EncodeUrlString(UserNameExpanded);    if (FLAGSET(Flags, sufPassword) && !Password.IsEmpty())    {      Url += L":" + EncodeUrlString(Password);    }    if (FLAGSET(Flags, sufHostKey) && !HostKey.IsEmpty())    {      Url +=        UnicodeString(UrlParamSeparator) + UrlHostKeyParamName +        UnicodeString(UrlParamValueSeparator) + NormalizeFingerprint(HostKey);    }    Url += L"@";  }  DebugAssert(!HostNameExpanded.IsEmpty());  if (IsIPv6Literal(HostNameExpanded))  {    Url += L"[" + HostNameExpanded + L"]";  }  else  {    Url += EncodeUrlString(HostNameExpanded);  }  if (PortNumber != DefaultPort(FSProtocol, Ftps))  {    Url += L":" + IntToStr(PortNumber);  }  Url += L"/";  return Url;}//---------------------------------------------------------------------UnicodeString ScriptCommandOpenLink = ScriptCommandLink(L"open");//---------------------------------------------------------------------void __fastcall TSessionData::AddSwitch(  UnicodeString & Result, const UnicodeString & Name, bool Rtf){  Result += RtfSwitch(Name, ScriptCommandOpenLink, Rtf);}//---------------------------------------------------------------------void __fastcall TSessionData::AddSwitch(  UnicodeString & Result, const UnicodeString & Name, const UnicodeString & Value, bool Rtf){  Result += RtfSwitch(Name, ScriptCommandOpenLink, Value, Rtf);}//---------------------------------------------------------------------void __fastcall TSessionData::AddSwitch(  UnicodeString & Result, const UnicodeString & Name, int Value, bool Rtf){  Result += RtfSwitch(Name, ScriptCommandOpenLink, Value, Rtf);}//---------------------------------------------------------------------void __fastcall TSessionData::LookupLastFingerprint(){  UnicodeString FingerprintType;  if (IsSshProtocol(FSProtocol))  {    FingerprintType = SshFingerprintType;  }  else if (Ftps != ftpsNone)  {    FingerprintType = TlsFingerprintType;  }  if (!FingerprintType.IsEmpty())  {    HostKey = Configuration->LastFingerprint(SiteKey, FingerprintType);  }}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GenerateOpenCommandArgs(bool Rtf){  std::unique_ptr<TSessionData> FactoryDefaults(new TSessionData(L""));  std::unique_ptr<TSessionData> SessionData(new TSessionData(L""));  SessionData->Assign(this);  UnicodeString Result = SessionData->GenerateSessionUrl(sufOpen);  // Before we reset the FSProtocol  bool AUsesSsh = SessionData->UsesSsh;  // SFTP-only is not reflected by the protocol prefix, we have to use rawsettings for that  if (SessionData->FSProtocol != fsSFTPonly)  {    SessionData->FSProtocol = FactoryDefaults->FSProtocol;  }  SessionData->HostName = FactoryDefaults->HostName;  SessionData->PortNumber = FactoryDefaults->PortNumber;  SessionData->UserName = FactoryDefaults->UserName;  SessionData->Password = FactoryDefaults->Password;  SessionData->CopyNonCoreData(FactoryDefaults.get());  SessionData->Ftps = FactoryDefaults->Ftps;  if (SessionData->HostKey != FactoryDefaults->HostKey)  {    UnicodeString SwitchName = AUsesSsh ? L"hostkey" : L"certificate";    AddSwitch(Result, SwitchName, SessionData->HostKey, Rtf);    SessionData->HostKey = FactoryDefaults->HostKey;  }  if (SessionData->PublicKeyFile != FactoryDefaults->PublicKeyFile)  {    AddSwitch(Result, L"privatekey", SessionData->PublicKeyFile, Rtf);    SessionData->PublicKeyFile = FactoryDefaults->PublicKeyFile;  }  if (SessionData->TlsCertificateFile != FactoryDefaults->TlsCertificateFile)  {    AddSwitch(Result, L"clientcert", SessionData->TlsCertificateFile, Rtf);    SessionData->TlsCertificateFile = FactoryDefaults->TlsCertificateFile;  }  if (SessionData->Passphrase != FactoryDefaults->Passphrase)  {    AddSwitch(Result, PassphraseOption, SessionData->Passphrase, Rtf);    SessionData->Passphrase = FactoryDefaults->Passphrase;  }  if (SessionData->FtpPasvMode != FactoryDefaults->FtpPasvMode)  {    AddSwitch(Result, L"passive", SessionData->FtpPasvMode ? 1 : 0, Rtf);    SessionData->FtpPasvMode = FactoryDefaults->FtpPasvMode;  }  if (SessionData->Timeout != FactoryDefaults->Timeout)  {    AddSwitch(Result, L"timeout", SessionData->Timeout, Rtf);    SessionData->Timeout = FactoryDefaults->Timeout;  }  std::unique_ptr<TStrings> RawSettings(SessionData->SaveToOptions(FactoryDefaults.get()));  if (RawSettings->Count > 0)  {    AddSwitch(Result, L"rawsettings", Rtf);    for (int Index = 0; Index < RawSettings->Count; Index++)    {      UnicodeString Name = RawSettings->Names[Index];      UnicodeString Value = RawSettings->ValueFromIndex[Index];      // Do not quote if it is all-numeric      if (IntToStr(StrToIntDef(Value, -1)) != Value)      {        Value = FORMAT(L"\"%s\"", (EscapeParam(Value)));      }      Result += FORMAT(L" %s=%s", (Name, Value));    }  }  return Result;}//---------------------------------------------------------------------UnicodeString SessionOptionsClassName(L"SessionOptions");//---------------------------------------------------------------------void __fastcall TSessionData::AddAssemblyProperty(  UnicodeString & Result, TAssemblyLanguage Language,  const UnicodeString & Name, const UnicodeString & Type,  const UnicodeString & Member){  Result += AssemblyProperty(Language, SessionOptionsClassName, Name, Type, Member, false);}//---------------------------------------------------------------------void __fastcall TSessionData::AddAssemblyProperty(  UnicodeString & Result, TAssemblyLanguage Language,  const UnicodeString & Name, const UnicodeString & Value){  Result += AssemblyProperty(Language, SessionOptionsClassName, Name, Value, false);}//---------------------------------------------------------------------void __fastcall TSessionData::AddAssemblyProperty(  UnicodeString & Result, TAssemblyLanguage Language,  const UnicodeString & Name, int Value){  Result += AssemblyProperty(Language, SessionOptionsClassName, Name, Value, false);}//---------------------------------------------------------------------void __fastcall TSessionData::AddAssemblyProperty(  UnicodeString & Result, TAssemblyLanguage Language,  const UnicodeString & Name, bool Value){  Result += AssemblyProperty(Language, SessionOptionsClassName, Name, Value, false);}//---------------------------------------------------------------------void __fastcall TSessionData::GenerateAssemblyCode(  TAssemblyLanguage Language, UnicodeString & Head, UnicodeString & Tail, int & Indent){  std::unique_ptr<TSessionData> FactoryDefaults(new TSessionData(L""));  std::unique_ptr<TSessionData> SessionData(Clone());  switch (Language)  {    case alCSharp:    case alVBNET:      // noop      break;    case alPowerShell:      Head +=        AssemblyCommentLine(Language, LoadStr(CODE_PS_ADD_TYPE)) +        RtfKeyword(L"Add-Type") + RtfText(" -Path ") + AssemblyString(Language, "WinSCPnet.dll") + RtfPara +        RtfPara;      break;    default:      DebugFail();      break;  }  Head +=    AssemblyCommentLine(Language, LoadStr(CODE_SESSION_OPTIONS)) +    AssemblyNewClassInstanceStart(Language, SessionOptionsClassName, false);  UnicodeString ProtocolMember;  switch (SessionData->FSProtocol)  {    case fsSCPonly:      ProtocolMember = "Scp";      break;    default:      DebugFail();      // fallback    case fsSFTP:    case fsSFTPonly:      ProtocolMember = "Sftp";      break;    case fsFTP:      ProtocolMember = "Ftp";      break;    case fsWebDAV:      ProtocolMember = "Webdav";      break;  }  // Before we reset the FSProtocol  bool AUsesSsh = SessionData->UsesSsh;  // Protocol is set unconditionally, we want even the default SFTP  AddAssemblyProperty(Head, Language, L"Protocol", L"Protocol", ProtocolMember);  // SFTP-only is not reflected by the protocol prefix, we have to use rawsettings for that  if (SessionData->FSProtocol != fsSFTPonly)  {    SessionData->FSProtocol = FactoryDefaults->FSProtocol;  }  if (SessionData->HostName != FactoryDefaults->HostName)  {    AddAssemblyProperty(Head, Language, L"HostName", HostName);    SessionData->HostName = FactoryDefaults->HostName;  }  if (SessionData->PortNumber != FactoryDefaults->PortNumber)  {    AddAssemblyProperty(Head, Language, L"PortNumber", PortNumber);    SessionData->PortNumber = FactoryDefaults->PortNumber;  }  if (SessionData->UserName != FactoryDefaults->UserName)  {    AddAssemblyProperty(Head, Language, L"UserName", UserName);    SessionData->UserName = FactoryDefaults->UserName;  }  if (SessionData->Password != FactoryDefaults->Password)  {    AddAssemblyProperty(Head, Language, L"Password", Password);    SessionData->Password = FactoryDefaults->Password;  }  SessionData->CopyNonCoreData(FactoryDefaults.get());  if (SessionData->Ftps != FactoryDefaults->Ftps)  {    // SessionData->FSProtocol is reset already    switch (FSProtocol)    {      case fsFTP:        {          UnicodeString FtpSecureMember;          switch (SessionData->Ftps)          {            case ftpsNone:              // noop              break;            case ftpsImplicit:              FtpSecureMember = L"Implicit";              break;            case ftpsExplicitTls:            case ftpsExplicitSsl:              FtpSecureMember = L"Explicit";              break;            default:              DebugFail();              break;          }          AddAssemblyProperty(Head, Language, L"FtpSecure", L"FtpSecure", FtpSecureMember);        }        break;      case fsWebDAV:        AddAssemblyProperty(Head, Language, L"WebdavSecure", (SessionData->Ftps != ftpsNone));        break;      default:        DebugFail();        break;    }    SessionData->Ftps = FactoryDefaults->Ftps;  }  if (SessionData->HostKey != FactoryDefaults->HostKey)  {    UnicodeString PropertyName = AUsesSsh ? L"SshHostKeyFingerprint" : L"TlsHostCertificateFingerprint";    AddAssemblyProperty(Head, Language, PropertyName, SessionData->HostKey);    SessionData->HostKey = FactoryDefaults->HostKey;  }  if (SessionData->PublicKeyFile != FactoryDefaults->PublicKeyFile)  {    AddAssemblyProperty(Head, Language, L"SshPrivateKeyPath", SessionData->PublicKeyFile);    SessionData->PublicKeyFile = FactoryDefaults->PublicKeyFile;  }  if (SessionData->TlsCertificateFile != FactoryDefaults->TlsCertificateFile)  {    AddAssemblyProperty(Head, Language, L"TlsClientCertificatePath", SessionData->TlsCertificateFile);    SessionData->TlsCertificateFile = FactoryDefaults->TlsCertificateFile;  }  if (SessionData->Passphrase != FactoryDefaults->Passphrase)  {    AddAssemblyProperty(Head, Language, L"PrivateKeyPassphrase", SessionData->Passphrase);    SessionData->Passphrase = FactoryDefaults->Passphrase;  }  if (SessionData->FtpPasvMode != FactoryDefaults->FtpPasvMode)  {    AddAssemblyProperty(Head, Language, L"FtpMode", L"FtpMode", (SessionData->FtpPasvMode ? L"Passive" : L"Active"));    SessionData->FtpPasvMode = FactoryDefaults->FtpPasvMode;  }  if (SessionData->Timeout != FactoryDefaults->Timeout)  {    AddAssemblyProperty(Head, Language, L"TimeoutInMilliseconds", SessionData->Timeout * 1000);    SessionData->Timeout = FactoryDefaults->Timeout;  }  switch (Language)  {    case alCSharp:    case alPowerShell:      Head += AssemblyNewClassInstanceEnd(Language, false);      break;    case alVBNET:      // noop      // Ending With only after AddRawSettings      break;  }  std::unique_ptr<TStrings> RawSettings(SessionData->SaveToOptions(FactoryDefaults.get()));  UnicodeString SessionOptionsVariableName = AssemblyVariableName(Language, SessionOptionsClassName);  if (RawSettings->Count > 0)  {    Head += RtfPara;    for (int Index = 0; Index < RawSettings->Count; Index++)    {      UnicodeString Name = RawSettings->Names[Index];      UnicodeString Value = RawSettings->ValueFromIndex[Index];      UnicodeString AddRawSettingsMethod =        RtfLibraryMethod(SessionOptionsClassName, L"AddRawSettings", false) +        FORMAT(L"(%s, %s)", (AssemblyString(Language, Name), AssemblyString(Language, Value)));      switch (Language)      {        case alCSharp:          Head += RtfText(SessionOptionsVariableName + L".") + AddRawSettingsMethod + RtfText(L";") + RtfPara;          break;        case alVBNET:          Head += RtfText(L"    .") + AddRawSettingsMethod + RtfPara;          break;        case alPowerShell:          Head += RtfText(SessionOptionsVariableName + L".") + AddRawSettingsMethod + RtfPara;          break;      }    }  }  UnicodeString Indentation = L"    ";  UnicodeString SessionVariableName = AssemblyVariableName(Language, SessionClassName);  UnicodeString RtfSessionClass = RtfLibraryClass(SessionClassName);  UnicodeString RtfSessionOpenMethod = RtfLibraryMethod(SessionClassName, L"Open", false);  UnicodeString NewSessionInstance = AssemblyNewClassInstance(Language, SessionClassName, false);  UnicodeString OpenCall =    Indentation + AssemblyCommentLine(Language, LoadStr(CODE_CONNECT)) +    Indentation + RtfText(SessionVariableName + L".") + RtfSessionOpenMethod + RtfText(L"(" + SessionOptionsVariableName + L")") +      AssemblyStatementSeparator(Language) + RtfPara;  switch (Language)  {    case alCSharp:      Head +=        RtfPara +        RtfKeyword(L"using") + RtfText(" (") + NewSessionInstance + RtfText(L"())") + RtfPara +        RtfText(L"{") + RtfPara +        OpenCall;      Tail =        RtfText(L"}") + RtfPara;      break;    case alVBNET:      Head +=        AssemblyNewClassInstanceEnd(Language, false) +        RtfPara +        RtfKeyword(L"Using") + RtfText(L" ") + NewSessionInstance + RtfPara +        OpenCall;      Tail =        RtfKeyword(L"End Using");      break;    case alPowerShell:      Head +=        RtfPara +        NewSessionInstance + RtfPara +        RtfPara +        RtfKeyword(L"try") + RtfPara +        RtfText(L"{") + RtfPara +        OpenCall;      Tail =        RtfText(L"}") + RtfPara +        RtfKeyword(L"finally") + RtfPara +        RtfText(L"{") + RtfPara +        RtfText(Indentation + SessionVariableName + L".") +          RtfLibraryMethod(SessionClassName, L"Dispose", false) + RtfText(L"()") + RtfPara +        RtfText(L"}") + RtfPara;      break;  }  Head += RtfPara;  Indent = 4; // the same for all languages so far}//---------------------------------------------------------------------void __fastcall TSessionData::SetTimeDifference(TDateTime value){  SET_SESSION_PROPERTY(TimeDifference);}//---------------------------------------------------------------------void __fastcall TSessionData::SetTimeDifferenceAuto(bool value){  SET_SESSION_PROPERTY(TimeDifferenceAuto);}//---------------------------------------------------------------------void __fastcall TSessionData::SetLocalDirectory(UnicodeString value){  SET_SESSION_PROPERTY(LocalDirectory);}//---------------------------------------------------------------------void __fastcall TSessionData::SetRemoteDirectory(UnicodeString value){  SET_SESSION_PROPERTY(RemoteDirectory);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSynchronizeBrowsing(bool value){  SET_SESSION_PROPERTY(SynchronizeBrowsing);}//---------------------------------------------------------------------void __fastcall TSessionData::SetUpdateDirectories(bool value){  SET_SESSION_PROPERTY(UpdateDirectories);}//---------------------------------------------------------------------void __fastcall TSessionData::SetCacheDirectories(bool value){  SET_SESSION_PROPERTY(CacheDirectories);}//---------------------------------------------------------------------void __fastcall TSessionData::SetCacheDirectoryChanges(bool value){  SET_SESSION_PROPERTY(CacheDirectoryChanges);}//---------------------------------------------------------------------void __fastcall TSessionData::SetPreserveDirectoryChanges(bool value){  SET_SESSION_PROPERTY(PreserveDirectoryChanges);}//---------------------------------------------------------------------void __fastcall TSessionData::SetResolveSymlinks(bool value){  SET_SESSION_PROPERTY(ResolveSymlinks);}//---------------------------------------------------------------------void __fastcall TSessionData::SetFollowDirectorySymlinks(bool value){  SET_SESSION_PROPERTY(FollowDirectorySymlinks);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetDSTMode(TDSTMode value){  SET_SESSION_PROPERTY(DSTMode);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetDeleteToRecycleBin(bool value){  SET_SESSION_PROPERTY(DeleteToRecycleBin);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetOverwrittenToRecycleBin(bool value){  SET_SESSION_PROPERTY(OverwrittenToRecycleBin);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetRecycleBinPath(UnicodeString value){  SET_SESSION_PROPERTY(RecycleBinPath);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetPostLoginCommands(UnicodeString value){  SET_SESSION_PROPERTY(PostLoginCommands);}//---------------------------------------------------------------------void __fastcall TSessionData::SetLockInHome(bool value){  SET_SESSION_PROPERTY(LockInHome);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSpecial(bool value){  SET_SESSION_PROPERTY(Special);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetScp1Compatibility(bool value){  SET_SESSION_PROPERTY(Scp1Compatibility);}//---------------------------------------------------------------------void __fastcall TSessionData::SetTcpNoDelay(bool value){  SET_SESSION_PROPERTY(TcpNoDelay);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSendBuf(int value){  SET_SESSION_PROPERTY(SendBuf);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSshSimple(bool value){  SET_SESSION_PROPERTY(SshSimple);}//---------------------------------------------------------------------void __fastcall TSessionData::SetProxyMethod(TProxyMethod value){  SET_SESSION_PROPERTY(ProxyMethod);}//---------------------------------------------------------------------void __fastcall TSessionData::SetProxyHost(UnicodeString value){  SET_SESSION_PROPERTY(ProxyHost);}//---------------------------------------------------------------------void __fastcall TSessionData::SetProxyPort(int value){  SET_SESSION_PROPERTY(ProxyPort);}//---------------------------------------------------------------------void __fastcall TSessionData::SetProxyUsername(UnicodeString value){  SET_SESSION_PROPERTY(ProxyUsername);}//---------------------------------------------------------------------void __fastcall TSessionData::SetProxyPassword(UnicodeString avalue){  RawByteString value = EncryptPassword(avalue, ProxyUsername+ProxyHost);  SET_SESSION_PROPERTY(ProxyPassword);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetProxyPassword() const{  return DecryptPassword(FProxyPassword, ProxyUsername+ProxyHost);}//---------------------------------------------------------------------void __fastcall TSessionData::SetProxyTelnetCommand(UnicodeString value){  SET_SESSION_PROPERTY(ProxyTelnetCommand);}//---------------------------------------------------------------------void __fastcall TSessionData::SetProxyLocalCommand(UnicodeString value){  SET_SESSION_PROPERTY(ProxyLocalCommand);}//---------------------------------------------------------------------void __fastcall TSessionData::SetProxyDNS(TAutoSwitch value){  SET_SESSION_PROPERTY(ProxyDNS);}//---------------------------------------------------------------------void __fastcall TSessionData::SetProxyLocalhost(bool value){  SET_SESSION_PROPERTY(ProxyLocalhost);}//---------------------------------------------------------------------void __fastcall TSessionData::SetFtpProxyLogonType(int value){  SET_SESSION_PROPERTY(FtpProxyLogonType);}//---------------------------------------------------------------------void __fastcall TSessionData::SetBug(TSshBug Bug, TAutoSwitch value){  DebugAssert(Bug >= 0 && static_cast<unsigned int>(Bug) < LENOF(FBugs));  SET_SESSION_PROPERTY(Bugs[Bug]);}//---------------------------------------------------------------------TAutoSwitch __fastcall TSessionData::GetBug(TSshBug Bug) const{  DebugAssert(Bug >= 0 && static_cast<unsigned int>(Bug) < LENOF(FBugs));  return FBugs[Bug];}//---------------------------------------------------------------------void __fastcall TSessionData::SetCustomParam1(UnicodeString value){  SET_SESSION_PROPERTY(CustomParam1);}//---------------------------------------------------------------------void __fastcall TSessionData::SetCustomParam2(UnicodeString value){  SET_SESSION_PROPERTY(CustomParam2);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSFTPDownloadQueue(int value){  SET_SESSION_PROPERTY(SFTPDownloadQueue);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSFTPUploadQueue(int value){  SET_SESSION_PROPERTY(SFTPUploadQueue);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSFTPListingQueue(int value){  SET_SESSION_PROPERTY(SFTPListingQueue);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSFTPMaxVersion(int value){  SET_SESSION_PROPERTY(SFTPMaxVersion);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSFTPMaxPacketSize(unsigned long value){  SET_SESSION_PROPERTY(SFTPMaxPacketSize);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSFTPBug(TSftpBug Bug, TAutoSwitch value){  DebugAssert(Bug >= 0 && static_cast<unsigned int>(Bug) < LENOF(FSFTPBugs));  SET_SESSION_PROPERTY(SFTPBugs[Bug]);}//---------------------------------------------------------------------TAutoSwitch __fastcall TSessionData::GetSFTPBug(TSftpBug Bug) const{  DebugAssert(Bug >= 0 && static_cast<unsigned int>(Bug) < LENOF(FSFTPBugs));  return FSFTPBugs[Bug];}//---------------------------------------------------------------------void __fastcall TSessionData::SetSCPLsFullTime(TAutoSwitch value){  SET_SESSION_PROPERTY(SCPLsFullTime);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetColor(int value){  SET_SESSION_PROPERTY(Color);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetTunnel(bool value){  SET_SESSION_PROPERTY(Tunnel);}//---------------------------------------------------------------------void __fastcall TSessionData::SetTunnelHostName(UnicodeString value){  if (FTunnelHostName != value)  {    // HostName is key for password encryption    UnicodeString XTunnelPassword = TunnelPassword;    int P = value.LastDelimiter(L"@");    if (P > 0)    {      TunnelUserName = value.SubString(1, P - 1);      value = value.SubString(P + 1, value.Length() - P);    }    FTunnelHostName = value;    Modify();    TunnelPassword = XTunnelPassword;    Shred(XTunnelPassword);  }}//---------------------------------------------------------------------void __fastcall TSessionData::SetTunnelPortNumber(int value){  SET_SESSION_PROPERTY(TunnelPortNumber);}//---------------------------------------------------------------------void __fastcall TSessionData::SetTunnelUserName(UnicodeString value){  // Avoid password recryption (what may popup master password prompt)  if (FTunnelUserName != value)  {    // TunnelUserName is key for password encryption    UnicodeString XTunnelPassword = TunnelPassword;    SET_SESSION_PROPERTY(TunnelUserName);    TunnelPassword = XTunnelPassword;    Shred(XTunnelPassword);  }}//---------------------------------------------------------------------void __fastcall TSessionData::SetTunnelPassword(UnicodeString avalue){  RawByteString value = EncryptPassword(avalue, TunnelUserName+TunnelHostName);  SET_SESSION_PROPERTY(TunnelPassword);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetTunnelPassword() const{  return DecryptPassword(FTunnelPassword, TunnelUserName+TunnelHostName);}//---------------------------------------------------------------------void __fastcall TSessionData::SetTunnelPublicKeyFile(UnicodeString value){  if (FTunnelPublicKeyFile != value)  {    // StripPathQuotes should not be needed as we do not feed quotes anymore    FTunnelPublicKeyFile = StripPathQuotes(value);    Modify();  }}//---------------------------------------------------------------------void __fastcall TSessionData::SetTunnelLocalPortNumber(int value){  SET_SESSION_PROPERTY(TunnelLocalPortNumber);}//---------------------------------------------------------------------bool __fastcall TSessionData::GetTunnelAutoassignLocalPortNumber(){  return (FTunnelLocalPortNumber <= 0);}//---------------------------------------------------------------------void __fastcall TSessionData::SetTunnelPortFwd(UnicodeString value){  SET_SESSION_PROPERTY(TunnelPortFwd);}//---------------------------------------------------------------------void __fastcall TSessionData::SetTunnelHostKey(UnicodeString value){  SET_SESSION_PROPERTY(TunnelHostKey);}//---------------------------------------------------------------------void __fastcall TSessionData::SetFtpPasvMode(bool value){  SET_SESSION_PROPERTY(FtpPasvMode);}//---------------------------------------------------------------------void __fastcall TSessionData::SetFtpForcePasvIp(TAutoSwitch value){  SET_SESSION_PROPERTY(FtpForcePasvIp);}//---------------------------------------------------------------------void __fastcall TSessionData::SetFtpUseMlsd(TAutoSwitch value){  SET_SESSION_PROPERTY(FtpUseMlsd);}//---------------------------------------------------------------------void __fastcall TSessionData::SetFtpAccount(UnicodeString value){  SET_SESSION_PROPERTY(FtpAccount);}//---------------------------------------------------------------------void __fastcall TSessionData::SetFtpPingInterval(int value){  SET_SESSION_PROPERTY(FtpPingInterval);}//---------------------------------------------------------------------------TDateTime __fastcall TSessionData::GetFtpPingIntervalDT(){  return SecToDateTime(FtpPingInterval);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetFtpPingType(TPingType value){  SET_SESSION_PROPERTY(FtpPingType);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetFtpTransferActiveImmediately(TAutoSwitch value){  SET_SESSION_PROPERTY(FtpTransferActiveImmediately);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetFtps(TFtps value){  SET_SESSION_PROPERTY(Ftps);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetMinTlsVersion(TTlsVersion value){  SET_SESSION_PROPERTY(MinTlsVersion);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetMaxTlsVersion(TTlsVersion value){  SET_SESSION_PROPERTY(MaxTlsVersion);}//---------------------------------------------------------------------void __fastcall TSessionData::SetFtpListAll(TAutoSwitch value){  SET_SESSION_PROPERTY(FtpListAll);}//---------------------------------------------------------------------void __fastcall TSessionData::SetFtpHost(TAutoSwitch value){  SET_SESSION_PROPERTY(FtpHost);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSslSessionReuse(bool value){  SET_SESSION_PROPERTY(SslSessionReuse);}//---------------------------------------------------------------------void __fastcall TSessionData::SetTlsCertificateFile(UnicodeString value){  SET_SESSION_PROPERTY(TlsCertificateFile);}//---------------------------------------------------------------------void __fastcall TSessionData::SetNotUtf(TAutoSwitch value){  SET_SESSION_PROPERTY(NotUtf);}//---------------------------------------------------------------------void __fastcall TSessionData::SetIsWorkspace(bool value){  SET_SESSION_PROPERTY(IsWorkspace);}//---------------------------------------------------------------------void __fastcall TSessionData::SetLink(UnicodeString value){  SET_SESSION_PROPERTY(Link);}//---------------------------------------------------------------------void __fastcall TSessionData::SetHostKey(UnicodeString value){  SET_SESSION_PROPERTY(HostKey);}//---------------------------------------------------------------------void __fastcall TSessionData::SetNote(UnicodeString value){  SET_SESSION_PROPERTY(Note);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetInfoTip(){  if (UsesSsh)  {    return FMTLOAD(SESSION_INFO_TIP2,        (HostName, UserName,         (PublicKeyFile.IsEmpty() ? LoadStr(NO_STR) : LoadStr(YES_STR)),         FSProtocolStr));  }  else  {    return FMTLOAD(SESSION_INFO_TIP_NO_SSH,      (HostName, UserName, FSProtocolStr));  }}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::ExtractLocalName(const UnicodeString & Name){  UnicodeString Result = Name;  int P = Result.LastDelimiter(L"/");  if (P > 0)  {    Result.Delete(1, P);  }  return Result;}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetLocalName(){  UnicodeString Result;  if (HasSessionName())  {    Result = ExtractLocalName(Name);  }  else  {    Result = DefaultSessionName;  }  return Result;}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::ExtractFolderName(const UnicodeString & Name){  UnicodeString Result;  int P = Name.LastDelimiter(L"/");  if (P > 0)  {    Result = Name.SubString(1, P - 1);  }  return Result;}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetFolderName(){  UnicodeString Result;  if (HasSessionName() || IsWorkspace)  {    Result = ExtractFolderName(Name);  }  return Result;}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::ComposePath(  const UnicodeString & Path, const UnicodeString & Name){  return UnixIncludeTrailingBackslash(Path) + Name;}//=== TStoredSessionList ----------------------------------------------__fastcall TStoredSessionList::TStoredSessionList(bool aReadOnly):  TNamedObjectList(), FReadOnly(aReadOnly){  DebugAssert(Configuration);  FDefaultSettings = new TSessionData(DefaultName);}//---------------------------------------------------------------------__fastcall TStoredSessionList::~TStoredSessionList(){  DebugAssert(Configuration);  delete FDefaultSettings;}//---------------------------------------------------------------------void __fastcall TStoredSessionList::Load(THierarchicalStorage * Storage,  bool AsModified, bool UseDefaults){  TStringList *SubKeys = new TStringList();  TList * Loaded = new TList;  try  {    DebugAssert(AutoSort);    AutoSort = false;    bool WasEmpty = (Count == 0);    Storage->GetSubKeyNames(SubKeys);    for (int Index = 0; Index < SubKeys->Count; Index++)    {      UnicodeString SessionName = SubKeys->Strings[Index];      bool ValidName = true;      try      {        TSessionData::ValidatePath(SessionName);      }      catch(...)      {        ValidName = false;      }      if (ValidName)      {        TSessionData * SessionData;        if (SessionName == FDefaultSettings->Name)        {          SessionData = FDefaultSettings;        }        else        {          // if the list was empty before loading, do not waste time trying to          // find existing sites to overwrite (we rely on underlying storage          // to secure uniqueness of the key names)          if (WasEmpty)          {            SessionData = NULL;          }          else          {            SessionData = (TSessionData*)FindByName(SessionName);          }        }        if ((SessionData != FDefaultSettings) || !UseDefaults)        {          if (SessionData == NULL)          {            SessionData = new TSessionData(L"");            if (UseDefaults)            {              SessionData->CopyData(DefaultSettings);            }            SessionData->Name = SessionName;            Add(SessionData);          }          Loaded->Add(SessionData);          SessionData->Load(Storage);          if (AsModified)          {            SessionData->Modified = true;          }        }      }    }    if (!AsModified)    {      for (int Index = 0; Index < TObjectList::Count; Index++)      {        if (Loaded->IndexOf(Items[Index]) < 0)        {          Delete(Index);          Index--;        }      }    }  }  __finally  {    AutoSort = true;    AlphaSort();    delete SubKeys;    delete Loaded;  }}//---------------------------------------------------------------------void __fastcall TStoredSessionList::Load(){  bool SessionList = true;  THierarchicalStorage * Storage = Configuration->CreateScpStorage(SessionList);  try  {    if (Storage->OpenSubKey(Configuration->StoredSessionsSubKey, False))      Load(Storage);  }  __finally  {    delete Storage;  }}//---------------------------------------------------------------------void __fastcall TStoredSessionList::DoSave(THierarchicalStorage * Storage,  TSessionData * Data, bool All, bool RecryptPasswordOnly,  TSessionData * FactoryDefaults){  if (All || Data->Modified)  {    if (RecryptPasswordOnly)    {      Data->SaveRecryptedPasswords(Storage);    }    else    {      Data->Save(Storage, false, FactoryDefaults);    }  }}//---------------------------------------------------------------------void __fastcall TStoredSessionList::DoSave(THierarchicalStorage * Storage,  bool All, bool RecryptPasswordOnly, TStrings * RecryptPasswordErrors){  TSessionData * FactoryDefaults = new TSessionData(L"");  try  {    DoSave(Storage, FDefaultSettings, All, RecryptPasswordOnly, FactoryDefaults);    for (int Index = 0; Index < CountIncludingHidden; Index++)    {      TSessionData * SessionData = (TSessionData *)Items[Index];      try      {        DoSave(Storage, SessionData, All, RecryptPasswordOnly, FactoryDefaults);      }      catch (Exception & E)      {        UnicodeString Message;        if (RecryptPasswordOnly && DebugAlwaysTrue(RecryptPasswordErrors != NULL) &&            ExceptionMessage(&E, Message))        {          RecryptPasswordErrors->Add(FORMAT("%s: %s", (SessionData->SessionName, Message)));        }        else        {          throw;        }      }    }  }  __finally  {    delete FactoryDefaults;  }}//---------------------------------------------------------------------void __fastcall TStoredSessionList::Save(THierarchicalStorage * Storage, bool All){  DoSave(Storage, All, false, NULL);}//---------------------------------------------------------------------void __fastcall TStoredSessionList::DoSave(bool All, bool Explicit,  bool RecryptPasswordOnly, TStrings * RecryptPasswordErrors){  bool SessionList = true;  THierarchicalStorage * Storage = Configuration->CreateScpStorage(SessionList);  try  {    Storage->AccessMode = smReadWrite;    Storage->Explicit = Explicit;    if (Storage->OpenSubKey(Configuration->StoredSessionsSubKey, true))    {      DoSave(Storage, All, RecryptPasswordOnly, RecryptPasswordErrors);    }  }  __finally  {    delete Storage;  }  Saved();}//---------------------------------------------------------------------void __fastcall TStoredSessionList::Save(bool All, bool Explicit){  DoSave(All, Explicit, false, NULL);}//---------------------------------------------------------------------void __fastcall TStoredSessionList::RecryptPasswords(TStrings * RecryptPasswordErrors){  DoSave(true, true, true, RecryptPasswordErrors);}//---------------------------------------------------------------------void __fastcall TStoredSessionList::Saved(){  FDefaultSettings->Modified = false;  for (int Index = 0; Index < CountIncludingHidden; Index++)  {    ((TSessionData *)Items[Index])->Modified = false;  }}//---------------------------------------------------------------------void __fastcall TStoredSessionList::ImportLevelFromFilezilla(_di_IXMLNode Node, const UnicodeString & Path){  for (int Index = 0; Index < Node->ChildNodes->Count; Index++)  {    _di_IXMLNode ChildNode = Node->ChildNodes->Get(Index);    if (ChildNode->NodeName == L"Server")    {      std::unique_ptr<TSessionData> SessionData(new TSessionData(L""));      SessionData->CopyData(DefaultSettings);      SessionData->ImportFromFilezilla(ChildNode, Path);      Add(SessionData.release());    }    else if (ChildNode->NodeName == L"Folder")    {      UnicodeString Name;      for (int Index = 0; Index < ChildNode->ChildNodes->Count; Index++)      {        _di_IXMLNode PossibleTextMode = ChildNode->ChildNodes->Get(Index);        if (PossibleTextMode->NodeType == ntText)        {          UnicodeString NodeValue = PossibleTextMode->NodeValue;          AddToList(Name, NodeValue.Trim(), L" ");        }      }      Name = TSessionData::MakeValidName(Name).Trim();      ImportLevelFromFilezilla(ChildNode, TSessionData::ComposePath(Path, Name));    }  }}//---------------------------------------------------------------------void __fastcall TStoredSessionList::ImportFromFilezilla(const UnicodeString FileName){  const _di_IXMLDocument Document = interface_cast<Xmlintf::IXMLDocument>(new TXMLDocument(NULL));  Document->LoadFromFile(FileName);  _di_IXMLNode FileZilla3Node = Document->ChildNodes->FindNode(L"FileZilla3");  if (FileZilla3Node != NULL)  {    _di_IXMLNode ServersNode = FileZilla3Node->ChildNodes->FindNode(L"Servers");    if (ServersNode != NULL)    {      ImportLevelFromFilezilla(ServersNode, L"");    }  }}//---------------------------------------------------------------------void __fastcall TStoredSessionList::Export(const UnicodeString FileName){  THierarchicalStorage * Storage = new TIniFileStorage(FileName);  try  {    Storage->AccessMode = smReadWrite;    if (Storage->OpenSubKey(Configuration->StoredSessionsSubKey, true))    {      Save(Storage, true);    }  }  __finally  {    delete Storage;  }}//---------------------------------------------------------------------void __fastcall TStoredSessionList::SelectAll(bool Select){  for (int Index = 0; Index < Count; Index++)    Sessions[Index]->Selected = Select;}//---------------------------------------------------------------------void __fastcall TStoredSessionList::Import(TStoredSessionList * From,  bool OnlySelected, TList * Imported){  for (int Index = 0; Index < From->Count; Index++)  {    if (!OnlySelected || From->Sessions[Index]->Selected)    {      TSessionData *Session = new TSessionData(L"");      Session->Assign(From->Sessions[Index]);      Session->Modified = true;      Session->MakeUniqueIn(this);      Add(Session);      if (Imported != NULL)      {        Imported->Add(Session);      }    }  }  // only modified, explicit  Save(false, true);}//---------------------------------------------------------------------void __fastcall TStoredSessionList::SelectSessionsToImport  (TStoredSessionList * Dest, bool SSHOnly){  for (int Index = 0; Index < Count; Index++)  {    Sessions[Index]->Selected =      (!SSHOnly || (Sessions[Index]->GetNormalizedPuttyProtocol() == PuttySshProtocol)) &&      !Dest->FindByName(Sessions[Index]->Name);  }}//---------------------------------------------------------------------void __fastcall TStoredSessionList::Cleanup(){  try  {    if (Configuration->Storage == stRegistry) Clear();    TRegistryStorage * Storage = new TRegistryStorage(Configuration->RegistryStorageKey);    try    {      Storage->AccessMode = smReadWrite;      if (Storage->OpenRootKey(False))        Storage->RecursiveDeleteSubKey(Configuration->StoredSessionsSubKey);    }    __finally    {      delete Storage;    }  }  catch (Exception &E)  {    throw ExtException(&E, LoadStr(CLEANUP_SESSIONS_ERROR));  }}//---------------------------------------------------------------------------void __fastcall TStoredSessionList::UpdateStaticUsage(){  int SCP = 0;  int SFTP = 0;  int FTP = 0;  int FTPS = 0;  int WebDAV = 0;  int WebDAVS = 0;  int Password = 0;  int Advanced = 0;  int Color = 0;  int Note = 0;  int Tunnel = 0;  bool Folders = false;  bool Workspaces = false;  std::unique_ptr<TSessionData> FactoryDefaults(new TSessionData(L""));  std::unique_ptr<TStringList> DifferentAdvancedProperties(CreateSortedStringList());  for (int Index = 0; Index < Count; Index++)  {    TSessionData * Data = Sessions[Index];    if (Data->IsWorkspace)    {      Workspaces = true;    }    else    {      switch (Data->FSProtocol)      {        case fsSCPonly:          SCP++;          break;        case fsSFTP:        case fsSFTPonly:          SFTP++;          break;        case fsFTP:          if (Data->Ftps == ftpsNone)          {            FTP++;          }          else          {            FTPS++;          }          break;        case fsWebDAV:          if (Data->Ftps == ftpsNone)          {            WebDAV++;          }          else          {            WebDAVS++;          }          break;      }      if (Data->HasAnySessionPassword())      {        Password++;      }      if (Data->Color != 0)      {        Color++;      }      if (!Data->Note.IsEmpty())      {        Note++;      }      // this effectively does not take passwords (proxy + tunnel) into account,      // when master password is set, as master password handler in not set up yet      if (!Data->IsSame(FactoryDefaults.get(), true, DifferentAdvancedProperties.get()))      {        Advanced++;      }      if (Data->Tunnel)      {        Tunnel++;      }      if (!Data->FolderName.IsEmpty())      {        Folders = true;      }    }  }  Configuration->Usage->Set(L"StoredSessionsCountSCP", SCP);  Configuration->Usage->Set(L"StoredSessionsCountSFTP", SFTP);  Configuration->Usage->Set(L"StoredSessionsCountFTP", FTP);  Configuration->Usage->Set(L"StoredSessionsCountFTPS", FTPS);  Configuration->Usage->Set(L"StoredSessionsCountWebDAV", WebDAV);  Configuration->Usage->Set(L"StoredSessionsCountWebDAVS", WebDAVS);  Configuration->Usage->Set(L"StoredSessionsCountPassword", Password);  Configuration->Usage->Set(L"StoredSessionsCountColor", Color);  Configuration->Usage->Set(L"StoredSessionsCountNote", Note);  Configuration->Usage->Set(L"StoredSessionsCountAdvanced", Advanced);  DifferentAdvancedProperties->Delimiter = L',';  Configuration->Usage->Set(L"StoredSessionsAdvancedSettings", DifferentAdvancedProperties->DelimitedText);  Configuration->Usage->Set(L"StoredSessionsCountTunnel", Tunnel);  // actually default might be true, see below for when the default is actually used  bool CustomDefaultStoredSession = false;  try  {    // this can throw, when the default session settings have password set    // (and no other basic property, like hostname/username),    // and master password is enabled as we are called before master password    // handler is set    CustomDefaultStoredSession = !FDefaultSettings->IsSame(FactoryDefaults.get(), false);  }  catch (...)  {  }  Configuration->Usage->Set(L"UsingDefaultStoredSession", CustomDefaultStoredSession);  Configuration->Usage->Set(L"UsingStoredSessionsFolders", Folders);  Configuration->Usage->Set(L"UsingWorkspaces", Workspaces);}//---------------------------------------------------------------------------TSessionData * __fastcall TStoredSessionList::FindSame(TSessionData * Data){  TSessionData * Result;  if (Data->Hidden || Data->Name.IsEmpty() || Data->IsWorkspace)  {    Result = NULL;  }  else  {    Result = dynamic_cast<TSessionData *>(FindByName(Data->Name));  }  return Result;}//---------------------------------------------------------------------------int __fastcall TStoredSessionList::IndexOf(TSessionData * Data){  for (int Index = 0; Index < Count; Index++)    if (Data == Sessions[Index]) return Index;  return -1;}//---------------------------------------------------------------------------TSessionData * __fastcall TStoredSessionList::NewSession(  UnicodeString SessionName, TSessionData * Session){  TSessionData * DuplicateSession = (TSessionData*)FindByName(SessionName);  if (!DuplicateSession)  {    DuplicateSession = new TSessionData(L"");    DuplicateSession->Assign(Session);    DuplicateSession->Name = SessionName;    // make sure, that new stored session is saved to registry    DuplicateSession->Modified = true;    Add(DuplicateSession);  }  else  {    DuplicateSession->Assign(Session);    DuplicateSession->Name = SessionName;    DuplicateSession->Modified = true;  }  // list was saved here before to default storage, but it would not allow  // to work with special lists (export/import) not using default storage  return DuplicateSession;}//---------------------------------------------------------------------------void __fastcall TStoredSessionList::SetDefaultSettings(TSessionData * value){  DebugAssert(FDefaultSettings);  if (FDefaultSettings != value)  {    FDefaultSettings->Assign(value);    // make sure default settings are saved    FDefaultSettings->Modified = true;    FDefaultSettings->Name = DefaultName;    if (!FReadOnly)    {      // only modified, explicit      Save(false, true);    }  }}//---------------------------------------------------------------------------void __fastcall TStoredSessionList::ImportHostKeys(const UnicodeString TargetKey,  const UnicodeString SourceKey, TStoredSessionList * Sessions,  bool OnlySelected){  TRegistryStorage * SourceStorage = NULL;  TRegistryStorage * TargetStorage = NULL;  TStringList * KeyList = NULL;  try  {    SourceStorage = new TRegistryStorage(SourceKey);    TargetStorage = new TRegistryStorage(TargetKey);    TargetStorage->AccessMode = smReadWrite;    KeyList = new TStringList();    if (SourceStorage->OpenRootKey(false) &&        TargetStorage->OpenRootKey(true))    {      SourceStorage->GetValueNames(KeyList);      TSessionData * Session;      UnicodeString HostKeyName;      DebugAssert(Sessions != NULL);      for (int Index = 0; Index < Sessions->Count; Index++)      {        Session = Sessions->Sessions[Index];        if (!OnlySelected || Session->Selected)        {          HostKeyName = PuttyMungeStr(FORMAT(L"@%d:%s", (Session->PortNumber, Session->HostNameExpanded)));          UnicodeString KeyName;          for (int KeyIndex = 0; KeyIndex < KeyList->Count; KeyIndex++)          {            KeyName = KeyList->Strings[KeyIndex];            int P = KeyName.Pos(HostKeyName);            if ((P > 0) && (P == KeyName.Length() - HostKeyName.Length() + 1))            {              TargetStorage->WriteStringRaw(KeyName,                SourceStorage->ReadStringRaw(KeyName, L""));            }          }        }      }    }  }  __finally  {    delete SourceStorage;    delete TargetStorage;    delete KeyList;  }}//---------------------------------------------------------------------------bool __fastcall TStoredSessionList::IsFolderOrWorkspace(  const UnicodeString & Name, bool Workspace){  bool Result = false;  TSessionData * FirstData = NULL;  if (!Name.IsEmpty())  {    for (int Index = 0; !Result && (Index < Count); Index++)    {      Result = Sessions[Index]->IsInFolderOrWorkspace(Name);      if (Result)      {        FirstData = Sessions[Index];      }    }  }  return    Result &&    DebugAlwaysTrue(FirstData != NULL) &&    (FirstData->IsWorkspace == Workspace);}//---------------------------------------------------------------------------bool __fastcall TStoredSessionList::IsFolder(const UnicodeString & Name){  return IsFolderOrWorkspace(Name, false);}//---------------------------------------------------------------------------bool __fastcall TStoredSessionList::IsWorkspace(const UnicodeString & Name){  return IsFolderOrWorkspace(Name, true);}//---------------------------------------------------------------------------TSessionData * __fastcall TStoredSessionList::CheckIsInFolderOrWorkspaceAndResolve(  TSessionData * Data, const UnicodeString & Name){  if (Data->IsInFolderOrWorkspace(Name))  {    Data = ResolveWorkspaceData(Data);    if ((Data != NULL) && Data->CanLogin &&        DebugAlwaysTrue(Data->Link.IsEmpty()))    {      return Data;    }  }  return NULL;}//---------------------------------------------------------------------------void __fastcall TStoredSessionList::GetFolderOrWorkspace(const UnicodeString & Name, TList * List){  for (int Index = 0; (Index < Count); Index++)  {    TSessionData * RawData = Sessions[Index];    TSessionData * Data =      CheckIsInFolderOrWorkspaceAndResolve(RawData, Name);    if (Data != NULL)    {      TSessionData * Data2 = new TSessionData(L"");      Data2->Assign(Data);      if (!RawData->Link.IsEmpty() && (DebugAlwaysTrue(Data != RawData)) &&          // BACKWARD COMPATIBILITY          // When loading pre-5.6.4 workspace, that does not have state saved,          // do not overwrite the site "state" defaults          // with (empty) workspace state          RawData->HasStateData())      {        Data2->CopyStateData(RawData);      }      List->Add(Data2);    }  }}//---------------------------------------------------------------------------TStrings * __fastcall TStoredSessionList::GetFolderOrWorkspaceList(  const UnicodeString & Name){  std::unique_ptr<TStringList> Result(new TStringList());  for (int Index = 0; (Index < Count); Index++)  {    TSessionData * Data =      CheckIsInFolderOrWorkspaceAndResolve(Sessions[Index], Name);    if (Data != NULL)    {      Result->Add(Data->SessionName);    }  }  return Result.release();}//---------------------------------------------------------------------------TStrings * __fastcall TStoredSessionList::GetWorkspaces(){  std::unique_ptr<TStringList> Result(CreateSortedStringList());  for (int Index = 0; (Index < Count); Index++)  {    TSessionData * Data = Sessions[Index];    if (Data->IsWorkspace)    {      Result->Add(Data->FolderName);    }  }  return Result.release();}//---------------------------------------------------------------------------void __fastcall TStoredSessionList::NewWorkspace(  UnicodeString Name, TList * DataList){  for (int Index = 0; (Index < Count); Index++)  {    TSessionData * Data = Sessions[Index];    if (Data->IsInFolderOrWorkspace(Name))    {      Data->Remove();      Remove(Data);      Index--;    }  }  for (int Index = 0; (Index < DataList->Count); Index++)  {    TSessionData * Data = static_cast<TSessionData *>(DataList->Items[Index]);    TSessionData * Data2 = new TSessionData(L"");    Data2->Assign(Data);    Data2->Name = TSessionData::ComposePath(Name, Data->Name);    // make sure, that new stored session is saved to registry    Data2->Modified = true;    Add(Data2);  }}//---------------------------------------------------------------------------bool __fastcall TStoredSessionList::HasAnyWorkspace(){  bool Result = false;  for (int Index = 0; !Result && (Index < Count); Index++)  {    TSessionData * Data = Sessions[Index];    Result = Data->IsWorkspace;  }  return Result;}//---------------------------------------------------------------------------TSessionData * __fastcall TStoredSessionList::ParseUrl(UnicodeString Url,  TOptions * Options, bool & DefaultsOnly, UnicodeString * FileName,  bool * AProtocolDefined, UnicodeString * MaskedUrl){  TSessionData * Data = new TSessionData(L"");  try  {    Data->ParseUrl(Url, Options, this, DefaultsOnly, FileName, AProtocolDefined, MaskedUrl);  }  catch(...)  {    delete Data;    throw;  }  return Data;}//---------------------------------------------------------------------bool __fastcall TStoredSessionList::IsUrl(UnicodeString Url){  bool DefaultsOnly;  bool ProtocolDefined = false;  std::unique_ptr<TSessionData> ParsedData(ParseUrl(Url, NULL, DefaultsOnly, NULL, &ProtocolDefined));  bool Result = ProtocolDefined;  return Result;}//---------------------------------------------------------------------TSessionData * __fastcall TStoredSessionList::ResolveWorkspaceData(TSessionData * Data){  if (!Data->Link.IsEmpty())  {    Data = dynamic_cast<TSessionData *>(FindByName(Data->Link));    if (Data != NULL)    {      Data = ResolveWorkspaceData(Data);    }  }  return Data;}//---------------------------------------------------------------------TSessionData * __fastcall TStoredSessionList::SaveWorkspaceData(TSessionData * Data){  std::unique_ptr<TSessionData> Result(new TSessionData(L""));  TSessionData * SameData = StoredSessions->FindSame(Data);  if (SameData != NULL)  {    Result->CopyStateData(Data);    Result->Link = Data->Name;  }  else  {    Result->Assign(Data);  }  Result->IsWorkspace = true;  return Result.release();}//---------------------------------------------------------------------bool __fastcall TStoredSessionList::CanLogin(TSessionData * Data){  Data = ResolveWorkspaceData(Data);  return (Data != NULL) && Data->CanLogin;}//---------------------------------------------------------------------UnicodeString GetExpandedLogFileName(UnicodeString LogFileName, TSessionData * SessionData){  // StripPathQuotes should not be needed as we do not feed quotes anymore  UnicodeString ANewFileName = StripPathQuotes(ExpandEnvironmentVariables(LogFileName));  TDateTime N = Now();  for (int Index = 1; Index < ANewFileName.Length(); Index++)  {    if (ANewFileName[Index] == L'!')    {      UnicodeString Replacement;      // keep consistent with TFileCustomCommand::PatternReplacement      switch (tolower(ANewFileName[Index + 1]))      {        case L'y':          Replacement = FormatDateTime(L"yyyy", N);          break;        case L'm':          Replacement = FormatDateTime(L"mm", N);          break;        case L'd':          Replacement = FormatDateTime(L"dd", N);          break;        case L't':          Replacement = FormatDateTime(L"hhnnss", N);          break;        case 'p':          Replacement = IntToStr(static_cast<int>(GetCurrentProcessId()));          break;        case L'@':          if (SessionData != NULL)          {            Replacement = MakeValidFileName(SessionData->HostNameExpanded);          }          else          {            Replacement = L"nohost";          }          break;        case L's':          if (SessionData != NULL)          {            Replacement = MakeValidFileName(SessionData->SessionName);          }          else          {            Replacement = L"nosession";          }          break;        case L'!':          Replacement = L"!";          break;        default:          Replacement = UnicodeString(L"!") + ANewFileName[Index + 1];          break;      }      ANewFileName.Delete(Index, 2);      ANewFileName.Insert(Replacement, Index);      Index += Replacement.Length() - 1;    }  }  return ANewFileName;}//---------------------------------------------------------------------bool __fastcall IsSshProtocol(TFSProtocol FSProtocol){  return    (FSProtocol == fsSFTPonly) || (FSProtocol == fsSFTP) ||    (FSProtocol == fsSCPonly);}//---------------------------------------------------------------------------int __fastcall DefaultPort(TFSProtocol FSProtocol, TFtps Ftps){  int Result;  switch (FSProtocol)  {    case fsFTP:      if (Ftps == ftpsImplicit)      {        Result = FtpsImplicitPortNumber;      }      else      {        Result = FtpPortNumber;      }      break;    case fsWebDAV:      if (Ftps == ftpsNone)      {        Result = HTTPPortNumber;      }      else      {        Result = HTTPSPortNumber;      }      break;    default:      if (IsSshProtocol(FSProtocol))      {        Result = SshPortNumber;      }      else      {        DebugFail();        Result = -1;      }      break;  }  return Result;}//---------------------------------------------------------------------
 |