| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912 | //---------------------------------------------------------------------------#include <vcl.h>#pragma hdrstop#include <FileInfo.h>#include "Common.h"#include "Exceptions.h"#include "Configuration.h"#include "PuttyIntf.h"#include "TextsCore.h"#include "Interface.h"#include "CoreMain.h"#include "Security.h"#include "FileMasks.h"#include "CopyParam.h"#include <shlobj.h>#include <System.IOUtils.hpp>#include <System.StrUtils.hpp>//---------------------------------------------------------------------------#pragma package(smart_init)//---------------------------------------------------------------------------const wchar_t * AutoSwitchNames = L"On;Off;Auto";const wchar_t * NotAutoSwitchNames = L"Off;On;Auto";//---------------------------------------------------------------------------// See https://www.iana.org/assignments/hash-function-text-names/hash-function-text-names.xhtmlconst UnicodeString Sha1ChecksumAlg(L"sha-1");const UnicodeString Sha224ChecksumAlg(L"sha-224");const UnicodeString Sha256ChecksumAlg(L"sha-256");const UnicodeString Sha384ChecksumAlg(L"sha-384");const UnicodeString Sha512ChecksumAlg(L"sha-512");const UnicodeString Md5ChecksumAlg(L"md5");// Not defined by IANAconst UnicodeString Crc32ChecksumAlg(L"crc32");//---------------------------------------------------------------------------const UnicodeString SshFingerprintType(L"ssh");const UnicodeString TlsFingerprintType(L"tls");//---------------------------------------------------------------------------const UnicodeString FtpsCertificateStorageKey(L"FtpsCertificates");const UnicodeString HttpsCertificateStorageKey(L"HttpsCertificates");const UnicodeString LastFingerprintsStorageKey(L"LastFingerprints");const UnicodeString DirectoryStatisticsCacheKey(L"DirectoryStatisticsCache");const UnicodeString CDCacheKey(L"CDCache");const UnicodeString BannersKey(L"Banners");//---------------------------------------------------------------------------__fastcall TConfiguration::TConfiguration(){  FCriticalSection = new TCriticalSection();  FUpdating = 0;  FStorage = stDetect;  FDontSave = false;  FForceSave = false;  FApplicationInfo = NULL;  FUsage = new TUsage(this);  FDefaultCollectUsage = false;  FScripting = false;  UnicodeString RandomSeedPath;  if (!GetEnvironmentVariable(L"APPDATA").IsEmpty())  {    RandomSeedPath = L"%APPDATA%";  }  else  {    RandomSeedPath = GetShellFolderPath(CSIDL_LOCAL_APPDATA);    if (RandomSeedPath.IsEmpty())    {      RandomSeedPath = GetShellFolderPath(CSIDL_APPDATA);    }  }  FDefaultRandomSeedFile = IncludeTrailingBackslash(RandomSeedPath) + L"winscp.rnd";}//---------------------------------------------------------------------------void __fastcall TConfiguration::Default(){  TGuard Guard(FCriticalSection);  FDisablePasswordStoring = false;  FForceBanners = false;  FDisableAcceptingHostKeys = false;  TRegistryStorage * AdminStorage;  AdminStorage = new TRegistryStorage(RegistryStorageKey, HKEY_LOCAL_MACHINE);  try  {    if (AdminStorage->OpenRootKey(false))    {      LoadAdmin(AdminStorage);      AdminStorage->CloseSubKey();    }  }  __finally  {    delete AdminStorage;  }  RandomSeedFile = FDefaultRandomSeedFile;  PuttyRegistryStorageKey = OriginalPuttyRegistryStorageKey;  FConfirmOverwriting = true;  FConfirmResume = true;  FAutoReadDirectoryAfterOp = true;  FSessionReopenAuto = 5000;  FSessionReopenBackground = 2000;  FSessionReopenTimeout = 0;  FSessionReopenAutoStall = 60000;  FTunnelLocalPortNumberLow = 50000;  FTunnelLocalPortNumberHigh = 50099;  FCacheDirectoryChangesMaxSize = 100;  FShowFtpWelcomeMessage = false;  FExternalIpAddress = L"";  FTryFtpWhenSshFails = true;  FParallelDurationThreshold = 10;  FMimeTypes = UnicodeString();  FDontReloadMoreThanSessions = 1000;  FScriptProgressFileNameLimit = 25;  CollectUsage = FDefaultCollectUsage;  FLogging = false;  FPermanentLogging = false;  FLogFileName = DefaultLogFileName;  FPermanentLogFileName = FLogFileName;  FLogFileAppend = true;  FLogSensitive = false;  FPermanentLogSensitive = FLogSensitive;  FLogMaxSize = 0;  FPermanentLogMaxSize = FLogMaxSize;  FLogMaxCount = 0;  FPermanentLogMaxCount = FLogMaxCount;  FLogProtocol = 0;  FPermanentLogProtocol = FLogProtocol;  UpdateActualLogProtocol();  FLogActions = false;  FPermanentLogActions = false;  FLogActionsRequired = false;  FActionsLogFileName = L"%TEMP%\\!S.xml";  FPermanentActionsLogFileName = FActionsLogFileName;  FProgramIniPathWrittable = -1;  FCustomIniFileStorageName = LoadCustomIniFileStorageName();  Changed();}//---------------------------------------------------------------------------__fastcall TConfiguration::~TConfiguration(){  DebugAssert(!FUpdating);  if (FApplicationInfo) FreeFileInfo(FApplicationInfo);  delete FCriticalSection;  delete FUsage;}//---------------------------------------------------------------------------void __fastcall TConfiguration::UpdateStaticUsage(){  Usage->Set(L"ConfigurationIniFile", (Storage == stIniFile));  Usage->Set(L"ConfigurationIniFileCustom", !CustomIniFileStorageName.IsEmpty());  Usage->Set("Unofficial", IsUnofficial);  // this is called from here, because we are guarded from calling into  // master password handler here, see TWinConfiguration::UpdateStaticUsage  StoredSessions->UpdateStaticUsage();}//---------------------------------------------------------------------------THierarchicalStorage * TConfiguration::CreateConfigStorage(){  bool SessionList = false;  return CreateScpStorage(SessionList);}//---------------------------------------------------------------------------THierarchicalStorage * TConfiguration::CreateScpStorage(bool & SessionList){  TGuard Guard(FCriticalSection);  THierarchicalStorage * Result;  if (Storage == stRegistry)  {    Result = new TRegistryStorage(RegistryStorageKey);  }  else if (Storage == stNul)  {    Result = TIniFileStorage::CreateFromPath(INI_NUL);  }  else  {    UnicodeString StorageName = IniFileStorageName;    Result = TIniFileStorage::CreateFromPath(StorageName);  }  if ((FOptionsStorage.get() != NULL) && (FOptionsStorage->Count > 0))  {    if (!SessionList)    {      Result = new TOptionsStorage(FOptionsStorage.get(), ConfigurationSubKey, Result);    }    else    {      // cannot reuse session list storage for configuration as for it we need      // the option-override storage above    }  }  else  {    // All the above stores can be reused for configuration,    // if no options-overrides are set    SessionList = false;  }  return Result;}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::PropertyToKey(const UnicodeString & Property){  // no longer useful  int P = Property.LastDelimiter(L".>");  UnicodeString Result = Property.SubString(P + 1, Property.Length() - P);  if ((Result[1] == L'F') && ((wchar_t)toupper(Result[2]) == Result[2]))  {    Result.Delete(1, 1);  }  return Result;}//---------------------------------------------------------------------------#define BLOCK(KEY, CANCREATE, BLOCK) \  if (Storage->OpenSubKeyPath(KEY, CANCREATE)) try { BLOCK } __finally { Storage->CloseSubKeyPath(); }#define KEY(TYPE, VAR) KEYEX(TYPE, VAR, PropertyToKey(TEXT(#VAR)))#define REGCONFIG(CANCREATE) \  BLOCK(L"Interface", CANCREATE, \    KEY(String,   RandomSeedFile); \    KEY(String,   PuttyRegistryStorageKey); \    KEY(Bool,     ConfirmOverwriting); \    KEY(Bool,     ConfirmResume); \    KEY(Bool,     AutoReadDirectoryAfterOp); \    KEY(Integer,  SessionReopenAuto); \    KEY(Integer,  SessionReopenBackground); \    KEY(Integer,  SessionReopenTimeout); \    KEY(Integer,  SessionReopenAutoStall); \    KEY(Integer,  TunnelLocalPortNumberLow); \    KEY(Integer,  TunnelLocalPortNumberHigh); \    KEY(Integer,  CacheDirectoryChangesMaxSize); \    KEY(Bool,     ShowFtpWelcomeMessage); \    KEY(String,   ExternalIpAddress); \    KEY(Bool,     TryFtpWhenSshFails); \    KEY(Integer,  ParallelDurationThreshold); \    KEY(String,   MimeTypes); \    KEY(Integer,  DontReloadMoreThanSessions); \    KEY(Integer,  ScriptProgressFileNameLimit); \    KEY(Bool,     CollectUsage); \  ); \  BLOCK(L"Logging", CANCREATE, \    KEYEX(Bool,  PermanentLogging, L"Logging"); \    KEYEX(String,PermanentLogFileName, L"LogFileName"); \    KEY(Bool,    LogFileAppend); \    KEYEX(Bool,  PermanentLogSensitive, L"LogSensitive"); \    KEYEX(Int64, PermanentLogMaxSize, L"LogMaxSize"); \    KEYEX(Integer, PermanentLogMaxCount, L"LogMaxCount"); \    KEYEX(Integer,PermanentLogProtocol, L"LogProtocol"); \    KEYEX(Bool,  PermanentLogActions, L"LogActions"); \    KEYEX(String,PermanentActionsLogFileName, L"ActionsLogFileName"); \  );//---------------------------------------------------------------------------void __fastcall TConfiguration::SaveData(THierarchicalStorage * Storage, bool /*All*/){  #define KEYEX(TYPE, VAR, NAME) Storage->Write ## TYPE(NAME, VAR)  REGCONFIG(true);  #undef KEYEX  if (Storage->OpenSubKey(L"Usage", true))  {    FUsage->Save(Storage);    Storage->CloseSubKey();  }}//---------------------------------------------------------------------------void __fastcall TConfiguration::Save(){  // only modified, implicit  DoSave(false, false);}//---------------------------------------------------------------------------void __fastcall TConfiguration::SaveExplicit(){  // only modified, explicit  DoSave(false, true);}//---------------------------------------------------------------------------void __fastcall TConfiguration::DoSave(bool All, bool Explicit){  if (FDontSave) return;  THierarchicalStorage * AStorage = CreateConfigStorage();  try  {    AStorage->AccessMode = smReadWrite;    AStorage->Explicit = Explicit;    AStorage->ForceSave = FForceSave;    if (AStorage->OpenSubKey(ConfigurationSubKey, true))    {      // if saving to TOptionsStorage, make sure we save everything so that      // all configuration is properly transferred to the master storage      bool ConfigAll = All || AStorage->Temporary;      SaveData(AStorage, ConfigAll);    }  }  __finally  {    delete AStorage;  }  Saved();  if (All)  {    StoredSessions->Save(true, Explicit);  }  // clean up as last, so that if it fails (read only INI), the saving can proceed  if (Storage == stRegistry)  {    CleanupIniFile();  }  SaveCustomIniFileStorageName();}//---------------------------------------------------------------------------void __fastcall TConfiguration::SaveCustomIniFileStorageName(){  // Particularly, not to create an empty "Override" key, unless the custom INI file is ever set  if (CustomIniFileStorageName != LoadCustomIniFileStorageName())  {    std::unique_ptr<TRegistryStorage> RegistryStorage(new TRegistryStorage(GetRegistryStorageOverrideKey()));    RegistryStorage->AccessMode = smReadWrite;    RegistryStorage->Explicit = true;    if (RegistryStorage->OpenRootKey(true))    {      RegistryStorage->WriteString(L"IniFile", CustomIniFileStorageName);      RegistryStorage->CloseSubKey();    }  }}//---------------------------------------------------------------------------void __fastcall TConfiguration::Export(const UnicodeString & FileName){  // not to "append" the export to an existing file  if (FileExists(FileName))  {    DeleteFileChecked(FileName);  }  THierarchicalStorage * Storage = NULL;  THierarchicalStorage * ExportStorage = NULL;  try  {    ExportStorage = TIniFileStorage::CreateFromPath(FileName);    ExportStorage->AccessMode = smReadWrite;    ExportStorage->Explicit = true;    Storage = CreateConfigStorage();    Storage->AccessMode = smRead;    CopyData(Storage, ExportStorage);    if (ExportStorage->OpenSubKey(ConfigurationSubKey, true))    {      SaveData(ExportStorage, true);    }  }  __finally  {    delete ExportStorage;    delete Storage;  }  StoredSessions->Export(FileName);}//---------------------------------------------------------------------------void __fastcall TConfiguration::Import(const UnicodeString & FileName){  THierarchicalStorage * Storage = NULL;  THierarchicalStorage * ImportStorage = NULL;  try  {    ImportStorage = TIniFileStorage::CreateFromPath(FileName);    ImportStorage->AccessMode = smRead;    Storage = CreateConfigStorage();    Storage->AccessMode = smReadWrite;    Storage->Explicit = true;    CopyData(ImportStorage, Storage);    Default();    LoadFrom(ImportStorage);    if (ImportStorage->OpenSubKey(Configuration->StoredSessionsSubKey, false))    {      StoredSessions->Clear();      StoredSessions->DefaultSettings->Default();      StoredSessions->Load(ImportStorage);    }  }  __finally  {    delete ImportStorage;    delete Storage;  }  // save all and explicit  DoSave(true, true);}//---------------------------------------------------------------------------void __fastcall TConfiguration::LoadData(THierarchicalStorage * Storage){  #define KEYEX(TYPE, VAR, NAME) VAR = Storage->Read ## TYPE(NAME, VAR)  #pragma warn -eas  REGCONFIG(false);  #pragma warn +eas  #undef KEYEX  if (Storage->OpenSubKey(L"Usage", false))  {    FUsage->Load(Storage);    Storage->CloseSubKey();  }  if (FPermanentLogActions && FPermanentActionsLogFileName.IsEmpty() &&      FPermanentLogging && !FPermanentLogFileName.IsEmpty())  {     FPermanentActionsLogFileName = FPermanentLogFileName;     FPermanentLogging = false;     FPermanentLogFileName = L"";  }}//---------------------------------------------------------------------------void __fastcall TConfiguration::LoadAdmin(THierarchicalStorage * Storage){  FDisablePasswordStoring = Storage->ReadBool(L"DisablePasswordStoring", FDisablePasswordStoring);  FForceBanners = Storage->ReadBool(L"ForceBanners", FForceBanners);  FDisableAcceptingHostKeys = Storage->ReadBool(L"DisableAcceptingHostKeys", FDisableAcceptingHostKeys);  FDefaultCollectUsage = Storage->ReadBool(L"DefaultCollectUsage", FDefaultCollectUsage);}//---------------------------------------------------------------------------void __fastcall TConfiguration::LoadFrom(THierarchicalStorage * Storage){  if (Storage->OpenSubKey(ConfigurationSubKey, false))  {    LoadData(Storage);    Storage->CloseSubKey();  }}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetRegistryStorageOverrideKey(){  return GetRegistryStorageKey() + L" Override";}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::LoadCustomIniFileStorageName(){  UnicodeString Result;  std::unique_ptr<TRegistryStorage> RegistryStorage(new TRegistryStorage(GetRegistryStorageOverrideKey()));  if (RegistryStorage->OpenRootKey(false))  {    Result = RegistryStorage->ReadString(L"IniFile", L"");    RegistryStorage->CloseSubKey();  }  RegistryStorage.reset(NULL);  return Result;}//---------------------------------------------------------------------------void __fastcall TConfiguration::Load(THierarchicalStorage * Storage){  TGuard Guard(FCriticalSection);  TStorageAccessMode StorageAccessMode = Storage->AccessMode;  try  {    Storage->AccessMode = smRead;    LoadFrom(Storage);  }  __finally  {    Storage->AccessMode = StorageAccessMode;  }}//---------------------------------------------------------------------------bool __fastcall TConfiguration::CopySubKey(THierarchicalStorage * Source, THierarchicalStorage * Target, const UnicodeString & Name){  bool Result = Source->OpenSubKey(Name, false);  if (Result)  {    Result = Target->OpenSubKey(Name, true);    if (!Result)    {      Source->CloseSubKey();    }  }  return Result;}//---------------------------------------------------------------------------void __fastcall TConfiguration::CopyAllStringsInSubKey(  THierarchicalStorage * Source, THierarchicalStorage * Target, const UnicodeString & Name){  if (CopySubKey(Source, Target, Name))  {    std::unique_ptr<TStrings> Names(new TStringList());    Source->GetValueNames(Names.get());    for (int Index = 0; Index < Names->Count; Index++)    {      UnicodeString Buf = Source->ReadStringRaw(Names->Strings[Index], UnicodeString());      Target->WriteStringRaw(Names->Strings[Index], Buf);    }    Target->CloseSubKey();    Source->CloseSubKey();  }}//---------------------------------------------------------------------------void __fastcall TConfiguration::CopyData(THierarchicalStorage * Source,  THierarchicalStorage * Target){  if (CopySubKey(Source, Target, ConfigurationSubKey))  {    if (CopySubKey(Source, Target, CDCacheKey))    {      std::unique_ptr<TStrings> Names(new TStringList());      Source->GetValueNames(Names.get());      for (int Index = 0; Index < Names->Count; Index++)      {        Target->WriteBinaryData(Names->Strings[Index], Source->ReadBinaryData(Names->Strings[Index]));      }      Target->CloseSubKey();      Source->CloseSubKey();    }    CopyAllStringsInSubKey(Source, Target, BannersKey);    CopyAllStringsInSubKey(Source, Target, LastFingerprintsStorageKey);    Target->CloseSubKey();    Source->CloseSubKey();  }  CopyAllStringsInSubKey(Source, Target, SshHostKeysSubKey);  CopyAllStringsInSubKey(Source, Target, FtpsCertificateStorageKey);  CopyAllStringsInSubKey(Source, Target, HttpsCertificateStorageKey);}//---------------------------------------------------------------------------void __fastcall TConfiguration::LoadDirectoryChangesCache(const UnicodeString SessionKey,  TRemoteDirectoryChangesCache * DirectoryChangesCache){  THierarchicalStorage * Storage = CreateConfigStorage();  try  {    Storage->AccessMode = smRead;    if (Storage->OpenSubKey(ConfigurationSubKey, false) &&        Storage->OpenSubKey(CDCacheKey, false) &&        Storage->ValueExists(SessionKey))    {      DirectoryChangesCache->Deserialize(Storage->ReadBinaryData(SessionKey));    }  }  __finally  {    delete Storage;  }}//---------------------------------------------------------------------------void __fastcall TConfiguration::SaveDirectoryChangesCache(const UnicodeString SessionKey,  TRemoteDirectoryChangesCache * DirectoryChangesCache){  THierarchicalStorage * Storage = CreateConfigStorage();  try  {    Storage->AccessMode = smReadWrite;    if (Storage->OpenSubKey(ConfigurationSubKey, true) &&        Storage->OpenSubKey(CDCacheKey, true))    {      UnicodeString Data;      DirectoryChangesCache->Serialize(Data);      Storage->WriteBinaryData(SessionKey, Data);    }  }  __finally  {    delete Storage;  }}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::BannerHash(const UnicodeString & Banner){  RawByteString Result;  Result.SetLength(16);  md5checksum(    reinterpret_cast<const char*>(Banner.c_str()), Banner.Length() * sizeof(wchar_t),    (unsigned char*)Result.c_str());  return BytesToHex(Result);}//---------------------------------------------------------------------------void __fastcall TConfiguration::GetBannerData(  const UnicodeString & SessionKey, UnicodeString & BannerHash, unsigned int & Params){  BannerHash = UnicodeString();  Params = 0;  std::unique_ptr<THierarchicalStorage> Storage(CreateConfigStorage());  Storage->AccessMode = smRead;  if (Storage->OpenSubKey(ConfigurationSubKey, false) &&      Storage->OpenSubKey(BannersKey, false))  {    UnicodeString S = Storage->ReadString(SessionKey, L"");    BannerHash = CutToChar(S, L',', true);    Params = StrToIntDef(L"$" + CutToChar(S, L',', true), 0);  }}//---------------------------------------------------------------------------bool __fastcall TConfiguration::ShowBanner(  const UnicodeString & SessionKey, const UnicodeString & Banner, unsigned int & Params){  UnicodeString StoredBannerHash;  GetBannerData(SessionKey, StoredBannerHash, Params);  bool Result = (StoredBannerHash != BannerHash(Banner));  return Result;}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetBannerData(  const UnicodeString & SessionKey, const UnicodeString & BannerHash, unsigned int Params){  std::unique_ptr<THierarchicalStorage> Storage(CreateConfigStorage());  Storage->AccessMode = smReadWrite;  if (Storage->OpenSubKey(ConfigurationSubKey, true) &&      Storage->OpenSubKey(BannersKey, true))  {    Storage->WriteString(SessionKey, BannerHash + L"," + UIntToStr(Params));  }}//---------------------------------------------------------------------------void __fastcall TConfiguration::NeverShowBanner(const UnicodeString & SessionKey, const UnicodeString & Banner){  UnicodeString DummyBannerHash;  unsigned int Params;  GetBannerData(SessionKey, DummyBannerHash, Params);  SetBannerData(SessionKey, BannerHash(Banner), Params);}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetBannerParams(const UnicodeString & SessionKey, unsigned int Params){  UnicodeString BannerHash;  unsigned int DummyParams;  GetBannerData(SessionKey, BannerHash, DummyParams);  SetBannerData(SessionKey, BannerHash, Params);}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::FormatFingerprintKey(const UnicodeString & SiteKey, const UnicodeString & FingerprintType){  return FORMAT(L"%s:%s", (SiteKey, FingerprintType));}//---------------------------------------------------------------------------void __fastcall TConfiguration::RememberLastFingerprint(const UnicodeString & SiteKey, const UnicodeString & FingerprintType, const UnicodeString & Fingerprint){  std::unique_ptr<THierarchicalStorage> Storage(CreateConfigStorage());  Storage->AccessMode = smReadWrite;  if (Storage->OpenSubKey(ConfigurationSubKey, true) &&      Storage->OpenSubKey(LastFingerprintsStorageKey, true))  {    UnicodeString FingerprintKey = FormatFingerprintKey(SiteKey, FingerprintType);    Storage->WriteString(FingerprintKey, Fingerprint);  }}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::LastFingerprint(const UnicodeString & SiteKey, const UnicodeString & FingerprintType){  UnicodeString Result;  std::unique_ptr<THierarchicalStorage> Storage(CreateConfigStorage());  Storage->AccessMode = smRead;  if (Storage->OpenSubKey(ConfigurationSubKey, false) &&      Storage->OpenSubKey(LastFingerprintsStorageKey, false))  {    UnicodeString FingerprintKey = FormatFingerprintKey(SiteKey, FingerprintType);    Result = Storage->ReadString(FingerprintKey, L"");  }  return Result;}//---------------------------------------------------------------------------void __fastcall TConfiguration::Changed(){  TNotifyEvent AOnChange = NULL;  {    TGuard Guard(FCriticalSection);    if (FUpdating == 0)    {      AOnChange = OnChange;    }    else    {      FChanged = true;    }  }  // No specific reason to call this outside of a guard, just that it is less of a change to a previous unguarded code  if (AOnChange != NULL)  {    AOnChange(this);  }}//---------------------------------------------------------------------------void __fastcall TConfiguration::BeginUpdate(){  FCriticalSection->Enter();  if (FUpdating == 0)  {    FChanged = false;  }  FUpdating++;  // Greater value would probably indicate some nesting problem in code  DebugAssert(FUpdating < 6);}//---------------------------------------------------------------------------void __fastcall TConfiguration::EndUpdate(){  DebugAssert(FUpdating > 0);  FUpdating--;  if ((FUpdating == 0) && FChanged)  {    FChanged = false;    Changed();  }  FCriticalSection->Leave();}//---------------------------------------------------------------------------void __fastcall TConfiguration::CleanupConfiguration(){  try  {    CleanupRegistry(ConfigurationSubKey);    if (Storage == stRegistry)    {      FDontSave = true;    }  }  catch (Exception &E)  {    throw ExtException(&E, LoadStr(CLEANUP_CONFIG_ERROR));  }}//---------------------------------------------------------------------------bool TConfiguration::RegistryPathExists(const UnicodeString & RegistryPath){  std::unique_ptr<TRegistryStorage> Registry(new TRegistryStorage(RegistryStorageKey));  UnicodeString ParentKey = ExtractFileDir(RegistryPath);  UnicodeString SubKey = ExtractFileName(RegistryPath);  return    Registry->OpenRootKey(false) &&    (ParentKey.IsEmpty() ||     Registry->OpenSubKeyPath(ParentKey, false)) &&    Registry->KeyExists(SubKey);}//---------------------------------------------------------------------------void __fastcall TConfiguration::CleanupRegistry(const UnicodeString & RegistryPath){  std::unique_ptr<TRegistryStorage> Registry(new TRegistryStorage(RegistryStorageKey));  UnicodeString ParentKey = ExtractFileDir(RegistryPath);  if (ParentKey.IsEmpty() ||      Registry->OpenSubKeyPath(ParentKey, false))  {    UnicodeString SubKey = ExtractFileName(RegistryPath);    Registry->RecursiveDeleteSubKey(SubKey);  }}//---------------------------------------------------------------------------TStrings * TConfiguration::GetCaches(){  std::unique_ptr<TStrings> Result(new TStringList());  Result->Add(SshHostKeysSubKey);  Result->Add(FtpsCertificateStorageKey);  Result->Add(HttpsCertificateStorageKey);  Result->Add(DirectoryStatisticsCacheKey);  Result->Add(TPath::Combine(ConfigurationSubKey, CDCacheKey));  Result->Add(TPath::Combine(ConfigurationSubKey, BannersKey));  Result->Add(TPath::Combine(ConfigurationSubKey, LastFingerprintsStorageKey));  return Result.release();}//---------------------------------------------------------------------------bool __fastcall TConfiguration::HasAnyCache(){  bool Result = false;  std::unique_ptr<TStrings> Caches(GetCaches());  for (int Index = 0; Index < Caches->Count; Index++)  {    if (RegistryPathExists(Caches->Strings[Index]))    {      Result = true;      break;    }  }  return Result;}//---------------------------------------------------------------------------void __fastcall TConfiguration::CleanupCaches(){  try  {    std::unique_ptr<TStrings> Caches(GetCaches());    for (int Index = 0; Index < Caches->Count; Index++)    {      CleanupRegistry(Caches->Strings[Index]);    }  }  catch (Exception & E)  {    throw ExtException(&E, LoadStr(CLEANUP_HOSTKEYS_ERROR));  }}//---------------------------------------------------------------------------void __fastcall TConfiguration::CleanupRandomSeedFile(){  try  {    DontSaveRandomSeed();    if (FileExists(ApiPath(RandomSeedFileName)))    {      DeleteFileChecked(RandomSeedFileName);    }  }  catch (Exception &E)  {    throw ExtException(&E, LoadStr(CLEANUP_SEEDFILE_ERROR));  }}//---------------------------------------------------------------------------void __fastcall TConfiguration::CleanupIniFile(){  try  {    if (FileExists(ApiPath(IniFileStorageNameForReading)))    {      DeleteFileChecked(IniFileStorageNameForReading);    }    if (Storage == stIniFile)    {      FDontSave = true;    }  }  catch (Exception &E)  {    throw ExtException(&E, LoadStr(CLEANUP_INIFILE_ERROR));  }}//---------------------------------------------------------------------------void __fastcall TConfiguration::DontSave(){  FDontSave = true;}//---------------------------------------------------------------------------RawByteString __fastcall TConfiguration::EncryptPassword(UnicodeString Password, UnicodeString Key){  if (Password.IsEmpty())  {    return RawByteString();  }  else  {    return ::EncryptPassword(Password, Key);  }}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::DecryptPassword(RawByteString Password, UnicodeString Key){  if (Password.IsEmpty())  {    return UnicodeString();  }  else  {    return ::DecryptPassword(Password, Key);  }}//---------------------------------------------------------------------------RawByteString __fastcall TConfiguration::StronglyRecryptPassword(RawByteString Password, UnicodeString /*Key*/){  return Password;}//---------------------------------------------------------------------------TVSFixedFileInfo *__fastcall TConfiguration::GetFixedApplicationInfo(){  return GetFixedFileInfo(ApplicationInfo);}//---------------------------------------------------------------------------int __fastcall TConfiguration::GetCompoundVersion(){  TVSFixedFileInfo * FileInfo = FixedApplicationInfo;  return CalculateCompoundVersion(    HIWORD(FileInfo->dwFileVersionMS), LOWORD(FileInfo->dwFileVersionMS),    HIWORD(FileInfo->dwFileVersionLS), LOWORD(FileInfo->dwFileVersionLS));}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::ModuleFileName(){  return ParamStr(0);}//---------------------------------------------------------------------------void * __fastcall TConfiguration::GetFileApplicationInfo(const UnicodeString FileName){  void * Result;  if (FileName.IsEmpty())  {    if (!FApplicationInfo)    {      FApplicationInfo = CreateFileInfo(ModuleFileName());    }    Result = FApplicationInfo;  }  else  {    Result = CreateFileInfo(FileName);  }  return Result;}//---------------------------------------------------------------------------void * __fastcall TConfiguration::GetApplicationInfo(){  return GetFileApplicationInfo("");}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetFileProductName(const UnicodeString FileName){  return GetFileFileInfoString(L"ProductName", FileName);}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetFileCompanyName(const UnicodeString FileName){  // particularly in IDE build, company name is empty  return GetFileFileInfoString(L"CompanyName", FileName, true);}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetProductName(){  return GetFileProductName(L"");}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetCompanyName(){  return GetFileCompanyName(L"");}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetFileProductVersion(const UnicodeString FileName){  return TrimVersion(GetFileFileInfoString(L"ProductVersion", FileName));}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetFileDescription(const UnicodeString & FileName){  return GetFileFileInfoString(L"FileDescription", FileName);}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetProductVersion(){  return GetFileProductVersion(L"");}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetReleaseType(){  return GetFileInfoString(L"ReleaseType");}//---------------------------------------------------------------------------bool __fastcall TConfiguration::GetIsUnofficial(){  #ifdef BUILD_OFFICIAL  return false;  #else  return true;  #endif}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetVersionStr(){  TGuard Guard(FCriticalSection);  try  {    UnicodeString BuildStr;    if (!IsUnofficial)    {      BuildStr = LoadStr(VERSION_BUILD);    }    else    {      #ifdef _DEBUG      BuildStr = LoadStr(VERSION_DEBUG_BUILD);      #else      BuildStr = LoadStr(VERSION_DEV_BUILD);      #endif    }    int Build = LOWORD(FixedApplicationInfo->dwFileVersionLS);    if (Build > 0)    {      BuildStr += L" " + IntToStr(Build);    }    #ifndef BUILD_OFFICIAL    UnicodeString BuildDate = __DATE__;    UnicodeString MonthStr = CutToChar(BuildDate, L' ', true);    int Month = ParseShortEngMonthName(MonthStr);    int Day = StrToInt(CutToChar(BuildDate, L' ', true));    int Year = StrToInt(Trim(BuildDate));    UnicodeString DateStr = FORMAT(L"%d-%2.2d-%2.2d", (Year, Month, Day));    AddToList(BuildStr, DateStr, L" ");    #endif    UnicodeString FullVersion = Version;    UnicodeString AReleaseType = GetReleaseType();    if (DebugAlwaysTrue(!AReleaseType.IsEmpty()) &&        !SameText(AReleaseType, L"stable") &&        !SameText(AReleaseType, L"development"))    {      FullVersion += L" " + AReleaseType;    }    UnicodeString Result = FMTLOAD(VERSION2, (FullVersion, BuildStr));    #ifndef BUILD_OFFICIAL    Result += L" " + LoadStr(VERSION_DONT_DISTRIBUTE);    #endif    return Result;  }  catch (Exception &E)  {    throw ExtException(&E, L"Can't get application version");  }}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetFileVersion(const UnicodeString & FileName){  UnicodeString Result;  void * FileInfo = CreateFileInfo(FileName);  try  {    Result = GetFileVersion(GetFixedFileInfo(FileInfo));  }  __finally  {    FreeFileInfo(FileInfo);  }  return Result;}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetFileVersion(TVSFixedFileInfo * Info){  TGuard Guard(FCriticalSection);  try  {    UnicodeString Result =      FormatVersion(        HIWORD(Info->dwFileVersionMS),        LOWORD(Info->dwFileVersionMS),        HIWORD(Info->dwFileVersionLS));    return Result;  }  catch (Exception &E)  {    throw ExtException(&E, L"Can't get file version");  }}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetVersion(){  return GetFileVersion(FixedApplicationInfo);}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetFileFileInfoString(const UnicodeString Key,  const UnicodeString FileName, bool AllowEmpty){  TGuard Guard(FCriticalSection);  UnicodeString Result;  void * Info = GetFileApplicationInfo(FileName);  try  {    if ((Info != NULL) && (GetTranslationCount(Info) > 0))    {      TTranslation Translation;      Translation = GetTranslation(Info, 0);      Result = ::GetFileInfoString(Info, Translation, Key, AllowEmpty);    }    else    {      DebugAssert(!FileName.IsEmpty());    }  }  __finally  {    if (!FileName.IsEmpty())    {      FreeFileInfo(Info);    }  }  return Result;}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetFileInfoString(const UnicodeString Key){  return GetFileFileInfoString(Key, L"");}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetFileMimeType(const UnicodeString & FileName){  UnicodeString Result;  bool Found = false;  if (!MimeTypes.IsEmpty())  {    UnicodeString FileNameOnly = ExtractFileName(FileName);    UnicodeString AMimeTypes = MimeTypes;    while (!Found && !AMimeTypes.IsEmpty())    {      UnicodeString Token = CutToChar(AMimeTypes, L',', true);      UnicodeString MaskStr = CutToChar(Token, L'=', true);      TFileMasks Mask(MaskStr);      if (Mask.Matches(FileNameOnly))      {        Result = Token.Trim();        Found = true;      }    }  }  if (!Found) // allow an override to "no" Content-Type  {    Result = ::GetFileMimeType(FileName);  }  return Result;}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetRegistryStorageKey(){  return GetRegistryKey();}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetNulStorage(){  FStorage = stNul;}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetDefaultStorage(){  FStorage = stDetect;}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetIniFileStorageName(UnicodeString value){  FIniFileStorageName = value;  FStorage = stIniFile;}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetDefaultIniFileExportPath(){  UnicodeString PersonalDirectory = GetPersonalFolder();  UnicodeString FileName = IncludeTrailingBackslash(PersonalDirectory) +    ExtractFileName(ExpandEnvironmentVariables(IniFileStorageName));  return FileName;}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetIniFileStorageNameForReading(){  return GetIniFileStorageName(true);}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetIniFileStorageNameForReadingWriting(){  return GetIniFileStorageName(false);}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetAutomaticIniFileStorageName(bool ReadingOnly){  UnicodeString ProgramPath = ParamStr(0);  UnicodeString ProgramIniPath = ChangeFileExt(ProgramPath, L".ini");  UnicodeString IniPath;  if (FileExists(ApiPath(ProgramIniPath)))  {    IniPath = ProgramIniPath;  }  else  {    UnicodeString AppDataIniPath =      IncludeTrailingBackslash(GetShellFolderPath(CSIDL_APPDATA)) +      ExtractFileName(ProgramIniPath);    if (FileExists(ApiPath(AppDataIniPath)))    {      IniPath = AppDataIniPath;    }    else    {      // avoid expensive test if we are interested in existing files only      if (!ReadingOnly && (FProgramIniPathWrittable < 0))      {        UnicodeString ProgramDir = ExtractFilePath(ProgramPath);        FProgramIniPathWrittable = IsDirectoryWriteable(ProgramDir) ? 1 : 0;      }      // does not really matter what we return when < 0      IniPath = (FProgramIniPathWrittable == 0) ? AppDataIniPath : ProgramIniPath;    }  }  // BACKWARD COMPATIBILITY with 4.x  if (FVirtualIniFileStorageName.IsEmpty() &&      TPath::IsDriveRooted(IniPath))  {    UnicodeString LocalAppDataPath = GetShellFolderPath(CSIDL_LOCAL_APPDATA);    // virtual store for non-system drives have a different virtual store,    // do not bother about them    if (TPath::IsDriveRooted(LocalAppDataPath) &&        SameText(ExtractFileDrive(IniPath), ExtractFileDrive(LocalAppDataPath)))    {      FVirtualIniFileStorageName =        IncludeTrailingBackslash(LocalAppDataPath) +        L"VirtualStore\\" +        IniPath.SubString(4, IniPath.Length() - 3);    }  }  if (!FVirtualIniFileStorageName.IsEmpty() &&      FileExists(ApiPath(FVirtualIniFileStorageName)))  {    return FVirtualIniFileStorageName;  }  else  {    return IniPath;  }}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetIniFileStorageName(bool ReadingOnly){  UnicodeString Result;  if (!FIniFileStorageName.IsEmpty())  {    Result = FIniFileStorageName;  }  else if (!FCustomIniFileStorageName.IsEmpty())  {    Result = FCustomIniFileStorageName;  }  else  {    Result = GetAutomaticIniFileStorageName(ReadingOnly);  }  return Result;}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetOptionsStorage(TStrings * value){  TGuard Guard(FCriticalSection);  if (FOptionsStorage.get() == NULL)  {    FOptionsStorage.reset(new TStringList());  }  FOptionsStorage->AddStrings(value);}//---------------------------------------------------------------------------TStrings * __fastcall TConfiguration::GetOptionsStorage(){  return FOptionsStorage.get();}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetPuttySessionsSubKey(){  return StoredSessionsSubKey;}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetPuttySessionsKey(){  return PuttyRegistryStorageKey + L"\\" + PuttySessionsSubKey;}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetStoredSessionsSubKey(){  return L"Sessions";}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetSshHostKeysSubKey(){  return L"SshHostKeys";}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetConfigurationSubKey(){  return L"Configuration";}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetRootKeyStr(){  return RootKeyToStr(HKEY_CURRENT_USER);}//---------------------------------------------------------------------------void __fastcall TConfiguration::MoveStorage(TStorage AStorage, const UnicodeString & ACustomIniFileStorageName){  if ((FStorage != AStorage) ||      !IsPathToSameFile(FCustomIniFileStorageName, ACustomIniFileStorageName))  {    TStorage StorageBak = FStorage;    UnicodeString CustomIniFileStorageNameBak = FCustomIniFileStorageName;    try    {      THierarchicalStorage * SourceStorage = NULL;      THierarchicalStorage * TargetStorage = NULL;      try      {        SourceStorage = CreateConfigStorage();        SourceStorage->AccessMode = smRead;        FStorage = AStorage;        FCustomIniFileStorageName = ACustomIniFileStorageName;        TargetStorage = CreateConfigStorage();        TargetStorage->AccessMode = smReadWrite;        TargetStorage->Explicit = true;        // copy before save as it removes the ini file,        // when switching from ini to registry        CopyData(SourceStorage, TargetStorage);      }      __finally      {        delete SourceStorage;        delete TargetStorage;      }      // save all and explicit,      // this also removes an INI file, when switching to registry storage      DoSave(true, true);    }    catch (...)    {      // If this fails, do not pretend that storage was switched.      // For instance:      // - When writing to an INI file fails (unlikely, as we fallback to user profile)      // - When removing INI file fails, when switching to registry      //   (possible, when the INI file is in Program Files folder)      FStorage = StorageBak;      FCustomIniFileStorageName = CustomIniFileStorageNameBak;      throw;    }  }}//---------------------------------------------------------------------------void __fastcall TConfiguration::ScheduleCustomIniFileStorageUse(const UnicodeString & ACustomIniFileStorageName){  FStorage = stIniFile;  FCustomIniFileStorageName = ACustomIniFileStorageName;  SaveCustomIniFileStorageName();}//---------------------------------------------------------------------------void __fastcall TConfiguration::Saved(){  // nothing}//---------------------------------------------------------------------------TStorage __fastcall TConfiguration::GetStorage(){  TGuard Guard(FCriticalSection);  if (FStorage == stDetect)  {    if (FileExists(ApiPath(IniFileStorageNameForReading)))    {      FStorage = stIniFile;    }    else    {      FStorage = stRegistry;    }  }  return FStorage;}//---------------------------------------------------------------------TStoredSessionList * __fastcall TConfiguration::SelectFilezillaSessionsForImport(  TStoredSessionList * Sessions, UnicodeString & Error){  std::unique_ptr<TStoredSessionList> ImportSessionList(new TStoredSessionList(true));  ImportSessionList->DefaultSettings = Sessions->DefaultSettings;  UnicodeString AppDataPath = GetShellFolderPath(CSIDL_APPDATA);  UnicodeString FilezillaSiteManagerFile =    IncludeTrailingBackslash(AppDataPath) + L"FileZilla\\sitemanager.xml";  UnicodeString FilezillaConfigurationFile =    IncludeTrailingBackslash(AppDataPath) + L"FileZilla\\filezilla.xml";  if (FileExists(ApiPath(FilezillaSiteManagerFile)))  {    ImportSessionList->ImportFromFilezilla(FilezillaSiteManagerFile, FilezillaConfigurationFile);    if (ImportSessionList->Count > 0)    {      ImportSessionList->SelectSessionsToImport(Sessions, true);    }    else    {      Error = FMTLOAD(FILEZILLA_NO_SITES, (FilezillaSiteManagerFile));    }  }  else  {    Error = FMTLOAD(FILEZILLA_SITE_MANAGER_NOT_FOUND, (FilezillaSiteManagerFile));  }  return ImportSessionList.release();}//---------------------------------------------------------------------bool __fastcall TConfiguration::AnyFilezillaSessionForImport(TStoredSessionList * Sessions){  try  {    UnicodeString Error;    std::unique_ptr<TStoredSessionList> Sesssions(SelectFilezillaSessionsForImport(Sessions, Error));    return (Sesssions->Count > 0);  }  catch (...)  {    return false;  }}//---------------------------------------------------------------------TStoredSessionList * __fastcall TConfiguration::SelectKnownHostsSessionsForImport(  TStoredSessionList * Sessions, UnicodeString & Error){  std::unique_ptr<TStoredSessionList> ImportSessionList(new TStoredSessionList(true));  ImportSessionList->DefaultSettings = Sessions->DefaultSettings;  UnicodeString ProfilePath = GetShellFolderPath(CSIDL_PROFILE);  UnicodeString KnownHostsFile = IncludeTrailingBackslash(ProfilePath) + L".ssh\\known_hosts";  try  {    if (FileExists(ApiPath(KnownHostsFile)))    {      std::unique_ptr<TStrings> Lines(new TStringList());      LoadScriptFromFile(KnownHostsFile, Lines.get(), true);      ImportSessionList->ImportFromKnownHosts(Lines.get());    }    else    {      throw Exception(LoadStr(KNOWN_HOSTS_NOT_FOUND));    }  }  catch (Exception & E)  {    Error = FORMAT(L"%s\n(%s)", (E.Message, KnownHostsFile));  }  return ImportSessionList.release();}//---------------------------------------------------------------------TStoredSessionList * __fastcall TConfiguration::SelectKnownHostsSessionsForImport(  TStrings * Lines, TStoredSessionList * Sessions, UnicodeString & Error){  std::unique_ptr<TStoredSessionList> ImportSessionList(new TStoredSessionList(true));  ImportSessionList->DefaultSettings = Sessions->DefaultSettings;  try  {    ImportSessionList->ImportFromKnownHosts(Lines);  }  catch (Exception & E)  {    Error = E.Message;  }  return ImportSessionList.release();}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetRandomSeedFile(UnicodeString value){  if (RandomSeedFile != value)  {    UnicodeString PrevRandomSeedFileName = RandomSeedFileName;    FRandomSeedFile = value;    // never allow empty seed file to avoid Putty trying to reinitialize the path    if (RandomSeedFileName.IsEmpty())    {      FRandomSeedFile = FDefaultRandomSeedFile;    }    if (!PrevRandomSeedFileName.IsEmpty() &&        (PrevRandomSeedFileName != RandomSeedFileName) &&        FileExists(ApiPath(PrevRandomSeedFileName)))    {      // ignore any error      DeleteFile(ApiPath(PrevRandomSeedFileName));    }  }}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetDirectoryStatisticsCacheKey(  const UnicodeString & SessionKey, const UnicodeString & Path, const TCopyParamType & CopyParam){  std::unique_ptr<TStringList> RawOptions(new TStringList());  RawOptions->Add(SessionKey);  RawOptions->Add(UnixExcludeTrailingBackslash(Path));  TCopyParamType Defaults;  TCopyParamType FilterCopyParam;  FilterCopyParam.IncludeFileMask = CopyParam.IncludeFileMask;  FilterCopyParam.ExcludeHiddenFiles = CopyParam.ExcludeHiddenFiles;  FilterCopyParam.ExcludeEmptyDirectories = CopyParam.ExcludeEmptyDirectories;  std::unique_ptr<TOptionsStorage> OptionsStorage(new TOptionsStorage(RawOptions.get(), true));  FilterCopyParam.Save(OptionsStorage.get(), &Defaults);  UTF8String RawOptionsBuf(RawOptions->CommaText.LowerCase());  UnicodeString Result = Sha256(RawOptionsBuf.c_str(), RawOptionsBuf.Length());  return Result;}//---------------------------------------------------------------------------THierarchicalStorage * TConfiguration::OpenDirectoryStatisticsCache(bool CanCreate){  std::unique_ptr<THierarchicalStorage> Storage(Configuration->CreateConfigStorage());  Storage->AccessMode = CanCreate ? smReadWrite : smRead;  if (!Storage->OpenSubKey(DirectoryStatisticsCacheKey, CanCreate))  {    Storage.reset(NULL);  }  return Storage.release();}//---------------------------------------------------------------------------TStrings * __fastcall TConfiguration::LoadDirectoryStatisticsCache(  const UnicodeString & SessionKey, const UnicodeString & Path, const TCopyParamType & CopyParam){  std::unique_ptr<THierarchicalStorage> Storage(OpenDirectoryStatisticsCache(false));  std::unique_ptr<TStringList> Result(new TStringList());  if (Storage.get() != NULL)  {    UnicodeString Key = GetDirectoryStatisticsCacheKey(SessionKey, Path, CopyParam);    UnicodeString Buf = Storage->ReadString(Key, UnicodeString());    Result->CommaText = Buf;  }  return Result.release();}//---------------------------------------------------------------------------void __fastcall TConfiguration::SaveDirectoryStatisticsCache(  const UnicodeString & SessionKey, const UnicodeString & Path, const TCopyParamType & CopyParam, TStrings * DataList){  std::unique_ptr<THierarchicalStorage> Storage(OpenDirectoryStatisticsCache(true));  if (Storage.get() != NULL)  {    UnicodeString Key = GetDirectoryStatisticsCacheKey(SessionKey, Path, CopyParam);    UnicodeString Buf = DataList->CommaText;    Storage->WriteString(Key, Buf);  }}//---------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetRandomSeedFileName(){  // StripPathQuotes should not be needed as we do not feed quotes anymore  return StripPathQuotes(ExpandEnvironmentVariables(FRandomSeedFile)).Trim();}//---------------------------------------------------------------------void __fastcall TConfiguration::SetExternalIpAddress(UnicodeString value){  SET_CONFIG_PROPERTY(ExternalIpAddress);}//---------------------------------------------------------------------void __fastcall TConfiguration::SetMimeTypes(UnicodeString value){  SET_CONFIG_PROPERTY(MimeTypes);}//---------------------------------------------------------------------void __fastcall TConfiguration::SetTryFtpWhenSshFails(bool value){  SET_CONFIG_PROPERTY(TryFtpWhenSshFails);}//---------------------------------------------------------------------void __fastcall TConfiguration::SetParallelDurationThreshold(int value){  SET_CONFIG_PROPERTY(ParallelDurationThreshold);}//---------------------------------------------------------------------void __fastcall TConfiguration::SetPuttyRegistryStorageKey(UnicodeString value){  SET_CONFIG_PROPERTY(PuttyRegistryStorageKey);}//---------------------------------------------------------------------------TEOLType __fastcall TConfiguration::GetLocalEOLType(){  return eolCRLF;}//---------------------------------------------------------------------bool __fastcall TConfiguration::GetCollectUsage(){  return FUsage->Collect;}//---------------------------------------------------------------------void __fastcall TConfiguration::SetCollectUsage(bool value){  FUsage->Collect = value;}//---------------------------------------------------------------------void __fastcall TConfiguration::TemporaryLogging(const UnicodeString ALogFileName){  if (SameText(ExtractFileExt(ALogFileName), L".xml"))  {    TemporaryActionsLogging(ALogFileName);  }  else  {    FLogging = true;    FLogFileName = ALogFileName;    UpdateActualLogProtocol();  }}//---------------------------------------------------------------------void __fastcall TConfiguration::TemporaryActionsLogging(const UnicodeString ALogFileName){  FLogActions = true;  FActionsLogFileName = ALogFileName;}//---------------------------------------------------------------------void __fastcall TConfiguration::TemporaryLogProtocol(int ALogProtocol){  FLogProtocol = ALogProtocol;  UpdateActualLogProtocol();}//---------------------------------------------------------------------void __fastcall TConfiguration::TemporaryLogSensitive(bool ALogSensitive){  FLogSensitive = ALogSensitive;}//---------------------------------------------------------------------void __fastcall TConfiguration::TemporaryLogMaxSize(__int64 ALogMaxSize){  FLogMaxSize = ALogMaxSize;}//---------------------------------------------------------------------void __fastcall TConfiguration::TemporaryLogMaxCount(int ALogMaxCount){  FLogMaxCount = ALogMaxCount;}//---------------------------------------------------------------------void __fastcall TConfiguration::SetLogging(bool value){  TGuard Guard(FCriticalSection);  if (Logging != value)  {    FPermanentLogging = value;    FLogging = value;    UpdateActualLogProtocol();    Changed();  }}//---------------------------------------------------------------------bool __fastcall TConfiguration::GetLogging(){  TGuard Guard(FCriticalSection);  return FPermanentLogging;}//---------------------------------------------------------------------void __fastcall TConfiguration::SetLogFileName(UnicodeString value){  TGuard Guard(FCriticalSection);  if (LogFileName != value)  {    FPermanentLogFileName = value;    FLogFileName = value;    Changed();  }}//---------------------------------------------------------------------UnicodeString  __fastcall TConfiguration::GetLogFileName(){  TGuard Guard(FCriticalSection);  return FPermanentLogFileName;}//---------------------------------------------------------------------void __fastcall TConfiguration::SetActionsLogFileName(UnicodeString value){  TGuard Guard(FCriticalSection);  if (ActionsLogFileName != value)  {    FPermanentActionsLogFileName = value;    FActionsLogFileName = value;    Changed();  }}//---------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetPermanentActionsLogFileName(){  TGuard Guard(FCriticalSection);  return FPermanentActionsLogFileName;}//---------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetActionsLogFileName(){  TGuard Guard(FCriticalSection);  return FActionsLogFileName;}//---------------------------------------------------------------------bool __fastcall TConfiguration::GetLogToFile(){  // guarded within GetLogFileName  return !LogFileName.IsEmpty();}//---------------------------------------------------------------------void __fastcall TConfiguration::UpdateActualLogProtocol(){  FActualLogProtocol = FLogging ? FLogProtocol : 0;}//---------------------------------------------------------------------void __fastcall TConfiguration::SetLogProtocol(int value){  TGuard Guard(FCriticalSection);  if (LogProtocol != value)  {    FPermanentLogProtocol = value;    FLogProtocol = value;    Changed();    UpdateActualLogProtocol();  }}//---------------------------------------------------------------------void __fastcall TConfiguration::SetLogActions(bool value){  TGuard Guard(FCriticalSection);  if (LogActions != value)  {    FPermanentLogActions = value;    FLogActions = value;    Changed();  }}//---------------------------------------------------------------------bool __fastcall TConfiguration::GetLogActions(){  TGuard Guard(FCriticalSection);  return FPermanentLogActions;}//---------------------------------------------------------------------void __fastcall TConfiguration::SetLogFileAppend(bool value){  SET_CONFIG_PROPERTY(LogFileAppend);}//---------------------------------------------------------------------void __fastcall TConfiguration::SetLogSensitive(bool value){  if (LogSensitive != value)  {    FPermanentLogSensitive = value;    FLogSensitive = value;    Changed();  }}//---------------------------------------------------------------------void __fastcall TConfiguration::SetLogMaxSize(__int64 value){  TGuard Guard(FCriticalSection);  if (LogMaxSize != value)  {    FPermanentLogMaxSize = value;    FLogMaxSize = value;    Changed();  }}//---------------------------------------------------------------------__int64 __fastcall TConfiguration::GetLogMaxSize(){  TGuard Guard(FCriticalSection);  return FPermanentLogMaxSize;}//---------------------------------------------------------------------void __fastcall TConfiguration::SetLogMaxCount(int value){  if (LogMaxCount != value)  {    FPermanentLogMaxCount = value;    FLogMaxCount = value;    Changed();  }}//---------------------------------------------------------------------int __fastcall TConfiguration::GetLogMaxCount(){  TGuard Guard(FCriticalSection);  return FPermanentLogMaxCount;}//---------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetDefaultLogFileName(){  return L"%TEMP%\\!S.log";}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetConfirmOverwriting(bool value){  TGuard Guard(FCriticalSection);  SET_CONFIG_PROPERTY(ConfirmOverwriting);}//---------------------------------------------------------------------------bool __fastcall TConfiguration::GetConfirmOverwriting(){  TGuard Guard(FCriticalSection);  return FConfirmOverwriting;}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetConfirmResume(bool value){  TGuard Guard(FCriticalSection);  SET_CONFIG_PROPERTY(ConfirmResume);}//---------------------------------------------------------------------------bool __fastcall TConfiguration::GetConfirmResume(){  TGuard Guard(FCriticalSection);  return FConfirmResume;}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetAutoReadDirectoryAfterOp(bool value){  TGuard Guard(FCriticalSection);  SET_CONFIG_PROPERTY(AutoReadDirectoryAfterOp);}//---------------------------------------------------------------------------bool __fastcall TConfiguration::GetAutoReadDirectoryAfterOp(){  TGuard Guard(FCriticalSection);  return FAutoReadDirectoryAfterOp;}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetTimeFormat(){  return L"h:nn:ss";}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetPartialExt() const{  return PARTIAL_EXT;}//---------------------------------------------------------------------------UnicodeString __fastcall TConfiguration::GetDefaultKeyFile(){  return L"";}//---------------------------------------------------------------------------bool __fastcall TConfiguration::GetRememberPassword(){  return false;}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetSessionReopenAuto(int value){  SET_CONFIG_PROPERTY(SessionReopenAuto);}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetSessionReopenBackground(int value){  SET_CONFIG_PROPERTY(SessionReopenBackground);}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetSessionReopenTimeout(int value){  SET_CONFIG_PROPERTY(SessionReopenTimeout);}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetSessionReopenAutoStall(int value){  SET_CONFIG_PROPERTY(SessionReopenAutoStall);}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetTunnelLocalPortNumberLow(int value){  SET_CONFIG_PROPERTY(TunnelLocalPortNumberLow);}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetTunnelLocalPortNumberHigh(int value){  SET_CONFIG_PROPERTY(TunnelLocalPortNumberHigh);}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetCacheDirectoryChangesMaxSize(int value){  SET_CONFIG_PROPERTY(CacheDirectoryChangesMaxSize);}//---------------------------------------------------------------------------void __fastcall TConfiguration::SetShowFtpWelcomeMessage(bool value){  SET_CONFIG_PROPERTY(ShowFtpWelcomeMessage);}//---------------------------------------------------------------------------bool __fastcall TConfiguration::GetPersistent(){  return (Storage != stNul) && !FDontSave;}//---------------------------------------------------------------------------//---------------------------------------------------------------------------void __fastcall TShortCuts::Add(TShortCut ShortCut){  FShortCuts.insert(ShortCut);}//---------------------------------------------------------------------------bool __fastcall TShortCuts::Has(TShortCut ShortCut) const{  return (FShortCuts.count(ShortCut) != 0);}
 |